order-status.spec.ts 17 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454
  1. /**
  2. * 订单状态流转 E2E 测试
  3. *
  4. * 测试范围:
  5. * - 激活草稿订单
  6. * - 关闭进行中订单
  7. * - 状态限制验证
  8. * - 状态流转后的 Toast 消息验证
  9. *
  10. * @packageDocumentation
  11. */
  12. import { test, expect, Page } from '../../utils/test-setup';
  13. import { ORDER_STATUS, type OrderStatus, ORDER_STATUS_LABELS } from '../../pages/admin/order-management.page';
  14. /**
  15. * 辅助函数:在创建订单对话框中选择残疾人
  16. *
  17. * @param page - Playwright Page 对象
  18. * @returns 是否成功选择了残疾人
  19. */
  20. async function selectDisabledPersonForOrder(page: Page): Promise<boolean> {
  21. const selectPersonButton = page.getByRole('button', { name: '选择残疾人' });
  22. await selectPersonButton.click();
  23. await page.waitForSelector('[role="dialog"]', { state: 'visible', timeout: 5000 });
  24. let hasData = false;
  25. try {
  26. const firstCheckbox = page.locator('table tbody tr').first().locator('input[type="checkbox"]').first();
  27. await firstCheckbox.waitFor({ state: 'visible', timeout: 3000 });
  28. await firstCheckbox.check();
  29. console.debug('✓ 已选择第一个残疾人');
  30. hasData = true;
  31. } catch (error) {
  32. console.debug('没有可用的残疾人数据');
  33. hasData = false;
  34. }
  35. if (hasData) {
  36. const confirmButton = page.getByRole('button', { name: /^(确定|确认|选择)$/ });
  37. await confirmButton.click().catch(() => {
  38. console.debug('没有找到确认按钮,尝试关闭对话框');
  39. page.keyboard.press('Escape');
  40. });
  41. } else {
  42. await page.keyboard.press('Escape').catch(() => {
  43. console.debug('无法关闭对话框,可能已经自动关闭');
  44. });
  45. }
  46. // 等待对话框关闭
  47. await page.waitForSelector('[role="dialog"]', { state: 'hidden', timeout: 3000 })
  48. .catch(() => console.debug('对话框已关闭或不存在'));
  49. return hasData;
  50. }
  51. /**
  52. * 辅助函数:创建测试订单
  53. *
  54. * @param page - Playwright Page 对象
  55. * @param orderName - 订单名称
  56. * @param status - 订单状态(默认为草稿)
  57. * @returns 是否成功创建
  58. */
  59. async function createTestOrder(page: Page, orderName: string, status?: OrderStatus): Promise<boolean> {
  60. // 打开创建对话框
  61. const addOrderButton = page.getByTestId('create-order-button');
  62. await addOrderButton.click();
  63. await page.waitForSelector('[role="dialog"]', { state: 'visible', timeout: 5000 });
  64. // 填写必填字段
  65. await page.getByLabel(/订单名称|名称/).fill(orderName);
  66. await page.getByLabel(/预计开始日期|开始日期/).fill('2025-01-15');
  67. // 选择残疾人(必填)
  68. const hasDisabledPerson = await selectDisabledPersonForOrder(page);
  69. if (!hasDisabledPerson) {
  70. await page.keyboard.press('Escape');
  71. return false;
  72. }
  73. // 如果指定了非草稿状态,需要在编辑模式下修改(这里先创建草稿)
  74. // 提交表单
  75. const submitButton = page.getByRole('button', { name: /^(创建|更新|保存)$/ });
  76. await submitButton.click();
  77. // 等待网络请求完成
  78. try {
  79. await page.waitForLoadState('networkidle', { timeout: 5000 });
  80. } catch {
  81. console.debug('networkidle 超时,继续检查 Toast 消息');
  82. }
  83. // 等待 Toast 消息出现(成功或错误)
  84. const successToast = page.locator('[data-sonner-toast][data-type="success"]');
  85. await successToast.waitFor({ state: 'visible', timeout: 3000 })
  86. .catch(() => console.debug('未检测到成功 Toast,继续检查'));
  87. // 检查是否有成功消息
  88. const hasSuccess = await successToast.count() > 0;
  89. // 等待对话框关闭
  90. const dialog = page.locator('[role="dialog"]');
  91. await dialog.waitFor({ state: 'hidden', timeout: 5000 }).catch(() => {
  92. console.debug('对话框关闭超时,可能已经关闭');
  93. });
  94. return hasSuccess;
  95. }
  96. test.describe('订单状态流转测试', () => {
  97. test.describe('激活草稿订单', () => {
  98. let draftOrderName: string;
  99. test.beforeEach(async ({ adminLoginPage, orderManagementPage, testUsers }) => {
  100. // 以管理员身份登录后台
  101. await adminLoginPage.goto();
  102. await adminLoginPage.login(testUsers.admin.username, testUsers.admin.password);
  103. await adminLoginPage.expectLoginSuccess();
  104. await orderManagementPage.goto();
  105. // 创建一个草稿状态的测试订单
  106. const timestamp = Date.now();
  107. draftOrderName = `激活测试_${timestamp}`;
  108. const created = await createTestOrder(orderManagementPage.page, draftOrderName);
  109. if (!created) {
  110. test.skip(true, '没有残疾人数据,无法创建测试订单');
  111. }
  112. // 验证订单出现在列表中
  113. await expect(async () => {
  114. const exists = await orderManagementPage.orderExists(draftOrderName);
  115. expect(exists).toBe(true);
  116. }).toPass({ timeout: 5000 });
  117. // 验证初始状态为草稿
  118. const initialStatus = await orderManagementPage.getOrderStatus(draftOrderName);
  119. console.debug(`✓ 订单初始状态: ${initialStatus ? ORDER_STATUS_LABELS[initialStatus] : '未知'}`);
  120. });
  121. test('应该成功激活草稿订单', async ({ orderManagementPage }) => {
  122. // 执行激活操作
  123. const success = await orderManagementPage.activateOrder(draftOrderName);
  124. // 验证激活成功
  125. expect(success).toBe(true);
  126. // 验证状态从草稿变为进行中
  127. await expect(async () => {
  128. const currentStatus = await orderManagementPage.getOrderStatus(draftOrderName);
  129. expect(currentStatus).toBe(ORDER_STATUS.IN_PROGRESS);
  130. }).toPass({ timeout: 5000 });
  131. });
  132. test('激活后应该显示成功提示', async ({ orderManagementPage }) => {
  133. await orderManagementPage.activateOrder(draftOrderName);
  134. // 验证 Toast 成功消息
  135. const successToast = orderManagementPage.page.locator('[data-sonner-toast][data-type="success"]');
  136. await expect(successToast).toBeVisible();
  137. const message = await successToast.textContent();
  138. expect(message).toBeDefined();
  139. expect(message?.length).toBeGreaterThan(0);
  140. // 验证消息包含激活相关的关键词(更精确的正则)
  141. expect(message).toMatch(/激活.*成功|已激活/);
  142. });
  143. test('激活确认对话框应该正确显示', async ({ orderManagementPage, page }) => {
  144. await orderManagementPage.openActivateDialog(draftOrderName);
  145. // 验证对话框可见
  146. const dialog = page.locator('[role="alertdialog"]');
  147. await expect(dialog).toBeVisible();
  148. // 验证确认按钮存在(支持多种可能的按钮名称)
  149. const confirmButton = page.locator('[role="alertdialog"]').getByRole('button', {
  150. name: /^(确认激活|激活|确定|确认)$/
  151. });
  152. await expect(confirmButton).toBeVisible();
  153. // 验证取消按钮存在
  154. const cancelButton = page.locator('[role="alertdialog"]').getByRole('button', { name: '取消' });
  155. await expect(cancelButton).toBeVisible();
  156. // 取消激活以清理
  157. await page.keyboard.press('Escape');
  158. });
  159. });
  160. test.describe('关闭进行中订单', () => {
  161. let inProgressOrderName: string;
  162. test.beforeEach(async ({ adminLoginPage, orderManagementPage, testUsers }) => {
  163. // 以管理员身份登录后台
  164. await adminLoginPage.goto();
  165. await adminLoginPage.login(testUsers.admin.username, testUsers.admin.password);
  166. await adminLoginPage.expectLoginSuccess();
  167. await orderManagementPage.goto();
  168. // 创建一个草稿订单然后激活为进行中状态
  169. const timestamp = Date.now();
  170. inProgressOrderName = `关闭测试_${timestamp}`;
  171. const created = await createTestOrder(orderManagementPage.page, inProgressOrderName);
  172. if (!created) {
  173. test.skip(true, '没有残疾人数据,无法创建测试订单');
  174. }
  175. // 验证订单出现在列表中
  176. await expect(async () => {
  177. const exists = await orderManagementPage.orderExists(inProgressOrderName);
  178. expect(exists).toBe(true);
  179. }).toPass({ timeout: 5000 });
  180. // 激活订单到进行中状态
  181. await orderManagementPage.activateOrder(inProgressOrderName);
  182. // 验证状态变为进行中
  183. await expect(async () => {
  184. const currentStatus = await orderManagementPage.getOrderStatus(inProgressOrderName);
  185. expect(currentStatus).toBe(ORDER_STATUS.IN_PROGRESS);
  186. }).toPass({ timeout: 5000 });
  187. console.debug(`✓ 订单已激活为进行中状态`);
  188. });
  189. test('应该成功关闭进行中订单', async ({ orderManagementPage }) => {
  190. // 执行关闭操作
  191. const success = await orderManagementPage.closeOrder(inProgressOrderName);
  192. // 验证关闭成功
  193. expect(success).toBe(true);
  194. // 验证状态从进行中变为已完成
  195. await expect(async () => {
  196. const currentStatus = await orderManagementPage.getOrderStatus(inProgressOrderName);
  197. expect(currentStatus).toBe(ORDER_STATUS.COMPLETED);
  198. }).toPass({ timeout: 5000 });
  199. });
  200. test('关闭后应该显示成功提示', async ({ orderManagementPage }) => {
  201. await orderManagementPage.closeOrder(inProgressOrderName);
  202. // 验证 Toast 成功消息
  203. const successToast = orderManagementPage.page.locator('[data-sonner-toast][data-type="success"]');
  204. await expect(successToast).toBeVisible();
  205. const message = await successToast.textContent();
  206. expect(message).toBeDefined();
  207. expect(message?.length).toBeGreaterThan(0);
  208. // 验证消息包含关闭相关的关键词(更精确的正则)
  209. expect(message).toMatch(/关闭.*成功|已完成/);
  210. });
  211. test('关闭确认对话框应该正确显示', async ({ orderManagementPage, page }) => {
  212. await orderManagementPage.openCloseDialog(inProgressOrderName);
  213. // 验证对话框可见
  214. const dialog = page.locator('[role="alertdialog"]');
  215. await expect(dialog).toBeVisible();
  216. // 验证确认按钮存在(支持多种可能的按钮名称)
  217. const confirmButton = page.locator('[role="alertdialog"]').getByRole('button', {
  218. name: /^(确认关闭|关闭|确定|确认)$/
  219. });
  220. await expect(confirmButton).toBeVisible();
  221. // 验证取消按钮存在
  222. const cancelButton = page.locator('[role="alertdialog"]').getByRole('button', { name: '取消' });
  223. await expect(cancelButton).toBeVisible();
  224. // 取消关闭以清理
  225. await page.keyboard.press('Escape');
  226. });
  227. });
  228. test.describe('状态限制验证', () => {
  229. let shouldSkipTests = false;
  230. // 创建测试订单的辅助函数
  231. async function setupTestOrder(page: Page, orderName: string): Promise<boolean> {
  232. const created = await createTestOrder(page, orderName);
  233. if (!created) {
  234. shouldSkipTests = true;
  235. return false;
  236. }
  237. // 等待订单出现在列表中(使用条件等待而非硬编码超时)
  238. try {
  239. await page.waitForFunction(
  240. (name) => {
  241. const table = document.querySelector('table tbody tr');
  242. if (!table) return false;
  243. const rows = Array.from(document.querySelectorAll('table tbody tr'));
  244. return rows.some(row => row.textContent?.includes(name));
  245. },
  246. orderName,
  247. { timeout: 5000 }
  248. );
  249. } catch {
  250. console.debug(`订单 ${orderName} 未在列表中找到`);
  251. }
  252. return true;
  253. }
  254. test('不能激活非草稿状态的订单', async ({ adminLoginPage, orderManagementPage, testUsers }) => {
  255. if (shouldSkipTests) {
  256. test.skip(true, '测试数据不可用');
  257. return;
  258. }
  259. await adminLoginPage.goto();
  260. await adminLoginPage.login(testUsers.admin.username, testUsers.admin.password);
  261. await adminLoginPage.expectLoginSuccess();
  262. await orderManagementPage.goto();
  263. // 测试 IN_PROGRESS 状态不能激活
  264. const inProgressOrderName = `限制测试_进行中_${Date.now()}`;
  265. if (await setupTestOrder(orderManagementPage.page, inProgressOrderName)) {
  266. await orderManagementPage.activateOrder(inProgressOrderName);
  267. const isActivateEnabled = await orderManagementPage.checkActivateButtonEnabled(inProgressOrderName);
  268. expect(isActivateEnabled).toBe(false);
  269. console.debug(`✓ IN_PROGRESS 状态的订单不能激活`);
  270. }
  271. // 测试 COMPLETED 状态不能激活
  272. const completedOrderName = `限制测试_已完成_${Date.now()}`;
  273. if (await setupTestOrder(orderManagementPage.page, completedOrderName)) {
  274. await orderManagementPage.activateOrder(completedOrderName);
  275. await orderManagementPage.closeOrder(completedOrderName);
  276. const isActivateEnabled = await orderManagementPage.checkActivateButtonEnabled(completedOrderName);
  277. expect(isActivateEnabled).toBe(false);
  278. console.debug(`✓ COMPLETED 状态的订单不能激活`);
  279. }
  280. });
  281. test('不能关闭非进行中状态的订单', async ({ adminLoginPage, orderManagementPage, testUsers }) => {
  282. if (shouldSkipTests) {
  283. test.skip(true, '测试数据不可用');
  284. return;
  285. }
  286. await adminLoginPage.goto();
  287. await adminLoginPage.login(testUsers.admin.username, testUsers.admin.password);
  288. await adminLoginPage.expectLoginSuccess();
  289. await orderManagementPage.goto();
  290. // 测试 DRAFT 状态不能关闭
  291. const draftOrderName = `限制测试_草稿_${Date.now()}`;
  292. if (await setupTestOrder(orderManagementPage.page, draftOrderName)) {
  293. const isCloseEnabled = await orderManagementPage.checkCloseButtonEnabled(draftOrderName);
  294. expect(isCloseEnabled).toBe(false);
  295. console.debug(`✓ DRAFT 状态的订单不能关闭`);
  296. }
  297. // 测试 COMPLETED 状态不能关闭
  298. const completedOrderName = `限制测试_已完成_2_${Date.now()}`;
  299. if (await setupTestOrder(orderManagementPage.page, completedOrderName)) {
  300. await orderManagementPage.activateOrder(completedOrderName);
  301. await orderManagementPage.closeOrder(completedOrderName);
  302. const isCloseEnabled = await orderManagementPage.checkCloseButtonEnabled(completedOrderName);
  303. expect(isCloseEnabled).toBe(false);
  304. console.debug(`✓ COMPLETED 状态的订单不能关闭`);
  305. }
  306. });
  307. test('已完成状态不能再次操作', async ({ adminLoginPage, orderManagementPage, testUsers }) => {
  308. if (shouldSkipTests) {
  309. test.skip(true, '测试数据不可用');
  310. return;
  311. }
  312. await adminLoginPage.goto();
  313. await adminLoginPage.login(testUsers.admin.username, testUsers.admin.password);
  314. await adminLoginPage.expectLoginSuccess();
  315. await orderManagementPage.goto();
  316. const orderName = `限制测试_已完成操作_${Date.now()}`;
  317. if (!await setupTestOrder(orderManagementPage.page, orderName)) {
  318. test.skip(true, '测试数据不可用');
  319. return;
  320. }
  321. // 操作到已完成状态
  322. await orderManagementPage.activateOrder(orderName);
  323. await orderManagementPage.closeOrder(orderName);
  324. // 验证订单状态为已完成
  325. await expect(async () => {
  326. const status = await orderManagementPage.getOrderStatus(orderName);
  327. expect(status).toBe(ORDER_STATUS.COMPLETED);
  328. }).toPass({ timeout: 5000 });
  329. // 已完成状态的订单不能激活也不能关闭
  330. const isActivateEnabled = await orderManagementPage.checkActivateButtonEnabled(orderName);
  331. const isCloseEnabled = await orderManagementPage.checkCloseButtonEnabled(orderName);
  332. expect(isActivateEnabled).toBe(false);
  333. expect(isCloseEnabled).toBe(false);
  334. console.debug(`✓ 已完成状态的订单不能激活或关闭`);
  335. });
  336. });
  337. test.describe('完整状态流转流程', () => {
  338. test('应该支持草稿→进行中→已完成的完整流转', async ({ adminLoginPage, orderManagementPage, testUsers }) => {
  339. // 以管理员身份登录后台
  340. await adminLoginPage.goto();
  341. await adminLoginPage.login(testUsers.admin.username, testUsers.admin.password);
  342. await adminLoginPage.expectLoginSuccess();
  343. await orderManagementPage.goto();
  344. // 创建草稿订单
  345. const timestamp = Date.now();
  346. const orderName = `完整流转测试_${timestamp}`;
  347. const created = await createTestOrder(orderManagementPage.page, orderName);
  348. if (!created) {
  349. test.skip(true, '没有残疾人数据,无法创建测试订单');
  350. }
  351. // 验证订单出现在列表中
  352. await expect(async () => {
  353. const exists = await orderManagementPage.orderExists(orderName);
  354. expect(exists).toBe(true);
  355. }).toPass({ timeout: 5000 });
  356. // 验证初始状态为草稿
  357. let currentStatus = await orderManagementPage.getOrderStatus(orderName);
  358. expect(currentStatus).toBe(ORDER_STATUS.DRAFT);
  359. console.debug(`✓ 初始状态: ${ORDER_STATUS_LABELS[ORDER_STATUS.DRAFT]}`);
  360. // 激活订单
  361. await orderManagementPage.activateOrder(orderName);
  362. currentStatus = await orderManagementPage.getOrderStatus(orderName);
  363. expect(currentStatus).toBe(ORDER_STATUS.IN_PROGRESS);
  364. console.debug(`✓ 激活后状态: ${ORDER_STATUS_LABELS[ORDER_STATUS.IN_PROGRESS]}`);
  365. // 关闭订单
  366. await orderManagementPage.closeOrder(orderName);
  367. currentStatus = await orderManagementPage.getOrderStatus(orderName);
  368. expect(currentStatus).toBe(ORDER_STATUS.COMPLETED);
  369. console.debug(`✓ 关闭后状态: ${ORDER_STATUS_LABELS[ORDER_STATUS.COMPLETED]}`);
  370. });
  371. });
  372. });