2
0

order-delete.spec.ts 19 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500
  1. /**
  2. * 删除订单 E2E 测试
  3. *
  4. * 测试范围:
  5. * - 删除草稿状态订单
  6. * - 删除有关联人员的订单
  7. * - 取消删除操作
  8. * - 删除后列表更新验证
  9. * - Toast 消息验证
  10. *
  11. * @packageDocumentation
  12. */
  13. import { TIMEOUTS } from '../../utils/timeouts';
  14. import { test, expect, Page } from '../../utils/test-setup';
  15. /**
  16. * 辅助函数:获取订单列表中第一个订单的名称
  17. *
  18. * @param page - Playwright Page 对象
  19. * @returns 第一个订单的名称,如果没有则返回 null
  20. */
  21. async function getFirstOrderName(page: Page): Promise<string | null> {
  22. const table = page.locator('table tbody tr');
  23. // 等待表格数据加载完成(跳过"加载中"等占位符文本)
  24. await page.waitForTimeout(TIMEOUTS.LONG);
  25. const count = await table.count();
  26. if (count === 0) {
  27. return null;
  28. }
  29. // 查找第一个有效的订单行(排除"加载中"等占位符)
  30. for (let i = 0; i < count; i++) {
  31. const row = table.nth(i);
  32. const nameCell = row.locator('td').first();
  33. const name = await nameCell.textContent();
  34. const trimmedName = name?.trim() || '';
  35. // 跳过占位符文本
  36. if (trimmedName && trimmedName !== '加载中...' && trimmedName !== '暂无数据' && !trimmedName.includes('加载')) {
  37. return trimmedName;
  38. }
  39. }
  40. return null;
  41. }
  42. /**
  43. * 辅助函数:在创建订单对话框中选择残疾人
  44. *
  45. * @param page - Playwright Page 对象
  46. * @returns 是否成功选择了残疾人
  47. */
  48. async function selectDisabledPersonForOrder(page: Page): Promise<boolean> {
  49. const selectPersonButton = page.getByRole('button', { name: '选择残疾人' });
  50. await selectPersonButton.click();
  51. await page.waitForSelector('[role="dialog"]', { state: 'visible', timeout: TIMEOUTS.DIALOG });
  52. let hasData = false;
  53. try {
  54. const firstCheckbox = page.locator('table tbody tr').first().locator('input[type="checkbox"]').first();
  55. await firstCheckbox.waitFor({ state: 'visible', timeout: TIMEOUTS.ELEMENT_VISIBLE_SHORT });
  56. await firstCheckbox.check();
  57. console.debug('✓ 已选择第一个残疾人');
  58. hasData = true;
  59. } catch (_error) {
  60. console.debug('没有可用的残疾人数据');
  61. hasData = false;
  62. }
  63. if (hasData) {
  64. const confirmButton = page.getByRole('button', { name: /^(确定|确认|选择)$/ });
  65. await confirmButton.click().catch(() => {
  66. console.debug('没有找到确认按钮,尝试关闭对话框');
  67. page.keyboard.press('Escape');
  68. });
  69. } else {
  70. await page.keyboard.press('Escape').catch(() => {
  71. console.debug('无法关闭对话框,可能已经自动关闭');
  72. });
  73. }
  74. await page.waitForTimeout(TIMEOUTS.MEDIUM);
  75. return hasData;
  76. }
  77. /**
  78. * 辅助函数:创建测试订单
  79. *
  80. * @param page - Playwright Page 对象
  81. * @param orderName - 订单名称
  82. * @returns 是否成功创建
  83. */
  84. async function createTestOrder(page: Page, orderName: string): Promise<boolean> {
  85. // 打开创建对话框
  86. const addOrderButton = page.getByTestId('create-order-button');
  87. await addOrderButton.click();
  88. await page.waitForSelector('[role="dialog"]', { state: 'visible', timeout: TIMEOUTS.DIALOG });
  89. // 填写必填字段
  90. await page.getByLabel(/订单名称|名称/).fill(orderName);
  91. await page.getByLabel(/预计开始日期|开始日期/).fill('2025-01-15');
  92. // 选择残疾人(必填)
  93. const hasDisabledPerson = await selectDisabledPersonForOrder(page);
  94. if (!hasDisabledPerson) {
  95. await page.keyboard.press('Escape');
  96. return false;
  97. }
  98. // 提交表单
  99. const submitButton = page.getByRole('button', { name: /^(创建|更新|保存)$/ });
  100. await submitButton.click();
  101. // 等待网络请求完成
  102. try {
  103. await page.waitForLoadState('networkidle', { timeout: TIMEOUTS.DIALOG });
  104. } catch {
  105. console.debug('networkidle 超时,继续检查 Toast 消息');
  106. }
  107. await page.waitForTimeout(TIMEOUTS.VERY_LONG);
  108. // 检查成功消息
  109. const successToast = page.locator('[data-sonner-toast][data-type="success"]');
  110. const hasSuccess = await successToast.count() > 0;
  111. // 等待对话框关闭
  112. const dialog = page.locator('[role="dialog"]');
  113. await dialog.waitFor({ state: 'hidden', timeout: TIMEOUTS.DIALOG }).catch(() => {
  114. console.debug('对话框关闭超时,可能已经关闭');
  115. });
  116. return hasSuccess;
  117. }
  118. test.describe('删除订单测试', () => {
  119. let testOrderName: string;
  120. test.beforeEach(async ({ adminLoginPage, orderManagementPage, testUsers }) => {
  121. // 以管理员身份登录后台
  122. await adminLoginPage.goto();
  123. await adminLoginPage.login(testUsers.admin.username, testUsers.admin.password);
  124. await adminLoginPage.expectLoginSuccess();
  125. await orderManagementPage.goto();
  126. // 尝试使用现有订单
  127. const existingOrder = await getFirstOrderName(orderManagementPage.page);
  128. if (existingOrder) {
  129. testOrderName = existingOrder;
  130. console.debug(`✓ 使用现有订单: ${testOrderName}`);
  131. } else {
  132. // 如果没有现有订单,创建一个
  133. const timestamp = Date.now();
  134. testOrderName = `删除测试_${timestamp}`;
  135. const created = await createTestOrder(orderManagementPage.page, testOrderName);
  136. if (!created) {
  137. // 没有残疾人数据时,跳过测试
  138. test.skip(true, '没有残疾人数据,无法创建测试订单');
  139. }
  140. // 验证订单出现在列表中
  141. await expect(async () => {
  142. const exists = await orderManagementPage.orderExists(testOrderName);
  143. expect(exists).toBe(true);
  144. }).toPass({ timeout: TIMEOUTS.DIALOG });
  145. }
  146. });
  147. test.describe('删除草稿状态订单', () => {
  148. test('应该成功删除草稿订单', async ({ orderManagementPage }) => {
  149. // 获取或创建一个订单用于测试
  150. const existingOrder = await getFirstOrderName(orderManagementPage.page);
  151. const orderToDelete = existingOrder || testOrderName;
  152. // 打开删除确认对话框
  153. await orderManagementPage.openDeleteDialog(orderToDelete);
  154. // 确认删除
  155. await orderManagementPage.confirmDelete();
  156. // 验证订单不再存在
  157. await expect(async () => {
  158. const exists = await orderManagementPage.orderExists(orderToDelete);
  159. expect(exists).toBe(false);
  160. }).toPass({ timeout: TIMEOUTS.DIALOG });
  161. });
  162. test('应该在删除后显示成功提示', async ({ orderManagementPage }) => {
  163. // 获取或创建一个订单用于测试
  164. const existingOrder = await getFirstOrderName(orderManagementPage.page);
  165. const orderToDelete = existingOrder || testOrderName;
  166. await orderManagementPage.deleteOrder(orderToDelete);
  167. // 验证 Toast 成功消息 - 检查更具体的消息内容
  168. const successToast = orderManagementPage.page.locator('[data-sonner-toast][data-type="success"]');
  169. await expect(successToast).toBeVisible();
  170. const message = await successToast.textContent();
  171. expect(message).toBeDefined();
  172. expect(message?.length).toBeGreaterThan(0);
  173. // 验证消息包含删除相关的关键词
  174. expect(message).toMatch(/删除|成功|已删除/);
  175. // 验证订单被删除
  176. const exists = await orderManagementPage.orderExists(orderToDelete);
  177. expect(exists).toBe(false);
  178. });
  179. test('删除确认对话框应该正确显示', async ({ orderManagementPage, page }) => {
  180. // 获取或创建一个订单用于测试(此测试不实际删除,只打开对话框)
  181. const existingOrder = await getFirstOrderName(orderManagementPage.page);
  182. const orderToTest = existingOrder || testOrderName;
  183. await orderManagementPage.openDeleteDialog(orderToTest);
  184. // 验证对话框可见
  185. const dialog = page.locator('[role="alertdialog"]');
  186. await expect(dialog).toBeVisible();
  187. // 验证确认按钮存在(支持多种可能的按钮名称)
  188. const confirmButton = page.locator('[role="alertdialog"]').getByRole('button', {
  189. name: /^(确认删除|删除|确定|确认)$/
  190. });
  191. await expect(confirmButton).toBeVisible();
  192. // 验证取消按钮存在
  193. const cancelButton = page.locator('[role="alertdialog"]').getByRole('button', { name: '取消' });
  194. await expect(cancelButton).toBeVisible();
  195. // 取消删除以清理
  196. await orderManagementPage.cancelDelete();
  197. });
  198. });
  199. test.describe('取消删除', () => {
  200. test('应该能在确认对话框中取消删除', async ({ orderManagementPage }) => {
  201. // 获取或创建一个订单用于测试
  202. const existingOrder = await getFirstOrderName(orderManagementPage.page);
  203. const orderToTest = existingOrder || testOrderName;
  204. // 打开删除确认对话框
  205. await orderManagementPage.openDeleteDialog(orderToTest);
  206. // 取消删除
  207. await orderManagementPage.cancelDelete();
  208. // 验证订单仍然存在
  209. const exists = await orderManagementPage.orderExists(orderToTest);
  210. expect(exists).toBe(true);
  211. });
  212. test('取消删除后订单应该保持不变', async ({ orderManagementPage, page }) => {
  213. // 获取或创建一个订单用于测试
  214. const existingOrder = await getFirstOrderName(orderManagementPage.page);
  215. const orderToTest = existingOrder || testOrderName;
  216. // 获取删除前的订单行(用于后续验证)
  217. const orderRowBefore = page.locator('table tbody tr').filter({ hasText: orderToTest });
  218. await expect(orderRowBefore).toBeVisible();
  219. // 打开删除确认对话框
  220. await orderManagementPage.openDeleteDialog(orderToTest);
  221. // 取消删除
  222. await orderManagementPage.cancelDelete();
  223. // 等待对话框关闭
  224. await page.waitForTimeout(TIMEOUTS.MEDIUM);
  225. // 验证订单仍然在列表中
  226. const orderRowAfter = page.locator('table tbody tr').filter({ hasText: orderToTest });
  227. await expect(orderRowAfter).toBeVisible();
  228. });
  229. test('应该能通过关闭对话框取消删除', async ({ orderManagementPage, page }) => {
  230. // 获取或创建一个订单用于测试
  231. const existingOrder = await getFirstOrderName(orderManagementPage.page);
  232. const orderToTest = existingOrder || testOrderName;
  233. // 打开删除确认对话框
  234. await orderManagementPage.openDeleteDialog(orderToTest);
  235. // 按 Escape 键关闭对话框
  236. await page.keyboard.press('Escape');
  237. // 等待对话框关闭
  238. const dialog = page.locator('[role="alertdialog"]');
  239. await dialog.waitFor({ state: 'hidden', timeout: TIMEOUTS.DIALOG }).catch(() => {
  240. console.debug('对话框关闭超时,可能已经关闭');
  241. });
  242. // 验证订单仍然存在
  243. const exists = await orderManagementPage.orderExists(orderToTest);
  244. expect(exists).toBe(true);
  245. });
  246. });
  247. test.describe('删除有关联人员的订单', () => {
  248. let orderWithPersonName: string;
  249. test.beforeEach(async ({ orderManagementPage }) => {
  250. // 创建专用于此测试套件的订单(确保测试隔离)
  251. const timestamp = Date.now();
  252. orderWithPersonName = `删除测试_人员_${timestamp}`;
  253. const created = await createTestOrder(orderManagementPage.page, orderWithPersonName);
  254. if (!created) {
  255. test.skip(true, '无法创建测试订单用于人员关联测试');
  256. }
  257. // 验证订单出现在列表中
  258. await expect(async () => {
  259. const exists = await orderManagementPage.orderExists(orderWithPersonName);
  260. expect(exists).toBe(true);
  261. }).toPass({ timeout: TIMEOUTS.DIALOG });
  262. // 打开人员管理对话框并添加人员
  263. await orderManagementPage.openPersonManagementDialog(orderWithPersonName);
  264. // 检查是否已经有人员(如果有则使用现有人员,否则尝试添加)
  265. const personTable = orderManagementPage.page.locator('[role="dialog"]').locator('table tbody tr');
  266. const personCount = await personTable.count();
  267. if (personCount === 0) {
  268. // 尝试添加人员
  269. try {
  270. await orderManagementPage.addPersonToOrder({
  271. disabledPersonName: '测试', // 使用名称而不是硬编码 ID
  272. });
  273. console.debug('✓ 已添加人员到订单');
  274. } catch (_error) {
  275. console.debug('添加人员失败,可能没有可用数据:', error);
  276. }
  277. } else {
  278. console.debug(`✓ 订单已有 ${personCount} 个关联人员`);
  279. }
  280. // 关闭人员管理对话框
  281. await orderManagementPage.page.keyboard.press('Escape');
  282. await orderManagementPage.page.waitForTimeout(TIMEOUTS.MEDIUM);
  283. });
  284. test('应该能删除有人员的订单(级联删除)', async ({ orderManagementPage }) => {
  285. // 尝试删除有关联人员的订单
  286. await orderManagementPage.deleteOrder(orderWithPersonName);
  287. // 验证结果 - 根据实际业务逻辑,可能成功或失败
  288. const successToast = orderManagementPage.page.locator('[data-sonner-toast][data-type="success"]');
  289. const errorToast = orderManagementPage.page.locator('[data-sonner-toast][data-type="error"]');
  290. const hasSuccess = await successToast.count() > 0;
  291. const hasError = await errorToast.count() > 0;
  292. if (hasSuccess) {
  293. // 级联删除成功
  294. console.debug('✓ 订单及其关联人员已被删除');
  295. const exists = await orderManagementPage.orderExists(orderWithPersonName);
  296. expect(exists).toBe(false);
  297. } else if (hasError) {
  298. // 禁止删除,显示错误消息
  299. const errorMessage = await errorToast.textContent();
  300. console.debug('删除失败消息:', errorMessage);
  301. expect(errorMessage).toBeDefined();
  302. // 订单应该仍然存在
  303. const exists = await orderManagementPage.orderExists(orderWithPersonName);
  304. expect(exists).toBe(true);
  305. } else {
  306. // 如果没有明确的成功或失败消息,检查订单状态
  307. const exists = await orderManagementPage.orderExists(orderWithPersonName);
  308. console.debug(`删除后订单状态: ${exists ? '仍存在' : '已删除'}`);
  309. }
  310. });
  311. test('删除失败应该显示错误消息', async ({ orderManagementPage }) => {
  312. // 打开删除对话框
  313. await orderManagementPage.openDeleteDialog(orderWithPersonName);
  314. // 确认删除
  315. await orderManagementPage.confirmDelete();
  316. // 等待响应
  317. await orderManagementPage.page.waitForTimeout(TIMEOUTS.VERY_LONG);
  318. // 检查结果
  319. const successToast = orderManagementPage.page.locator('[data-sonner-toast][data-type="success"]');
  320. const errorToast = orderManagementPage.page.locator('[data-sonner-toast][data-type="error"]');
  321. const hasSuccess = await successToast.count() > 0;
  322. const hasError = await errorToast.count() > 0;
  323. // 根据业务逻辑验证结果
  324. if (hasError) {
  325. const errorMessage = await errorToast.textContent();
  326. console.debug('预期错误消息:', errorMessage);
  327. expect(errorMessage).toBeDefined();
  328. expect(errorMessage?.length).toBeGreaterThan(0);
  329. } else if (!hasSuccess) {
  330. // 如果没有明确的 Toast 消息,检查其他形式的反馈
  331. console.debug('没有检测到 Toast 消息,检查其他反馈形式');
  332. }
  333. // 清理:如果删除成功,则不需要操作;如果删除失败,订单仍在
  334. const exists = await orderManagementPage.orderExists(orderWithPersonName);
  335. if (exists) {
  336. console.debug('订单仍然存在,业务规则可能禁止删除有人员的订单');
  337. }
  338. });
  339. });
  340. test.describe('删除后列表更新验证', () => {
  341. test('删除后列表应该不再显示该订单', async ({ orderManagementPage }) => {
  342. // 获取或创建一个订单用于测试
  343. const existingOrder = await getFirstOrderName(orderManagementPage.page);
  344. const orderToDelete = existingOrder || testOrderName;
  345. // 验证订单在删除前存在
  346. const existsBefore = await orderManagementPage.orderExists(orderToDelete);
  347. expect(existsBefore).toBe(true);
  348. // 执行删除
  349. await orderManagementPage.deleteOrder(orderToDelete);
  350. // 验证订单在删除后不存在
  351. await expect(async () => {
  352. const existsAfter = await orderManagementPage.orderExists(orderToDelete);
  353. expect(existsAfter).toBe(false);
  354. }).toPass({ timeout: TIMEOUTS.DIALOG });
  355. });
  356. test('删除后列表应该正确更新', async ({ orderManagementPage, page }) => {
  357. // 获取或创建一个订单用于测试
  358. const existingOrder = await getFirstOrderName(orderManagementPage.page);
  359. const orderToDelete = existingOrder || testOrderName;
  360. // 执行删除
  361. await orderManagementPage.deleteOrder(orderToDelete);
  362. // 等待列表更新
  363. await page.waitForTimeout(TIMEOUTS.LONG);
  364. // 验证被删除的订单不再存在
  365. const exists = await orderManagementPage.orderExists(orderToDelete);
  366. expect(exists).toBe(false);
  367. // 注意:由于分页机制,如果总数据量大于页面大小,
  368. // 删除一行后列表会自动填充新行,所以行数可能不变
  369. // 这里的验证重点是确认被删除的订单确实不存在
  370. });
  371. });
  372. test.describe('Toast 消息验证', () => {
  373. test('成功删除应该显示正确的成功消息', async ({ orderManagementPage }) => {
  374. // 获取或创建一个订单用于测试
  375. const existingOrder = await getFirstOrderName(orderManagementPage.page);
  376. const orderToDelete = existingOrder || testOrderName;
  377. await orderManagementPage.deleteOrder(orderToDelete);
  378. const successToast = orderManagementPage.page.locator('[data-sonner-toast][data-type="success"]');
  379. await expect(successToast).toBeVisible();
  380. const message = await successToast.textContent();
  381. expect(message).toBeDefined();
  382. expect(message?.length).toBeGreaterThan(0);
  383. // 验证消息包含删除相关的关键词
  384. expect(message).toMatch(/删除|成功|已删除/);
  385. console.debug('删除成功消息:', message);
  386. });
  387. test('Toast 消息应该自动消失', async ({ orderManagementPage }) => {
  388. // 获取或创建一个订单用于测试
  389. const existingOrder = await getFirstOrderName(orderManagementPage.page);
  390. const orderToDelete = existingOrder || testOrderName;
  391. await orderManagementPage.deleteOrder(orderToDelete);
  392. const successToast = orderManagementPage.page.locator('[data-sonner-toast][data-type="success"]');
  393. // 等待 Toast 消息消失 - 使用更合理的超时时间
  394. await successToast.waitFor({ state: 'hidden', timeout: TIMEOUTS.TOAST_LONG }).catch(() => {
  395. console.debug('Toast 消息可能在 6 秒内未消失');
  396. });
  397. // 验证消息不再可见
  398. const isVisible = await successToast.isVisible().catch(() => false);
  399. expect(isVisible).toBe(false);
  400. });
  401. });
  402. });