feat: 报销预审会话状态管理与工作台交互增强
- 新增差旅报销会话状态管理与对话模型重构 - 增强风险观测服务与运行时聊天上下文作用域 - 优化工作台图标资源、助理意图识别与摘要工具 - 完善报销创建视图样式与差旅详情页标准调整交互 - 补充风险观测、运行时聊天与报销端点测试覆盖
This commit is contained in:
@@ -26,6 +26,7 @@ import {
|
||||
buildTravelReceiptMaterialPrompts
|
||||
} from '../src/views/scripts/travelRequestDetailAdviceModel.js'
|
||||
import {
|
||||
buildStandardAdjustmentPayload,
|
||||
filterSubmitterResolvedRiskCards
|
||||
} from '../src/views/scripts/travelRequestDetailStandardAdjustment.js'
|
||||
|
||||
@@ -712,6 +713,7 @@ test('expense detail table has per-item risk explanation column', () => {
|
||||
test('expense detail shows standard-adjusted reimbursable amount separately from receipt amount', () => {
|
||||
assert.match(detailViewTemplate, /v-if="item\.hasStandardAdjustment" class="expense-adjusted-amount"/)
|
||||
assert.match(detailViewTemplate, /class="expense-original-amount"[\s\S]*item\.originalAmountDisplay/)
|
||||
assert.match(detailViewTemplate, /class="expense-reimbursable-label"[\s\S]*职级测算/)
|
||||
assert.match(detailViewTemplate, /class="expense-reimbursable-amount"[\s\S]*item\.reimbursableAmountDisplay/)
|
||||
assert.match(detailViewTemplate, /submitConfirmAmountDisplay/)
|
||||
assert.match(detailViewStyle, /\.expense-original-amount[\s\S]*text-decoration-line: line-through/)
|
||||
@@ -753,6 +755,43 @@ test('expense detail shows standard-adjusted reimbursable amount separately from
|
||||
assert.equal(item.hasStandardAdjustment, true)
|
||||
})
|
||||
|
||||
test('standard adjustment payload carries application days for policy recalculation', async () => {
|
||||
const payload = await buildStandardAdjustmentPayload({
|
||||
warnings: [
|
||||
{
|
||||
id: 'risk-hotel-days-1',
|
||||
itemId: 'expense-item-days-1',
|
||||
title: '住宿超标待说明',
|
||||
risk: '住宿票据金额超过职级标准。'
|
||||
}
|
||||
],
|
||||
expenseItems: [
|
||||
{
|
||||
id: 'expense-item-days-1',
|
||||
itemType: 'hotel_ticket',
|
||||
itemLocation: '北京',
|
||||
itemAmount: 2000
|
||||
}
|
||||
],
|
||||
request: {
|
||||
relatedApplication: { days: '4天' },
|
||||
employeeGrade: 'P4',
|
||||
location: '北京'
|
||||
},
|
||||
calculateTravelReimbursement: async (query) => {
|
||||
assert.equal(query.days, 4)
|
||||
assert.equal(query.location, '北京')
|
||||
assert.equal(query.grade, 'P4')
|
||||
return { hotel_amount: 1800 }
|
||||
}
|
||||
})
|
||||
|
||||
assert.equal(payload.risks.length, 1)
|
||||
assert.equal(payload.risks[0].application_days, 4)
|
||||
assert.equal(payload.risks[0].original_amount, 2000)
|
||||
assert.equal(payload.risks[0].reimbursable_amount, 1800)
|
||||
})
|
||||
|
||||
test('plain reimbursable amount does not mark an item as standard-adjusted during detail rebuild', () => {
|
||||
const item = buildExpenseItemViewModel(
|
||||
{
|
||||
|
||||
@@ -15,6 +15,10 @@ const detailExpenseModelScript = readFileSync(
|
||||
fileURLToPath(new URL('../src/views/scripts/travelRequestDetailExpenseModel.js', import.meta.url)),
|
||||
'utf8'
|
||||
)
|
||||
const confirmDialogComponent = readFileSync(
|
||||
fileURLToPath(new URL('../src/components/shared/ConfirmDialog.vue', import.meta.url)),
|
||||
'utf8'
|
||||
)
|
||||
|
||||
function extractFunction(source, name) {
|
||||
let signatureIndex = source.indexOf(`function ${name}(`)
|
||||
@@ -44,10 +48,12 @@ function extractFunction(source, name) {
|
||||
|
||||
test('detail submit opens a confirmation dialog before calling submit API', () => {
|
||||
assert.match(detailViewTemplate, /<ConfirmDialog[\s\S]*:open="submitConfirmDialogOpen"[\s\S]*:confirm-text="submitConfirmText"[\s\S]*@close="closeSubmitConfirmDialog"[\s\S]*@confirm="confirmSubmitRequest"/)
|
||||
assert.match(detailViewTemplate, /:open="submitConfirmDialogOpen"[\s\S]*size="review"/)
|
||||
assert.match(detailViewTemplate, /cancel-text="返回核对"/)
|
||||
assert.match(detailViewTemplate, /@click="handleSubmit"/)
|
||||
|
||||
assert.match(detailViewScript, /const submitConfirmDialogOpen = ref\(false\)/)
|
||||
assert.match(detailViewTemplate, /v-if="submitBusy" class="expense-recognition-banner submit-progress-banner"/)
|
||||
assert.doesNotMatch(detailViewScript, /preReviewExpenseClaim\(request\.value\.claimId\)/)
|
||||
assert.match(detailViewScript, /const submitActionLabel = computed/)
|
||||
assert.match(detailViewScript, /submitConfirmDialogOpen\.value = true/)
|
||||
@@ -60,28 +66,57 @@ test('detail submit opens a confirmation dialog before calling submit API', () =
|
||||
const confirmSubmitRequest = extractFunction(detailViewScript, 'confirmSubmitRequest')
|
||||
assert.doesNotMatch(handleSubmit, /submitExpenseClaim/)
|
||||
assert.doesNotMatch(handleSubmit, /runAiPreReview\(\)/)
|
||||
assert.match(confirmSubmitRequest, /submitExpenseClaim\(request\.value\.claimId\)/)
|
||||
assert.match(confirmSubmitRequest, /submitConfirmDialogOpen\.value = false[\s\S]*void runSubmitRequest\(claimId, documentNo, isApplication, taskSeq\)/)
|
||||
assert.doesNotMatch(confirmSubmitRequest, /await /)
|
||||
assert.doesNotMatch(confirmSubmitRequest, /submitExpenseClaim/)
|
||||
const runSubmitRequest = extractFunction(detailViewScript, 'runSubmitRequest')
|
||||
assert.match(runSubmitRequest, /submitExpenseClaim\(claimId\)/)
|
||||
assert.match(runSubmitRequest, /taskSeq !== submitTaskSeq/)
|
||||
})
|
||||
|
||||
test('detail submit warns on missing risk explanation and supports standard adjustment', () => {
|
||||
assert.match(detailViewTemplate, /:open="riskOverrideDialogOpen"/)
|
||||
assert.match(detailViewTemplate, /:open="riskOverrideDialogOpen"[\s\S]*size="review"/)
|
||||
assert.match(detailViewTemplate, /异常说明/)
|
||||
assert.match(detailViewTemplate, /按职级标准重算/)
|
||||
assert.match(detailViewTemplate, /保存说明并继续提交/)
|
||||
assert.match(detailViewTemplate, /class="risk-override-guidance"/)
|
||||
assert.match(detailViewTemplate, /goToPreviousSubmitRisk/)
|
||||
assert.match(detailViewTemplate, /goToNextSubmitRisk/)
|
||||
assert.match(detailViewTemplate, /v-model="riskOverrideReasons\[currentSubmitRiskWarning\.id\]"/)
|
||||
assert.doesNotMatch(detailViewTemplate, /v-model="riskOverrideReasons\[currentSubmitRiskWarning\.id\]"/)
|
||||
assert.doesNotMatch(detailViewTemplate, /risk-override-save-btn/)
|
||||
assert.doesNotMatch(detailViewTemplate, /confirmRiskOverrideReasons/)
|
||||
assert.match(detailViewScript, /const submitRiskWarnings = computed/)
|
||||
const handleSubmit = extractFunction(detailViewScript, 'handleSubmit')
|
||||
const confirmSubmitRequest = extractFunction(detailViewScript, 'confirmSubmitRequest')
|
||||
assert.match(handleSubmit, /submitRiskWarnings\.value\.length[\s\S]*openRiskOverrideDialog\(\)/)
|
||||
assert.doesNotMatch(confirmSubmitRequest, /openRiskOverrideDialog/)
|
||||
assert.match(detailViewScript, /function confirmRiskOverrideReasons\(\)/)
|
||||
assert.match(detailViewScript, /updateExpenseClaimItem\(request\.value\.claimId, itemId,[\s\S]*item_note: nextNote/s)
|
||||
assert.doesNotMatch(detailViewScript, /riskOverrideReasons/)
|
||||
assert.doesNotMatch(detailViewScript, /function confirmRiskOverrideReasons\(\)/)
|
||||
assert.match(detailViewScript, /function confirmStandardAdjustment\(\)/)
|
||||
assert.match(detailViewScript, /acceptExpenseClaimStandardAdjustment\(request\.value\.claimId, payload\)/)
|
||||
assert.match(detailViewTemplate, /v-if="standardAdjustmentBusy" class="expense-recognition-banner standard-adjustment-banner"/)
|
||||
assert.match(detailViewScript, /const standardAdjustmentBusy = ref\(false\)/)
|
||||
const confirmStandardAdjustment = extractFunction(detailViewScript, 'confirmStandardAdjustment')
|
||||
assert.match(confirmStandardAdjustment, /const claimId = String\(request\.value\?\.claimId/)
|
||||
assert.match(confirmStandardAdjustment, /riskOverrideDialogOpen\.value = false[\s\S]*standardAdjustmentBusy\.value = true[\s\S]*void runStandardAdjustmentRecalculation\(claimId, taskSeq\)/)
|
||||
assert.doesNotMatch(confirmStandardAdjustment, /await /)
|
||||
assert.doesNotMatch(confirmStandardAdjustment, /acceptExpenseClaimStandardAdjustment/)
|
||||
const runStandardAdjustmentRecalculation = extractFunction(detailViewScript, 'runStandardAdjustmentRecalculation')
|
||||
assert.match(runStandardAdjustmentRecalculation, /acceptExpenseClaimStandardAdjustment\(claimId, payload\)/)
|
||||
assert.doesNotMatch(runStandardAdjustmentRecalculation, /submitConfirmDialogOpen\.value = true/)
|
||||
const actionBusyStart = detailViewScript.indexOf('const actionBusy = computed')
|
||||
const actionBusyEnd = detailViewScript.indexOf('const profile = computed', actionBusyStart)
|
||||
assert.ok(actionBusyStart > -1 && actionBusyEnd > actionBusyStart)
|
||||
assert.doesNotMatch(detailViewScript.slice(actionBusyStart, actionBusyEnd), /standardAdjustmentBusy/)
|
||||
assert.match(detailExpenseModelScript, /STANDARD_ADJUSTMENT_RISK_SOURCE = 'reimbursement_standard_adjustment'/)
|
||||
assert.match(detailViewTemplate, /异常说明/)
|
||||
})
|
||||
|
||||
test('submit confirm dialog is constrained for laptop viewport height', () => {
|
||||
assert.match(confirmDialogComponent, /max-height:\s*calc\(100vh - 40px\)/)
|
||||
assert.match(confirmDialogComponent, /max-height:\s*calc\(100dvh - 40px\)/)
|
||||
assert.match(confirmDialogComponent, /\.shared-confirm-body \{[\s\S]*min-height: 0;[\s\S]*overflow-y: auto;/)
|
||||
assert.match(confirmDialogComponent, /\.shared-confirm-actions \{[\s\S]*flex: 0 0 auto;/)
|
||||
assert.match(confirmDialogComponent, /\.shared-confirm-card--review \{[\s\S]*width: min\(560px, calc\(100vw - 40px\)\);/)
|
||||
assert.match(confirmDialogComponent, /@media \(max-width: 720px\) \{[\s\S]*max-height: calc\(100dvh - 28px\)/)
|
||||
})
|
||||
|
||||
test('detail header and fallback progress use reimbursement wording', () => {
|
||||
|
||||
Reference in New Issue
Block a user