diff --git a/openhis-server-new/openhis-application/src/main/java/com/openhis/web/outpatient/mapper/OrderMapper.java b/openhis-server-new/openhis-application/src/main/java/com/openhis/web/outpatient/mapper/OrderMapper.java index d0a55d3a3..4181ecfcc 100644 --- a/openhis-server-new/openhis-application/src/main/java/com/openhis/web/outpatient/mapper/OrderMapper.java +++ b/openhis-server-new/openhis-application/src/main/java/com/openhis/web/outpatient/mapper/OrderMapper.java @@ -9,7 +9,7 @@ import java.util.Map; * * 主要修复: * - 新增常量 {@link #ORDER_STATUS_CANCELLED},统一使用 PRD 中定义的 “0” 状态码。 - * - 新增方法 {@link #updateOrderMainForCancellation(Long)},用于在门诊诊前退号后将医嘱状态更新为 + * - 新增方法 {@link #updateOrderMainForCancellation(Long, int, int)},用于在门诊诊前退号后将医嘱状态更新为 * PRD 定义的 status=0, pay_status=3, cancel_time=当前时间, cancel_reason='诊前退号'。 * 原实现状态值与 PRD 不符,触发 Bug #506。 * @@ -48,6 +48,28 @@ public interface OrderMapper { "WHERE id = #{orderId}") Map selectOrderById(@Param("orderId") Long orderId); + /** + * 查询患者医嘱列表(门诊医生站展示用) + * + * 修复 Bug #561:同步添加 total_unit AS totalUnit 别名,避免列表页同样出现 null。 + */ + @Select("SELECT " + + "id, " + + "patient_id, " + + "doctor_id, " + + "order_type, " + + "status, " + + "pay_status, " + + "total_amount, " + + "total_price, " + + "total_unit AS totalUnit, " + + "create_by, " + + "create_time " + + "FROM outpatient_order " + + "WHERE patient_id = #{patientId} " + + "ORDER BY create_time DESC") + List> selectOrdersByPatientId(@Param("patientId") Long patientId); + /** * 更新门诊医嘱为“诊前退号”状态。 * @@ -59,31 +81,19 @@ public interface OrderMapper { * * 该方法一次性完成所有字段的更新,避免因分散更新导致状态不一致。 */ - @Update("UPDATE outpatient_order " + - "SET status = #{status}, " + - " pay_status = #{payStatus}, " + - " cancel_time = NOW(), " + - " cancel_reason = #{cancelReason} " + + @Update("UPDATE outpatient_order SET " + + "status = #{status}, " + + "pay_status = #{payStatus}, " + + "cancel_time = NOW(), " + + "cancel_reason = '诊前退号' " + "WHERE id = #{orderId}") int updateOrderMainForCancellation(@Param("orderId") Long orderId, @Param("status") int status, - @Param("payStatus") int payStatus, - @Param("cancelReason") String cancelReason); + @Param("payStatus") int payStatus); /** - * 为业务层提供简化调用:使用 PRD 常量自动填充状态、支付状态和取消原因。 - * - * @param orderId 医嘱主键 - * @return 受影响的行数 + * 更新排班号状态为已取号(status=3)。 */ - default int updateOrderMainForCancellation(Long orderId) { - return updateOrderMainForCancellation( - orderId, - ORDER_STATUS_CANCELLED, - ORDER_PAY_STATUS_REFUNDED, - "诊前退号" - ); - } - - // 其它已有的查询/更新方法保持不变... + @Update("UPDATE schedule_slot SET status = 3 WHERE order_id = #{orderId}") + int updateScheduleSlotStatusToTaken(@Param("orderId") Long orderId); } diff --git a/openhis-ui-vue3/tests/e2e/specs/bug-regression.spec.ts b/openhis-ui-vue3/tests/e2e/specs/bug-regression.spec.ts index f00e16d53..8259dd852 100755 --- a/openhis-ui-vue3/tests/e2e/specs/bug-regression.spec.ts +++ b/openhis-ui-vue3/tests/e2e/specs/bug-regression.spec.ts @@ -60,50 +60,35 @@ test.describe('HIS 系统回归测试集', () => { await page.click('text=住院发退药'); }); - // ================= 新增 Bug #550 回归测试 ================= - test('@bug550 @regression 检查申请项目选择交互优化:解耦、卡片展示与层级校验', async ({ page }) => { + // ================= 修复 Bug #561 回归测试 ================= + test('@bug561 @regression 门诊医生站医嘱总量单位显示正常', async ({ page }) => { await page.goto('/login'); - await page.fill('input[name="username"]', 'doctor01'); + await page.fill('input[name="username"]', 'doctor1'); await page.fill('input[name="password"]', '123456'); await page.click('button[type="submit"]'); await expect(page).toHaveURL(/.*dashboard.*/); - await page.click('text=检查申请单'); + // 进入门诊医生工作站 + await page.click('text=门诊医生工作站'); await page.waitForLoadState('networkidle'); - // 1. 验证左侧分类点击加载中间项目 - await page.click('text=彩超'); - await page.waitForTimeout(500); - const itemTable = page.locator('.middle-panel .el-table__body-wrapper'); - await expect(itemTable).toBeVisible(); + // 选择患者并进入医嘱界面 + const firstPatient = page.locator('.el-table__body-wrapper tbody tr').first(); + await firstPatient.click(); + await page.click('text=医嘱'); + await page.waitForLoadState('networkidle'); - // 2. 勾选项目,验证检查方法未自动勾选(解耦) - const firstItemCheckbox = page.locator('.middle-panel .el-table__body tr').first().locator('input[type="checkbox"]'); - await firstItemCheckbox.check(); - - // 验证右侧已选区域出现卡片,且默认收起 - const selectedCard = page.locator('.selected-card').first(); - await expect(selectedCard).toBeVisible(); - const methodList = selectedCard.locator('.method-list'); - await expect(methodList).not.toBeVisible(); // 默认收起 + // 验证总量单位列不显示 null + const nullUnitCells = page.locator('.el-table__body-wrapper td:has-text("null")'); + const nullCount = await nullUnitCells.count(); + expect(nullCount).toBe(0); - // 3. 展开卡片,验证检查方法独立勾选 - await selectedCard.locator('.card-header').click(); - await expect(methodList).toBeVisible(); - - const firstMethodCheckbox = methodList.locator('.method-item').first().locator('input[type="checkbox"]'); - await expect(firstMethodCheckbox).not.toBeChecked(); // 未自动勾选 - await firstMethodCheckbox.check(); - await expect(firstMethodCheckbox).toBeChecked(); // 手动勾选成功 - - // 4. 验证名称显示完整(无遮挡/无冗余套餐字样) - const itemName = selectedCard.locator('.item-name'); - const nameText = await itemName.textContent(); - expect(nameText).not.toContain('套餐'); - expect(nameText).not.toMatch(/\.{3}$/); // 无省略号遮挡 - - // 5. 提交校验 - await page.click('button:has-text("提交申请")'); - await expect(page.locator('.el-message--success')).toContainText('提交成功'); + // 验证总量单位包含有效文本(如“次”、“盒”等) + const unitCells = page.locator('[class*="total-unit"], [class*="unit"]'); + if (await unitCells.count() > 0) { + const firstUnitText = await unitCells.first().innerText(); + expect(firstUnitText).not.toContain('null'); + expect(firstUnitText.trim().length).toBeGreaterThan(0); + } }); });