utils.ts 14 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597
  1. import {StaticResponse} from 'cypress/types/net-stubbing';
  2. import menuJson from '../../fixtures/menu/basic.json';
  3. /** 点击select具体项 */
  4. export function selectClick(testid: string, eq = 0) {
  5. cy.getTestId(testid).click();
  6. cy.getTestId(testid)
  7. .find('.rc-virtual-list-holder-inner')
  8. .children('.ant-select-item')
  9. .eq(eq)
  10. .click();
  11. }
  12. /** 进入具体菜单 */
  13. export function intoMenu(parent: string, child: string) {
  14. cy.wait('@loginIntercept');
  15. const list = menuJson.data;
  16. const menu = list.find(val => val.name === child);
  17. menu && cy.visit(menu);
  18. }
  19. export function clickMenu(parent: string, child: string) {
  20. cy.getTestId('menu')
  21. .children()
  22. .each(function(el) {
  23. const titleElement = el.find('.ant-menu-submenu-title');
  24. if (titleElement.text() === parent) {
  25. titleElement.trigger('click');
  26. el.find(`ul>li[title="${child}"]`).trigger('click');
  27. }
  28. });
  29. }
  30. /** 校验select */
  31. export function validateSelect(testid: string, value: string) {
  32. cy.getTestId(testid)
  33. .find('.ant-select-selection-item')
  34. .should('have.attr', 'title', value)
  35. .and('have.text', value);
  36. }
  37. /** 校验table内容 */
  38. export function validateTableList(tableName: string) {
  39. cy.getTestId(tableName)
  40. .find('table')
  41. .find('tbody')
  42. .children('tr')
  43. .first()
  44. .find('td')
  45. .eq(1)
  46. .should('have.text', 'page-1');
  47. cy.getTestId(tableName)
  48. .siblings('.ant-pagination')
  49. .find('li[title="2"]')
  50. .click();
  51. cy.getTestId(tableName)
  52. .find('table')
  53. .find('tbody')
  54. .children('tr')
  55. .first()
  56. .find('td')
  57. .eq(1)
  58. .should('have.text', 'page-2');
  59. }
  60. /** 选择时间组件 */
  61. export function clickDatePicker(id: string, enableTime = false) {
  62. cy.getTestId(id).click();
  63. if (enableTime) {
  64. cy.get('.ant-picker-datetime-panel')
  65. .children()
  66. .eq(0)
  67. .find('.ant-picker-content')
  68. .find('tbody')
  69. .find('td')
  70. .eq(0)
  71. .click();
  72. cy.get('.ant-picker-footer').find('.ant-picker-ok').find('button').click();
  73. cy.get('.ant-picker-datetime-panel')
  74. .children()
  75. .eq(0)
  76. .find('.ant-picker-content')
  77. .find('tbody')
  78. .find('td')
  79. .eq(1)
  80. .click();
  81. cy.get('.ant-picker-footer').find('.ant-picker-ok').find('button').click();
  82. return;
  83. }
  84. cy.get('.ant-picker-panels')
  85. .children()
  86. .eq(0)
  87. .find('.ant-picker-content')
  88. .find('tbody')
  89. .find('td')
  90. .eq(0)
  91. .click();
  92. cy.get('.ant-picker-panels')
  93. .children()
  94. .eq(1)
  95. .find('.ant-picker-content')
  96. .find('tbody')
  97. .find('td')
  98. .eq(0)
  99. .click();
  100. }
  101. /** 校验table search */
  102. export function validateTableSearch(
  103. tableName: string,
  104. keys: (
  105. | {
  106. id: string;
  107. type: 'select';
  108. value: string;
  109. eq?: number;
  110. defaultValue?: string;
  111. defaultEq?: number;
  112. }
  113. | {id: string; type: 'date'; enableTime?: boolean}
  114. | string
  115. )[],
  116. options: {
  117. toolId: string;
  118. url: string;
  119. },
  120. ) {
  121. const {toolId, url} = options;
  122. cy.getTestId(tableName)
  123. .siblings('.ant-pagination')
  124. .find('li[title="2"]')
  125. .click();
  126. function clear(data?: (typeof keys)[0]) {
  127. function clearData(key: (typeof keys)[0]) {
  128. if (typeof key === 'string') {
  129. cy.getTestId(toolId).find(`[data-testid="filter_${key}"]`).clear();
  130. } else {
  131. if (key.type === 'select' && key.defaultEq !== void 0) {
  132. selectClick(`filter_${key.id}`, key.defaultEq);
  133. return;
  134. }
  135. const {id, type} = key;
  136. const clearId
  137. = type === 'select' ? '.ant-select-clear' : '.ant-picker-clear';
  138. cy.getTestId(toolId)
  139. .find(`[data-testid="filter_${id}"]`)
  140. .trigger('mouseover')
  141. .find(clearId)
  142. .click();
  143. }
  144. }
  145. if (data)
  146. return clearData(data);
  147. keys.forEach(clearData);
  148. }
  149. function validate(text: string, reset?: boolean) {
  150. !reset && cy.getTestId(toolId).find('[data-testid="search_btn"]').click();
  151. cy.getTestId(toolId)
  152. .find('[data-testid="search_btn"]')
  153. .should('have.class', 'ant-btn-loading');
  154. cy.getTestId(tableName)
  155. .find('table')
  156. .find('tbody')
  157. .children('tr')
  158. .first()
  159. .find('td')
  160. .eq(1)
  161. .should('have.text', text);
  162. cy.getTestId(tableName)
  163. .siblings('.ant-pagination')
  164. .find('li[title="1"]')
  165. .should('have.class', 'ant-pagination-item-active');
  166. cy.getTestId(toolId)
  167. .getTestId('search_btn')
  168. .should('not.have.class', 'ant-btn-loading');
  169. }
  170. function eachAllField(skipValidate = false) {
  171. keys.forEach(function(key, idx) {
  172. let validateText: string;
  173. if (typeof key === 'string') {
  174. cy.getTestId(toolId).find(`[data-testid="filter_${key}"]`).type(key);
  175. validateText = key;
  176. } else {
  177. const {id, type} = key;
  178. switch (type) {
  179. case 'select':
  180. selectClick(`filter_${id}`, key?.eq ?? 0);
  181. validateText = key.value;
  182. break;
  183. case 'date':
  184. clickDatePicker(`filter_${id}`, key.enableTime);
  185. validateText = 'searchTime';
  186. break;
  187. }
  188. }
  189. if (!skipValidate) {
  190. validate(validateText);
  191. clear(keys[idx]);
  192. }
  193. });
  194. }
  195. eachAllField();
  196. // 校验重置功能
  197. eachAllField(true);
  198. cy.getTestId(toolId).find('[data-testid="reset_btn"]').click();
  199. keys.forEach(function(key) {
  200. if (typeof key === 'string') {
  201. cy.getTestId(`filter_${key}`).invoke('val').should('be.empty');
  202. } else {
  203. switch (key.type) {
  204. case 'select':
  205. if (key.defaultValue) {
  206. validateSelect(`filter_${key.id}`, key.defaultValue);
  207. break;
  208. }
  209. cy.getTestId(`filter_${key.id}`)
  210. .find('.ant-select-selection-placeholder')
  211. .should('exist');
  212. break;
  213. case 'date':
  214. cy.getTestId(`filter_${key.id}`)
  215. .children('.ant-picker-input')
  216. .first()
  217. .find('input')
  218. .invoke('val')
  219. .should('be.empty');
  220. cy.getTestId(`filter_${key.id}`)
  221. .children('.ant-picker-input')
  222. .last()
  223. .find('input')
  224. .invoke('val')
  225. .should('be.empty');
  226. break;
  227. }
  228. }
  229. });
  230. validate('page-1', true);
  231. // 校验刷新功能
  232. cy.getTestId(tableName)
  233. .closest('.ant-card-body')
  234. .find('[data-testid="refresh_btn"]')
  235. .click();
  236. cy.getTestId(tableName)
  237. .closest('.ant-card-body')
  238. .find('[data-testid="refresh_btn"]')
  239. .should('have.class', 'ant-btn-loading');
  240. cy.wait(`@${url}`);
  241. cy.getTestId(tableName)
  242. .closest('.ant-card-body')
  243. .find('[data-testid="refresh_btn"]')
  244. .should('not.have.class', 'ant-btn-loading');
  245. // 校验保存配置
  246. cy.getTestId(tableName)
  247. .closest('.ant-card-body')
  248. .find('[data-testid="save_setting_btn"]')
  249. .click();
  250. cy.getTestId(tableName)
  251. .closest('.ant-card-body')
  252. .find('[data-testid="save_setting_btn"]')
  253. .should('have.class', 'ant-btn-loading');
  254. cy.getTestId(tableName)
  255. .closest('.ant-card-body')
  256. .find('[data-testid="save_setting_btn"]')
  257. .should('not.have.class', 'ant-btn-loading');
  258. }
  259. /** 表格内按钮点击 */
  260. export function tableBtnClick(tableName: string, index: number) {
  261. function getBtn() {
  262. return cy
  263. .getTestId(tableName)
  264. .find('table')
  265. .find('tbody')
  266. .children('tr')
  267. .first()
  268. .find('td')
  269. .last()
  270. .children()
  271. .eq(index);
  272. }
  273. getBtn().click();
  274. return getBtn;
  275. }
  276. /** 弹窗按钮校验 */
  277. export function validateModalBtnGroup(modalName: string) {
  278. cy.getTestId(modalName)
  279. .getTestId('modal_btn_group')
  280. .find('.ant-btn')
  281. .should('have.class', 'ant-btn-loading');
  282. }
  283. // 校验antd message内容
  284. export function validateMessageContent(msg: string) {
  285. cy.get('.ant-message-notice-content').should('include.text', msg);
  286. }
  287. // 校验modal的操作结果
  288. export function validateModalOperation(modalName: string, text: string) {
  289. cy.getTestId(modalName).find('form').submit();
  290. validateModalBtnGroup(modalName);
  291. validateMessageContent(text);
  292. cy.getTestId(modalName).should('not.exist');
  293. }
  294. /** 校验新增或修改事件 */
  295. export function validatePut(
  296. modalName: string,
  297. tableName: string,
  298. options: {label: string},
  299. ) {
  300. const {label} = options;
  301. function validateAdd(
  302. keys: (
  303. | string
  304. | {id: string; type: 'select'}
  305. | {id: string; type: 'keySelect'}
  306. | {id: string; type: 'field'; value: string}
  307. )[],
  308. ) {
  309. cy.getTestId(tableName)
  310. .closest('.ant-card-body')
  311. .find('[data-testid="add_btn"]')
  312. .click();
  313. cy.getTestId(modalName).should('exist').and('be.visible');
  314. cy.getTestId(modalName).find('h3').should('have.text', `新增${label}`);
  315. keys.forEach(function(key) {
  316. if (typeof key === 'string') {
  317. cy.getTestId(`field_${key}`).type(key);
  318. } else {
  319. const {type, id} = key;
  320. switch (type) {
  321. case 'select':
  322. selectClick(`field_${id}`, 0);
  323. break;
  324. case 'field':
  325. cy.getTestId(`field_${id}`).clear().type(key.value);
  326. break;
  327. case 'keySelect':
  328. cy.get(`#field_${id}`).clear().type(id);
  329. selectClick(`field_${id}`, 0);
  330. break;
  331. }
  332. }
  333. });
  334. validateModalOperation(modalName, '新增成功');
  335. }
  336. function validateEdit(
  337. keys: {id: string; type: 'field' | 'select'; value: string}[],
  338. eq = 0,
  339. ) {
  340. tableBtnClick(tableName, eq);
  341. cy.getTestId(modalName).should('exist').and('be.visible');
  342. cy.getTestId(modalName).find('h3').should('include.text', `修改${label}`);
  343. keys.forEach(function({id, type, value}) {
  344. switch (type) {
  345. case 'field':
  346. cy.getTestId(`field_${id}`).should('have.value', value);
  347. break;
  348. case 'select':
  349. validateSelect(`field_${id}`, value);
  350. break;
  351. }
  352. });
  353. validateModalOperation(modalName, '修改成功');
  354. }
  355. return {validateAdd, validateEdit};
  356. }
  357. /** 校验antd dialog信息 */
  358. export function validateDialog(
  359. title: string,
  360. content: string,
  361. options?: {confirm?: boolean},
  362. ) {
  363. const {confirm = true} = options ?? {};
  364. cy.get('.ant-modal-confirm-title').should('include.text', title);
  365. cy.get('.ant-modal-confirm-content').should('include.text', content);
  366. cy.get('.ant-modal-confirm-btns')
  367. .children()
  368. .eq(confirm ? 1 : 0)
  369. .click();
  370. }
  371. /** 校验删除 */
  372. export function validateDelete(
  373. tableName: string,
  374. label: string,
  375. options: {eq?: number; title: string},
  376. ) {
  377. const {eq = 1, title} = options;
  378. tableBtnClick(tableName, eq);
  379. cy.get('.ant-modal-content').should('be.visible');
  380. validateDialog(title, `你确定要删除当前${label}吗?`);
  381. // 删除按钮在加载
  382. cy.getTestId(tableName)
  383. .find('table')
  384. .find('tbody')
  385. .children('tr')
  386. .first()
  387. .find('td')
  388. .last()
  389. .children()
  390. .eq(eq)
  391. .should('have.class', 'ant-btn-loading');
  392. // 其他按钮禁用
  393. cy.getTestId(tableName)
  394. .find('table')
  395. .find('tbody')
  396. .children('tr')
  397. .first()
  398. .find('td')
  399. .last()
  400. .children()
  401. .each(function(el, idx) {
  402. if (idx !== eq)
  403. expect(el).to.have.attr('disabled');
  404. });
  405. validateMessageContent('删除成功');
  406. }
  407. /** 校验导出 */
  408. export function validateExport(tableName: string) {
  409. cy.getTestId(tableName)
  410. .closest('.ant-card-body')
  411. .find('[data-testid="export_btn"]')
  412. .click();
  413. cy.getTestId(tableName)
  414. .closest('.ant-card-body')
  415. .find('[data-testid="export_btn"]')
  416. .should('have.class', 'ant-btn-loading');
  417. cy.getTestId(tableName)
  418. .closest('.ant-card-body')
  419. .find('[data-testid="export_btn"]')
  420. .should('not.have.class', 'ant-btn-loading');
  421. }
  422. /** 关闭modal */
  423. export function closeModal() {
  424. cy.get('.ReactModal__Overlay').trigger('click', 'topRight');
  425. }
  426. /** 选择所有的筛选框 */
  427. export function selectAllFilters(toolId: string, count: number) {
  428. cy.getTestId(toolId).getTestId('filter_btn').click();
  429. cy.get('.ant-transfer')
  430. .find('.ant-transfer-list')
  431. .first()
  432. .find('.ant-transfer-list-header')
  433. .find('input[type="checkbox"]')
  434. .check();
  435. cy.get('.ant-transfer')
  436. .find('.ant-transfer-operation')
  437. .find('button')
  438. .first()
  439. .click();
  440. cy.getTestId('filter_select_modal').find('form').submit();
  441. cy.getTestId(toolId)
  442. .children('.ant-space')
  443. .first()
  444. .children()
  445. .should('have.length', count + 1);
  446. }
  447. /** 生成返回内容 */
  448. export function generateResult<T extends Record<string, any>>(
  449. data: T,
  450. length: number,
  451. options: {
  452. limit: string;
  453. key?: string;
  454. isList?: boolean;
  455. title?: keyof T;
  456. value?: string;
  457. },
  458. ) {
  459. const {key = 'id', isList = false, title, value, limit} = options;
  460. const result = Array.from({length: Number(limit)}, function(_, idx) {
  461. const temp = {...data, [key]: (idx + 1).toString()};
  462. return title ? {...temp, [title]: value ?? ''} : temp;
  463. });
  464. return {msg: '200', data: isList ? {total: length, list: result} : result};
  465. }
  466. /** 生成请求结果 */
  467. export function generateNetworkResult<
  468. T extends Record<string, unknown>,
  469. >(options: {
  470. search: URLSearchParams;
  471. basicData: T;
  472. reply: (params: StaticResponse) => void;
  473. title: keyof T;
  474. skipCondition?: (name: string) => boolean;
  475. replaceValue?: string;
  476. }) {
  477. const {
  478. search,
  479. basicData,
  480. reply,
  481. title,
  482. skipCondition = () => false,
  483. replaceValue,
  484. } = options;
  485. const page = search.get('page'),
  486. limit = search.get('limit');
  487. for (const [name, searchValue] of search.entries()) {
  488. if (
  489. name === 'page'
  490. || name === 'limit'
  491. || name === 'id'
  492. || skipCondition(name)
  493. )
  494. continue;
  495. if (searchValue) {
  496. return reply({
  497. body: generateResult(basicData, Number(limit) * 3, {
  498. limit,
  499. isList: true,
  500. title,
  501. value:
  502. name === 'startTime' || name === 'endTime'
  503. ? 'searchTime'
  504. : replaceValue ?? searchValue,
  505. }),
  506. });
  507. }
  508. }
  509. if (search.has('id') && search.get('id').length > 0) {
  510. return reply({
  511. body: generateResult(basicData, 1, {limit: '1', isList: true}),
  512. });
  513. }
  514. reply({
  515. body: generateResult(basicData, Number(limit) * 3, {
  516. limit,
  517. isList: true,
  518. title,
  519. value: replaceValue ?? `page-${page}`,
  520. }),
  521. });
  522. }