region-edit.spec.ts 19 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538
  1. import { test, expect } from '../../utils/test-setup';
  2. import { readFileSync } from 'fs';
  3. import { join, dirname } from 'path';
  4. import { fileURLToPath } from 'url';
  5. const __filename = fileURLToPath(import.meta.url);
  6. const __dirname = dirname(__filename);
  7. const testUsers = JSON.parse(readFileSync(join(__dirname, '../../fixtures/test-users.json'), 'utf-8'));
  8. /**
  9. * 生成唯一区域名称
  10. * @param prefix - 名称前缀
  11. * @returns 唯一的区域名称
  12. */
  13. function generateUniqueRegionName(prefix: string = '测试区域'): string {
  14. const timestamp = Date.now();
  15. const random = Math.floor(Math.random() * 1000);
  16. return `${prefix}_${timestamp}_${random}`;
  17. }
  18. /**
  19. * 生成唯一区域代码
  20. * @param level - 区域层级
  21. * @returns 唯一的区域代码
  22. */
  23. function generateUniqueRegionCode(level: string): string {
  24. const timestamp = Date.now();
  25. return `${level.toUpperCase()}_${timestamp}`;
  26. }
  27. test.describe.serial('编辑区域测试', () => {
  28. // 用于跟踪测试创建的区域,以便清理
  29. const createdProvinces: string[] = [];
  30. test.beforeEach(async ({ adminLoginPage, regionManagementPage }) => {
  31. // 以管理员身份登录后台
  32. await adminLoginPage.goto();
  33. await adminLoginPage.login(testUsers.admin.username, testUsers.admin.password);
  34. await adminLoginPage.expectLoginSuccess();
  35. await regionManagementPage.goto();
  36. await regionManagementPage.waitForTreeLoaded();
  37. });
  38. test.afterEach(async ({ regionManagementPage, page }) => {
  39. // 清理测试创建的数据
  40. let cleanupSuccessCount = 0;
  41. let cleanupFailCount = 0;
  42. for (const provinceName of createdProvinces) {
  43. try {
  44. // 尝试刷新页面并删除
  45. await page.goto('/admin/areas');
  46. await page.waitForLoadState('domcontentloaded', { timeout: 10000 });
  47. await page.waitForTimeout(1000);
  48. const exists = await regionManagementPage.regionExists(provinceName);
  49. if (exists) {
  50. const deleteSuccess = await regionManagementPage.deleteRegion(provinceName);
  51. if (deleteSuccess) {
  52. cleanupSuccessCount++;
  53. console.debug(`✅ 已清理测试数据: ${provinceName}`);
  54. } else {
  55. cleanupFailCount++;
  56. console.debug(`⚠️ 删除失败(无成功提示): ${provinceName}`);
  57. }
  58. } else {
  59. console.debug(`ℹ️ 区域不存在,跳过删除: ${provinceName}`);
  60. }
  61. } catch (error) {
  62. cleanupFailCount++;
  63. console.debug(`❌ 清理异常: ${provinceName}`, error);
  64. }
  65. }
  66. // 记录清理结果摘要
  67. console.debug(`🧹 测试数据清理: 成功 ${cleanupSuccessCount}, 失败 ${cleanupFailCount}`);
  68. // 如果有清理失败,记录警告但不阻塞测试
  69. if (cleanupFailCount > 0) {
  70. console.debug(`⚠️ 有 ${cleanupFailCount} 个区域清理失败,可能产生脏数据`);
  71. }
  72. // 清空跟踪列表
  73. createdProvinces.length = 0;
  74. });
  75. test.describe('编辑区域名称', () => {
  76. test('应该成功编辑区域名称', async ({ regionManagementPage }) => {
  77. // 首先创建一个测试省份
  78. const originalName = generateUniqueRegionName('测试省');
  79. await regionManagementPage.createProvince({
  80. name: originalName,
  81. code: generateUniqueRegionCode('PROV'),
  82. level: 1,
  83. });
  84. createdProvinces.push(originalName);
  85. // 等待树形结构刷新
  86. await regionManagementPage.waitForTreeLoaded();
  87. // 编辑区域名称
  88. const newName = generateUniqueRegionName('编辑后的省');
  89. const result = await regionManagementPage.editRegion(originalName, {
  90. name: newName,
  91. });
  92. // 验证编辑成功
  93. expect(result.success).toBe(true);
  94. expect(result.hasError).toBe(false);
  95. // 等待树形结构刷新
  96. await regionManagementPage.waitForTreeLoaded();
  97. // 验证列表中显示新名称
  98. const exists = await regionManagementPage.regionExists(newName);
  99. expect(exists).toBe(true);
  100. // 更新清理列表中的名称
  101. const index = createdProvinces.indexOf(originalName);
  102. if (index > -1) {
  103. createdProvinces[index] = newName;
  104. }
  105. });
  106. test('编辑后原名称不应存在', async ({ regionManagementPage }) => {
  107. const originalName = generateUniqueRegionName('测试省');
  108. await regionManagementPage.createProvince({
  109. name: originalName,
  110. code: generateUniqueRegionCode('PROV'),
  111. level: 1,
  112. });
  113. createdProvinces.push(originalName);
  114. await regionManagementPage.waitForTreeLoaded();
  115. const newName = generateUniqueRegionName('编辑后的省');
  116. const result = await regionManagementPage.editRegion(originalName, { name: newName });
  117. expect(result.success).toBe(true);
  118. await regionManagementPage.waitForTreeLoaded();
  119. // 验证原名称不存在
  120. const originalExists = await regionManagementPage.regionExists(originalName);
  121. expect(originalExists).toBe(false);
  122. // 验证新名称存在
  123. const newExists = await regionManagementPage.regionExists(newName);
  124. expect(newExists).toBe(true);
  125. // 更新清理列表
  126. const index = createdProvinces.indexOf(originalName);
  127. if (index > -1) {
  128. createdProvinces[index] = newName;
  129. }
  130. });
  131. });
  132. test.describe('修改区域代码', () => {
  133. test('应该成功修改行政区划代码', async ({ regionManagementPage }) => {
  134. const provinceName = generateUniqueRegionName('测试省');
  135. await regionManagementPage.createProvince({
  136. name: provinceName,
  137. code: 'OLD_CODE',
  138. level: 1,
  139. });
  140. createdProvinces.push(provinceName);
  141. await regionManagementPage.waitForTreeLoaded();
  142. // 修改代码
  143. const newCode = generateUniqueRegionCode('NEW');
  144. const result = await regionManagementPage.editRegion(provinceName, {
  145. code: newCode,
  146. });
  147. expect(result.success).toBe(true);
  148. expect(result.hasError).toBe(false);
  149. });
  150. test('应该能同时修改名称和代码', async ({ regionManagementPage }) => {
  151. const originalName = generateUniqueRegionName('测试省');
  152. await regionManagementPage.createProvince({
  153. name: originalName,
  154. code: 'OLD_CODE',
  155. level: 1,
  156. });
  157. createdProvinces.push(originalName);
  158. await regionManagementPage.waitForTreeLoaded();
  159. const newName = generateUniqueRegionName('新省名');
  160. const newCode = generateUniqueRegionCode('NEW');
  161. const result = await regionManagementPage.editRegion(originalName, {
  162. name: newName,
  163. code: newCode,
  164. });
  165. expect(result.success).toBe(true);
  166. expect(result.hasError).toBe(false);
  167. await regionManagementPage.waitForTreeLoaded();
  168. expect(await regionManagementPage.regionExists(newName)).toBe(true);
  169. // 更新清理列表
  170. const index = createdProvinces.indexOf(originalName);
  171. if (index > -1) {
  172. createdProvinces[index] = newName;
  173. }
  174. });
  175. });
  176. test.describe('区域状态切换', () => {
  177. test('应该成功禁用已启用的区域', async ({ regionManagementPage, page }) => {
  178. const provinceName = generateUniqueRegionName('测试省');
  179. console.debug(`创建省份: ${provinceName}`);
  180. await regionManagementPage.createProvince({
  181. name: provinceName,
  182. code: generateUniqueRegionCode('PROV'),
  183. level: 1,
  184. });
  185. createdProvinces.push(provinceName);
  186. // 刷新页面以确保获取最新状态
  187. await page.goto('/admin/areas');
  188. await regionManagementPage.waitForTreeLoaded();
  189. // 等待新创建的区域在树中可见
  190. await page.waitForTimeout(1000);
  191. const exists = await regionManagementPage.regionExists(provinceName);
  192. console.debug(`区域 "${provinceName}" 在树中存在: ${exists}`);
  193. if (!exists) {
  194. // 尝试滚动页面查找
  195. await page.mouse.wheel(0, 500);
  196. await page.waitForTimeout(500);
  197. }
  198. // 获取初始状态
  199. const initialStatus = await regionManagementPage.getRegionStatus(provinceName);
  200. console.debug(`区域 "${provinceName}" 初始状态: ${initialStatus}`);
  201. expect(initialStatus).toBe('启用');
  202. // 禁用区域
  203. const success = await regionManagementPage.toggleRegionStatus(provinceName);
  204. console.debug(`禁用操作返回: ${success}`);
  205. expect(success).toBe(true);
  206. // 再次刷新页面以确保看到更新后的状态
  207. await page.goto('/admin/areas');
  208. await regionManagementPage.waitForTreeLoaded();
  209. // 等待新创建的区域在树中可见
  210. await page.waitForTimeout(1000);
  211. // 验证状态已更新
  212. const newStatus = await regionManagementPage.getRegionStatus(provinceName);
  213. console.debug(`区域 "${provinceName}" 新状态: ${newStatus}`);
  214. expect(newStatus).toBe('禁用');
  215. });
  216. test('应该成功启用已禁用的区域', async ({ regionManagementPage, page }) => {
  217. const provinceName = generateUniqueRegionName('测试省');
  218. await regionManagementPage.createProvince({
  219. name: provinceName,
  220. code: generateUniqueRegionCode('PROV'),
  221. level: 1,
  222. });
  223. createdProvinces.push(provinceName);
  224. await page.goto('/admin/areas');
  225. await regionManagementPage.waitForTreeLoaded();
  226. await page.waitForTimeout(1000);
  227. // 先禁用
  228. await regionManagementPage.toggleRegionStatus(provinceName);
  229. await page.goto('/admin/areas');
  230. await regionManagementPage.waitForTreeLoaded();
  231. await page.waitForTimeout(1000);
  232. // 验证已禁用
  233. const disabledStatus = await regionManagementPage.getRegionStatus(provinceName);
  234. expect(disabledStatus).toBe('禁用');
  235. // 再启用
  236. const success = await regionManagementPage.toggleRegionStatus(provinceName);
  237. expect(success).toBe(true);
  238. await page.goto('/admin/areas');
  239. await regionManagementPage.waitForTreeLoaded();
  240. await page.waitForTimeout(1000);
  241. // 验证状态已恢复为启用
  242. const status = await regionManagementPage.getRegionStatus(provinceName);
  243. expect(status).toBe('启用');
  244. });
  245. test('取消状态切换应保持原状态', async ({ regionManagementPage, page }) => {
  246. const provinceName = generateUniqueRegionName('测试省');
  247. await regionManagementPage.createProvince({
  248. name: provinceName,
  249. code: generateUniqueRegionCode('PROV'),
  250. level: 1,
  251. });
  252. createdProvinces.push(provinceName);
  253. await page.goto('/admin/areas');
  254. await regionManagementPage.waitForTreeLoaded();
  255. const initialStatus = await regionManagementPage.getRegionStatus(provinceName);
  256. expect(initialStatus).toBe('启用');
  257. // 打开状态切换对话框但取消
  258. await regionManagementPage.openToggleStatusDialog(provinceName);
  259. await regionManagementPage.cancelToggleStatus();
  260. await page.goto('/admin/areas');
  261. await regionManagementPage.waitForTreeLoaded();
  262. // 验证状态未改变
  263. const currentStatus = await regionManagementPage.getRegionStatus(provinceName);
  264. expect(currentStatus).toBe(initialStatus);
  265. });
  266. });
  267. test.describe.skip('编辑子区域 - TODO: 需要修复 createChildRegion 功能', () => {
  268. test('应该成功编辑市级区域名称', async ({ regionManagementPage, page }) => {
  269. const provinceName = generateUniqueRegionName('测试省');
  270. const originalCityName = generateUniqueRegionName('测试市');
  271. // 创建省和市
  272. await regionManagementPage.createProvince({
  273. name: provinceName,
  274. code: generateUniqueRegionCode('PROV'),
  275. level: 1,
  276. });
  277. createdProvinces.push(provinceName);
  278. await page.goto('/admin/areas');
  279. await regionManagementPage.waitForTreeLoaded();
  280. // 创建市后,先展开省节点验证市已创建
  281. const cityResult = await regionManagementPage.createChildRegion(provinceName, '市', {
  282. name: originalCityName,
  283. code: generateUniqueRegionCode('CITY'),
  284. level: 2,
  285. });
  286. expect(cityResult.success).toBe(true);
  287. await page.goto('/admin/areas');
  288. await regionManagementPage.waitForTreeLoaded();
  289. // 直接展开新创建的省节点(滚动到可见区域)
  290. await regionManagementPage.expandNode(provinceName);
  291. await page.waitForTimeout(1000);
  292. // 验证市级区域可见
  293. const cityVisible = await regionManagementPage.regionExists(originalCityName);
  294. console.debug(`市级区域 "${originalCityName}" 可见: ${cityVisible}`);
  295. expect(cityVisible).toBe(true);
  296. // 编辑城市名称
  297. const newCityName = generateUniqueRegionName('编辑后的市');
  298. const result = await regionManagementPage.editRegion(originalCityName, {
  299. name: newCityName,
  300. });
  301. expect(result.success).toBe(true);
  302. expect(result.hasError).toBe(false);
  303. });
  304. test('应该成功编辑区级区域状态', async ({ regionManagementPage, page }) => {
  305. const provinceName = generateUniqueRegionName('测试省');
  306. const cityName = generateUniqueRegionName('测试市');
  307. const districtName = generateUniqueRegionName('测试区');
  308. // 创建省市区三级结构
  309. await regionManagementPage.createProvince({
  310. name: provinceName,
  311. code: generateUniqueRegionCode('PROV'),
  312. level: 1,
  313. });
  314. createdProvinces.push(provinceName);
  315. await page.goto('/admin/areas');
  316. await regionManagementPage.waitForTreeLoaded();
  317. const cityResult = await regionManagementPage.createChildRegion(provinceName, '市', {
  318. name: cityName,
  319. code: generateUniqueRegionCode('CITY'),
  320. level: 2,
  321. });
  322. expect(cityResult.success).toBe(true);
  323. await page.goto('/admin/areas');
  324. await regionManagementPage.waitForTreeLoaded();
  325. const districtResult = await regionManagementPage.createChildRegion(provinceName, '市', {
  326. name: districtName,
  327. code: generateUniqueRegionCode('DISTRICT'),
  328. level: 3,
  329. });
  330. expect(districtResult.success).toBe(true);
  331. await page.goto('/admin/areas');
  332. await regionManagementPage.waitForTreeLoaded();
  333. // 展开省节点
  334. await regionManagementPage.expandNode(provinceName);
  335. await page.waitForTimeout(1000);
  336. // 切换区的状态
  337. const success = await regionManagementPage.toggleRegionStatus(districtName);
  338. expect(success).toBe(true);
  339. });
  340. });
  341. test.describe('表单验证', () => {
  342. test('清空名称时应显示错误提示', async ({ regionManagementPage }) => {
  343. const provinceName = generateUniqueRegionName('测试省');
  344. await regionManagementPage.createProvince({
  345. name: provinceName,
  346. code: generateUniqueRegionCode('PROV'),
  347. level: 1,
  348. });
  349. createdProvinces.push(provinceName);
  350. await regionManagementPage.waitForTreeLoaded();
  351. // 打开编辑对话框并清空名称
  352. await regionManagementPage.openEditDialog(provinceName);
  353. await regionManagementPage.page.getByLabel('区域名称').fill('');
  354. // 提交表单
  355. const submitButton = regionManagementPage.page.getByRole('button', { name: '更新' });
  356. await submitButton.click();
  357. // 验证错误提示 - 可能是内联错误或 toast
  358. await regionManagementPage.page.waitForTimeout(500);
  359. // 检查内联错误
  360. const nameError = regionManagementPage.page.getByText('区域名称不能为空');
  361. const hasInlineError = await nameError.count() > 0;
  362. // 检查 toast 错误
  363. const errorToast = regionManagementPage.page.locator('[data-sonner-toast][data-type="error"]');
  364. const hasToastError = await errorToast.count() > 0;
  365. // 至少应该有一种错误提示
  366. expect(hasInlineError || hasToastError).toBe(true);
  367. // 取消对话框
  368. await regionManagementPage.cancelDialog();
  369. });
  370. test('应该支持取消编辑操作', async ({ regionManagementPage }) => {
  371. const provinceName = generateUniqueRegionName('测试省');
  372. await regionManagementPage.createProvince({
  373. name: provinceName,
  374. code: generateUniqueRegionCode('PROV'),
  375. level: 1,
  376. });
  377. createdProvinces.push(provinceName);
  378. await regionManagementPage.waitForTreeLoaded();
  379. // 打开编辑对话框
  380. await regionManagementPage.openEditDialog(provinceName);
  381. // 修改名称
  382. const newName = generateUniqueRegionName('修改后的省');
  383. await regionManagementPage.page.getByLabel('区域名称').fill(newName);
  384. // 点击取消按钮
  385. await regionManagementPage.cancelDialog();
  386. // 验证对话框已关闭
  387. const dialog = regionManagementPage.page.locator('[role="dialog"]');
  388. await expect(dialog).not.toBeVisible();
  389. // 验证数据未修改 - 原名称仍存在
  390. await regionManagementPage.waitForTreeLoaded();
  391. const originalExists = await regionManagementPage.regionExists(provinceName);
  392. expect(originalExists).toBe(true);
  393. });
  394. });
  395. test.describe('连续编辑操作', () => {
  396. test('应该能连续编辑多个区域', async ({ regionManagementPage, page }) => {
  397. // 创建多个省份
  398. const province1 = generateUniqueRegionName('测试省1');
  399. const province2 = generateUniqueRegionName('测试省2');
  400. await regionManagementPage.createProvince({
  401. name: province1,
  402. code: generateUniqueRegionCode('PROV1'),
  403. level: 1,
  404. });
  405. createdProvinces.push(province1);
  406. await page.goto('/admin/areas');
  407. await regionManagementPage.waitForTreeLoaded();
  408. await regionManagementPage.createProvince({
  409. name: province2,
  410. code: generateUniqueRegionCode('PROV2'),
  411. level: 1,
  412. });
  413. createdProvinces.push(province2);
  414. await page.goto('/admin/areas');
  415. await regionManagementPage.waitForTreeLoaded();
  416. // 编辑第一个省份
  417. const newProvince1Name = generateUniqueRegionName('编辑后的省1');
  418. const result1 = await regionManagementPage.editRegion(province1, { name: newProvince1Name });
  419. expect(result1.success).toBe(true);
  420. // 更新清理列表
  421. const index1 = createdProvinces.indexOf(province1);
  422. if (index1 > -1) {
  423. createdProvinces[index1] = newProvince1Name;
  424. }
  425. // 编辑第二个省份
  426. const newProvince2Name = generateUniqueRegionName('编辑后的省2');
  427. const result2 = await regionManagementPage.editRegion(province2, { name: newProvince2Name });
  428. expect(result2.success).toBe(true);
  429. // 更新清理列表
  430. const index2 = createdProvinces.indexOf(province2);
  431. if (index2 > -1) {
  432. createdProvinces[index2] = newProvince2Name;
  433. }
  434. });
  435. });
  436. });