Commit 0d108388a95505c5929c6a98451810d90348bd4a

Authored by 赵旭婷
1 parent 0247a9d4

新增账目管理测试用例,客户分组PO封装

pages/accountPage.ts 0 → 100644
  1 +import { Page, Locator, expect } from '@playwright/test';
  2 +import { BasePage } from './basePage';
  3 +
  4 +/**
  5 + * 账目管理页面选择器
  6 + */
  7 +const selectors = {
  8 + // 导航
  9 + moreMenu: 'text=更多 >',
  10 + accountMenu: 'text=账目管理',
  11 +
  12 + // 操作按钮
  13 + addButton: 'text=新增',
  14 + saveButton: 'text=保存',
  15 +
  16 + // 表单字段
  17 + accountNameInput: 'uni-input:has-text("科目名称") input',
  18 + remarkTextarea: 'uni-scroll-view textarea',
  19 +
  20 + // 收支类型单选
  21 + incomeRadio: '.nut-radio-group > uni-view:nth-child(1) > .nut-icon',
  22 + expenseRadio: '.nut-radio-group > uni-view:nth-child(2) > .nut-icon',
  23 +};
  24 +
  25 +/**
  26 + * 账目管理页面类
  27 + * 处理账目管理相关操作
  28 + */
  29 +export class AccountPage extends BasePage {
  30 + // 导航定位器
  31 + readonly moreMenu: Locator;
  32 + readonly accountMenu: Locator;
  33 +
  34 + // 操作按钮
  35 + readonly addButton: Locator;
  36 + readonly saveButton: Locator;
  37 + readonly editButton: Locator;
  38 +
  39 + // 表单字段
  40 + readonly accountNameInput: Locator;
  41 + readonly remarkTextarea: Locator;
  42 +
  43 + // 收支类型
  44 + readonly incomeRadio: Locator;
  45 + readonly expenseRadio: Locator;
  46 +
  47 + constructor(page: Page) {
  48 + super(page);
  49 +
  50 + this.moreMenu = page.getByText('更多 >');
  51 + this.accountMenu = page.getByText('账目管理').first();
  52 + this.addButton = page.getByText('新增', { exact: true });
  53 + this.saveButton = page.locator('uni-button.execute-btn').filter({ hasText: '保存' });
  54 + this.editButton = page.locator('uni-button').filter({ hasText: '修改' });
  55 + this.accountNameInput = page.locator('uni-input').filter({ hasText: '科目名称' }).getByRole('textbox');
  56 + this.remarkTextarea = page.locator('uni-scroll-view textarea');
  57 + this.incomeRadio = page.locator('.nut-radio-group > uni-view:nth-child(1) > .nut-icon');
  58 + this.expenseRadio = page.locator('.nut-radio-group > uni-view:nth-child(2) > .nut-icon');
  59 + }
  60 +
  61 + /**
  62 + * 进入账目管理页面
  63 + */
  64 + async openAccountManagement(): Promise<void> {
  65 + await this.navigate('/');
  66 + await this.moreMenu.click();
  67 + await this.page.waitForTimeout(500);
  68 + await this.accountMenu.click();
  69 + await this.page.waitForLoadState('networkidle', { timeout: 30000 });
  70 + }
  71 +
  72 + /**
  73 + * 点击新增按钮
  74 + */
  75 + async clickAddButton(): Promise<void> {
  76 + await this.addButton.click();
  77 + await this.page.waitForTimeout(500);
  78 + }
  79 +
  80 + /**
  81 + * 填写账目名称
  82 + * @param name 账目名称
  83 + */
  84 + async fillAccountName(name: string): Promise<void> {
  85 + await this.accountNameInput.click();
  86 + await this.accountNameInput.fill(name);
  87 + }
  88 +
  89 + /**
  90 + * 选择收支类型
  91 + * @param type '收入' 或 '支出'
  92 + */
  93 + async selectIncomeType(type: '收入' | '支出'): Promise<void> {
  94 + if (type === '支出') {
  95 + await this.expenseRadio.click();
  96 + }
  97 + // 收入是默认选项,无需选择
  98 + }
  99 +
  100 + /**
  101 + * 填写备注
  102 + * @param remark 备注内容
  103 + */
  104 + async fillRemark(remark: string): Promise<void> {
  105 + await this.remarkTextarea.click();
  106 + await this.remarkTextarea.fill(remark);
  107 + }
  108 +
  109 + /**
  110 + * 保存账目
  111 + */
  112 + async saveAccount(): Promise<void> {
  113 + await this.saveButton.click();
  114 + await this.page.waitForTimeout(1000);
  115 + }
  116 +
  117 + /**
  118 + * 搜索账目
  119 + * @param name 账目名称
  120 + */
  121 + async searchAccount(name: string): Promise<void> {
  122 + await this.page.locator('uni-input').filter({ hasText: '账目名称' }).getByRole('textbox').click();
  123 + await this.page.locator('uni-input').filter({ hasText: '账目名称' }).getByRole('textbox').fill(name);
  124 + await this.page.waitForTimeout(1000);
  125 + }
  126 +
  127 + /**
  128 + * 验证账目是否存在
  129 + * @param name 账目名称
  130 + * @returns 是否存在
  131 + */
  132 + async verifyAccountExists(name: string): Promise<boolean> {
  133 + const isVisible = await this.page.getByText(name, { exact: true }).isVisible().catch(() => false);
  134 + return isVisible;
  135 + }
  136 +
  137 + /**
  138 + * 删除账目
  139 + * @param name 账目名称
  140 + */
  141 + async deleteAccount(name: string): Promise<void> {
  142 + // 搜索账目
  143 + await this.searchAccount(name);
  144 + await this.page.waitForTimeout(500);
  145 + // 点击删除按钮
  146 + await this.page.getByText('删除').click();
  147 + await this.page.waitForTimeout(500);
  148 + // 确认删除
  149 + await this.page.getByText('确定').click();
  150 + await this.page.waitForTimeout(1000);
  151 + }
  152 +
  153 + /**
  154 + * 验证账目删除成功
  155 + * @param name 账目名称
  156 + * @returns 是否删除成功
  157 + */
  158 + async verifyAccountDeleted(name: string): Promise<boolean> {
  159 + await this.searchAccount(name);
  160 + await this.page.waitForTimeout(500);
  161 + const isVisible = await this.page.getByText(name, { exact: true }).isVisible().catch(() => false);
  162 + return !isVisible;
  163 + }
  164 +
  165 + /**
  166 + * 点击修改按钮
  167 + */
  168 + async clickEditButton(): Promise<void> {
  169 + await this.editButton.click();
  170 + await this.page.waitForTimeout(500);
  171 + }
  172 +
  173 + /**
  174 + * 修改账目完整流程
  175 + * @param originalName 原账目名称
  176 + * @param newName 新账目名称
  177 + * @param incomeType 收支类型(收入/支出)
  178 + * @param remark 备注
  179 + */
  180 + async updateAccount(originalName: string, newName: string, incomeType?: '收入' | '支出', remark?: string): Promise<void> {
  181 + // 搜索原账目
  182 + await this.searchAccount(originalName);
  183 + await this.page.waitForTimeout(500);
  184 + // 点击修改按钮
  185 + await this.clickEditButton();
  186 + // 修改账目名称
  187 + await this.accountNameInput.click();
  188 + await this.accountNameInput.fill(newName);
  189 + // 修改收支类型
  190 + if (incomeType === '支出') {
  191 + await this.expenseRadio.click();
  192 + }
  193 + // 修改备注
  194 + if (remark) {
  195 + await this.remarkTextarea.click();
  196 + await this.remarkTextarea.fill(remark);
  197 + }
  198 + // 保存
  199 + await this.saveAccount();
  200 + }
  201 +
  202 + /**
  203 + * 清除搜索框
  204 + */
  205 + async clearSearchBox(): Promise<void> {
  206 + await this.page.locator('.nut-icon.nutui-iconfont.nut-icon-circle-close').click();
  207 + await this.page.waitForTimeout(500);
  208 + }
  209 +
  210 + /**
  211 + * 新增账目完整流程
  212 + * @param accountName 账目名称
  213 + * @param incomeType 收支类型(默认'收入')
  214 + * @param remark 备注
  215 + */
  216 + async createAccount(accountName: string, incomeType: '收入' | '支出' = '收入', remark?: string): Promise<void> {
  217 + await this.openAccountManagement();
  218 + await this.clickAddButton();
  219 + await this.fillAccountName(accountName);
  220 + await this.selectIncomeType(incomeType);
  221 + if (remark) {
  222 + await this.fillRemark(remark);
  223 + }
  224 + await this.saveAccount();
  225 + }
  226 +
  227 + /**
  228 + * 验证账目创建成功
  229 + * @param accountName 账目名称
  230 + * @returns 是否创建成功
  231 + */
  232 + async verifyAccountCreated(accountName: string): Promise<boolean> {
  233 + await this.searchAccount(accountName);
  234 + return await this.verifyAccountExists(accountName);
  235 + }
  236 +}
pages/customerGroupPage.ts 0 → 100644
  1 +import { Page, Locator, expect } from '@playwright/test';
  2 +import { BasePage } from './basePage';
  3 +
  4 +/**
  5 + * 客户分组页面选择器
  6 + */
  7 +const selectors = {
  8 + // 导航
  9 + moreMenu: 'text=更多 >',
  10 + customerGroupMenu: 'text=客户分组',
  11 +
  12 + // 操作按钮
  13 + newButton: 'text=新建',
  14 + confirmButton: 'text=确定',
  15 + saveButton: 'text=保存',
  16 + deleteButton: 'text=删除',
  17 + editButton: 'text=编辑',
  18 +
  19 + // 表单字段
  20 + groupNameInput: 'text=分组名称*',
  21 + sortInput: 'textbox',
  22 +
  23 + // 搜索
  24 + searchInput: 'textbox',
  25 + clearSearchButton: '.nut-searchbar__search-icon',
  26 +};
  27 +
  28 +/**
  29 + * 客户分组页面类
  30 + * 处理客户分组相关操作
  31 + */
  32 +export class CustomerGroupPage extends BasePage {
  33 + // 导航定位器
  34 + readonly moreMenu: Locator;
  35 + readonly customerGroupMenu: Locator;
  36 +
  37 + // 操作按钮
  38 + readonly newButton: Locator;
  39 + readonly confirmButton: Locator;
  40 + readonly saveButton: Locator;
  41 + readonly deleteButton: Locator;
  42 + readonly editButton: Locator;
  43 +
  44 + // 表单字段
  45 + readonly groupNameInput: Locator;
  46 + readonly sortInput: Locator;
  47 +
  48 + // 搜索
  49 + readonly searchInput: Locator;
  50 + readonly clearSearchButton: Locator;
  51 +
  52 + constructor(page: Page) {
  53 + super(page);
  54 +
  55 + this.moreMenu = page.getByText('更多 >');
  56 + this.customerGroupMenu = page.getByText('客户分组').first();
  57 + this.newButton = page.getByText('新建');
  58 + this.confirmButton = page.getByText('确定');
  59 + this.saveButton = page.getByText('保存');
  60 + this.deleteButton = page.getByText('删除');
  61 + this.editButton = page.getByText('编辑');
  62 + this.groupNameInput = page.locator('uni-view').filter({ hasText: /^分组名称\*$/ }).first();
  63 + this.sortInput = page.getByRole('textbox').nth(2);
  64 + this.searchInput = page.getByRole('textbox');
  65 + this.clearSearchButton = page.locator('.nut-searchbar__search-icon');
  66 + }
  67 +
  68 + /**
  69 + * 进入客户分组页面
  70 + */
  71 + async openCustomerGroupManagement(): Promise<void> {
  72 + await this.navigate('/');
  73 + await this.moreMenu.click();
  74 + await this.page.waitForTimeout(500);
  75 + await this.customerGroupMenu.click();
  76 + await this.page.waitForLoadState('networkidle', { timeout: 30000 });
  77 + }
  78 +
  79 + /**
  80 + * 点击新建按钮
  81 + */
  82 + async clickNewButton(): Promise<void> {
  83 + await this.newButton.click();
  84 + await this.page.waitForTimeout(500);
  85 + }
  86 +
  87 + /**
  88 + * 填写分组名称
  89 + * @param name 分组名称
  90 + */
  91 + async fillGroupName(name: string): Promise<void> {
  92 + await this.page.getByRole('textbox').nth(1).click();
  93 + await this.page.getByRole('textbox').nth(1).fill(name);
  94 + }
  95 +
  96 + /**
  97 + * 填写排序号
  98 + * @param sort 排序号
  99 + */
  100 + async fillSortNumber(sort: string): Promise<void> {
  101 + await this.page.getByRole('textbox').nth(2).click();
  102 + await this.page.getByRole('textbox').nth(2).fill(sort);
  103 + }
  104 +
  105 + /**
  106 + * 点击确定按钮
  107 + */
  108 + async clickConfirm(): Promise<void> {
  109 + await this.confirmButton.click();
  110 + await this.page.waitForTimeout(1000);
  111 + }
  112 +
  113 + /**
  114 + * 搜索分组
  115 + * @param name 分组名称
  116 + */
  117 + async searchGroup(name: string): Promise<void> {
  118 + await this.searchInput.click();
  119 + await this.searchInput.fill(name);
  120 + await this.page.waitForTimeout(500);
  121 + }
  122 +
  123 + /**
  124 + * 点击清除搜索按钮
  125 + */
  126 + async clearSearch(): Promise<void> {
  127 + await this.clearSearchButton.click();
  128 + await this.page.waitForTimeout(2000);
  129 + }
  130 +
  131 + /**
  132 + * 点击编辑按钮
  133 + */
  134 + async clickEdit(): Promise<void> {
  135 + await this.editButton.click();
  136 + await this.page.waitForTimeout(500);
  137 + }
  138 +
  139 + /**
  140 + * 点击删除按钮
  141 + */
  142 + async clickDelete(): Promise<void> {
  143 + await this.deleteButton.click();
  144 + await this.page.waitForTimeout(500);
  145 + }
  146 +
  147 + /**
  148 + * 验证分组是否存在(验证前先搜索)
  149 + * @param name 分组名称
  150 + * @returns 是否存在
  151 + */
  152 + async verifyGroupExists(name: string): Promise<boolean> {
  153 + await this.searchGroup(name);
  154 + await this.page.waitForTimeout(500);
  155 + const count = await this.page.locator('uni-view').filter({ hasText: new RegExp(`^${name}$`) }).count();
  156 + return count > 0;
  157 + }
  158 +
  159 + /**
  160 + * 验证分组是否不存在
  161 + * @param name 分组名称
  162 + * @returns 是否不存在
  163 + */
  164 + async verifyGroupNotExists(name: string): Promise<boolean> {
  165 + const count = await this.page.locator('uni-view').filter({ hasText: new RegExp(`^${name}$`) }).count();
  166 + return count === 0;
  167 + }
  168 +
  169 + /**
  170 + * 新增客户分组完整流程
  171 + * @param groupName 分组名称
  172 + * @param sortNumber 排序号(可选)
  173 + */
  174 + async createGroup(groupName: string, sortNumber?: string): Promise<void> {
  175 + await this.openCustomerGroupManagement();
  176 + await this.clickNewButton();
  177 + await this.fillGroupName(groupName);
  178 + if (sortNumber) {
  179 + await this.fillSortNumber(sortNumber);
  180 + }
  181 + await this.clickConfirm();
  182 + }
  183 +
  184 + /**
  185 + * 修改客户分组完整流程
  186 + * @param originalName 原分组名称
  187 + * @param newName 新分组名称
  188 + * @param newSortNumber 新排序号(可选)
  189 + */
  190 + async updateGroup(originalName: string, newName: string, newSortNumber?: string): Promise<void> {
  191 + await this.searchGroup(originalName);
  192 + await this.page.locator('uni-view').filter({ hasText: new RegExp(`^${originalName}$`) }).first().click();
  193 + await this.page.waitForTimeout(500);
  194 + await this.clickEdit();
  195 + // 清空分组名称并填写新名称
  196 + await this.page.getByRole('textbox').nth(1).click();
  197 + await this.page.locator('.nut-input__clear-icon').first().click();
  198 + await this.page.getByRole('textbox').nth(1).click();
  199 + await this.page.getByRole('textbox').nth(1).fill(newName);
  200 + if (newSortNumber) {
  201 + await this.page.getByRole('textbox').nth(2).click();
  202 + await this.page.getByRole('textbox').nth(2).fill(newSortNumber);
  203 + }
  204 + await this.clickConfirm();
  205 + }
  206 +
  207 + /**
  208 + * 删除客户分组完整流程
  209 + * @param groupName 分组名称
  210 + */
  211 + async deleteGroup(groupName: string): Promise<void> {
  212 + await this.searchGroup(groupName);
  213 + await this.page.locator('uni-view').filter({ hasText: new RegExp(`^${groupName}$`) }).first().click();
  214 + await this.page.waitForTimeout(500);
  215 + await this.clickDelete();
  216 + await this.page.getByText('确定', { exact: true }).click();
  217 + await this.page.waitForTimeout(1000);
  218 + await this.clearSearch();
  219 + }
  220 +}
scripts/save-auth.ts
@@ -7,7 +7,7 @@ dotenv.config({ path: path.resolve(__dirname, &#39;../.env&#39;) }); @@ -7,7 +7,7 @@ dotenv.config({ path: path.resolve(__dirname, &#39;../.env&#39;) });
7 7
8 const TEST_PHONE = process.env.phone; 8 const TEST_PHONE = process.env.phone;
9 const TEST_USER_NAME = process.env.TEST_USER_NAME; 9 const TEST_USER_NAME = process.env.TEST_USER_NAME;
10 -const BASE_URL = process.env.BASE_URL || 'https://erp-pad.test.gszdtop.com'; 10 +const BASE_URL = process.env.BASE_URL ;
11 11
12 if (!TEST_PHONE || !TEST_USER_NAME) { 12 if (!TEST_PHONE || !TEST_USER_NAME) {
13 throw new Error('phone 和 TEST_USER_NAME 环境变量未设置'); 13 throw new Error('phone 和 TEST_USER_NAME 环境变量未设置');
tests/account.spec.ts 0 → 100644
  1 +import { test, expect } from '@playwright/test';
  2 +import * as allure from 'allure-js-commons';
  3 +import { AccountPage } from '../pages/accountPage';
  4 +
  5 +/**
  6 + * 账目管理测试
  7 + */
  8 +test.describe('账目管理', () => {
  9 + // 使用已保存的认证状态
  10 + test.use({ storageState: 'auth.json' });
  11 +
  12 + // 强制测试串行执行
  13 + test.describe.configure({ mode: 'serial' });
  14 +
  15 + /**
  16 + * 生成随机账目名称(三个字+"费",带时间戳后缀防止重复)
  17 + */
  18 + function generateAccountName(): string {
  19 + const prefixes = ['管理', '维护', '看管', '保管', '仓储', '代理', '服务', '咨询', '技术', '运营'];
  20 + const prefix = prefixes[Math.floor(Math.random() * prefixes.length)];
  21 + const timestamp = Date.now().toString().slice(-4);
  22 + return `${prefix}费${timestamp}`;
  23 + }
  24 +
  25 + /**
  26 + * 生成随机收支类型
  27 + */
  28 + function generateIncomeType(): '收入' | '支出' {
  29 + return Math.random() > 0.5 ? '收入' : '支出';
  30 + }
  31 +
  32 + /**
  33 + * 生成包含"自动化"的备注
  34 + */
  35 + function generateRemark(): string {
  36 + const timestamp = Date.now().toString().slice(-6);
  37 + return `自动化测试备注${timestamp}`;
  38 + }
  39 +
  40 + test('新增账目', async ({ page }, testInfo) => {
  41 + const accountPage = new AccountPage(page);
  42 +
  43 + // 添加allure元素
  44 + await allure.epic('账目管理');
  45 + await allure.feature('账目信息');
  46 + await allure.story('新增账目');
  47 +
  48 + // 生成随机账目数据
  49 + const accountName = generateAccountName();
  50 + const incomeType = generateIncomeType();
  51 + const remark = generateRemark();
  52 +
  53 + console.log('账目名称:', accountName);
  54 + console.log('收支类型:', incomeType);
  55 + console.log('备注:', remark);
  56 +
  57 + // 步骤1:进入账目管理页面
  58 + await allure.step('进入账目管理页面', async () => {
  59 + await accountPage.openAccountManagement();
  60 + });
  61 +
  62 + // 步骤2:点击新增按钮
  63 + await allure.step('点击新增按钮', async () => {
  64 + await accountPage.clickAddButton();
  65 + });
  66 +
  67 + // 步骤3:填写账目表单
  68 + await allure.step('填写账目表单', async () => {
  69 + await accountPage.fillAccountName(accountName);
  70 + await accountPage.selectIncomeType(incomeType);
  71 + await accountPage.fillRemark(remark);
  72 + });
  73 +
  74 + // 步骤4:保存账目
  75 + await allure.step('保存账目', async () => {
  76 + await accountPage.saveAccount();
  77 + });
  78 +
  79 + // 步骤5:验证账目创建成功
  80 + await allure.step('验证账目创建成功', async () => {
  81 + await accountPage.searchAccount(accountName);
  82 + const isCreated = await accountPage.verifyAccountExists(accountName);
  83 + expect(isCreated).toBeTruthy();
  84 + });
  85 + });
  86 +
  87 + test('删除账目', async ({ page }, testInfo) => {
  88 + const accountPage = new AccountPage(page);
  89 +
  90 + // 添加allure元素
  91 + await allure.epic('账目管理');
  92 + await allure.feature('账目信息');
  93 + await allure.story('删除账目');
  94 +
  95 + // 生成随机账目数据
  96 + const accountName = generateAccountName();
  97 + console.log('待删除账目名称:', accountName);
  98 +
  99 + // 步骤1:新增账目
  100 + await allure.step('新增账目', async () => {
  101 + await accountPage.openAccountManagement();
  102 + await accountPage.clickAddButton();
  103 + await accountPage.fillAccountName(accountName);
  104 + await accountPage.saveAccount();
  105 + });
  106 +
  107 + // 步骤2:删除账目
  108 + await allure.step('删除账目', async () => {
  109 + await accountPage.deleteAccount(accountName);
  110 + });
  111 +
  112 + // 步骤3:验证账目删除成功
  113 + await allure.step('验证账目删除成功', async () => {
  114 + const isDeleted = await accountPage.verifyAccountDeleted(accountName);
  115 + expect(isDeleted).toBeTruthy();
  116 + });
  117 + });
  118 +
  119 + test('修改账目', async ({ page }, testInfo) => {
  120 + const accountPage = new AccountPage(page);
  121 +
  122 + // 添加allure元素
  123 + await allure.epic('账目管理');
  124 + await allure.feature('账目信息');
  125 + await allure.story('修改账目');
  126 +
  127 + // 生成随机账目数据
  128 + const originalName = generateAccountName();
  129 + const newName = generateAccountName();
  130 + const incomeType = generateIncomeType();
  131 + const remark = generateRemark();
  132 +
  133 + console.log('原账目名称:', originalName);
  134 + console.log('新账目名称:', newName);
  135 + console.log('收支类型:', incomeType);
  136 + console.log('备注:', remark);
  137 +
  138 + // 步骤1:新增账目
  139 + await allure.step('新增账目', async () => {
  140 + await accountPage.openAccountManagement();
  141 + await accountPage.clickAddButton();
  142 + await accountPage.fillAccountName(originalName);
  143 + await accountPage.saveAccount();
  144 + });
  145 +
  146 + // 步骤2:修改账目
  147 + await allure.step('修改账目', async () => {
  148 + await accountPage.updateAccount(originalName, newName, incomeType, remark);
  149 + });
  150 +
  151 + // 步骤3:验证账目修改成功
  152 + await allure.step('验证账目修改成功', async () => {
  153 + // 清除搜索框
  154 + await accountPage.clearSearchBox();
  155 + // 搜索修改后的账目名称
  156 + await accountPage.searchAccount(newName);
  157 + const isExists = await accountPage.verifyAccountExists(newName);
  158 + expect(isExists).toBeTruthy();
  159 + });
  160 + });
  161 +});
tests/customerGroup.spec.ts
1 import { test, expect } from '@playwright/test'; 1 import { test, expect } from '@playwright/test';
2 import * as allure from 'allure-js-commons'; 2 import * as allure from 'allure-js-commons';
  3 +import { CustomerGroupPage } from '../pages/customerGroupPage';
3 4
4 /** 5 /**
5 * 客户分组测试 6 * 客户分组测试
@@ -32,6 +33,8 @@ test.describe(&#39;客户分组&#39;, () =&gt; { @@ -32,6 +33,8 @@ test.describe(&#39;客户分组&#39;, () =&gt; {
32 } 33 }
33 34
34 test('新增客户分组', async ({ page }, testInfo) => { 35 test('新增客户分组', async ({ page }, testInfo) => {
  36 + const customerGroupPage = new CustomerGroupPage(page);
  37 +
35 // 添加allure元素 38 // 添加allure元素
36 await allure.epic('客户管理'); 39 await allure.epic('客户管理');
37 await allure.feature('客户分组'); 40 await allure.feature('客户分组');
@@ -43,47 +46,34 @@ test.describe(&#39;客户分组&#39;, () =&gt; { @@ -43,47 +46,34 @@ test.describe(&#39;客户分组&#39;, () =&gt; {
43 46
44 // 步骤1:进入客户分组页面 47 // 步骤1:进入客户分组页面
45 await allure.step('进入客户分组页面', async () => { 48 await allure.step('进入客户分组页面', async () => {
46 - await page.goto('/');  
47 - await page.waitForLoadState('networkidle', { timeout: 30000 });  
48 - await page.getByText('更多 >').click();  
49 - await page.waitForTimeout(500);  
50 - await page.getByText('客户分组').first().click();  
51 - await page.waitForLoadState('networkidle', { timeout: 30000 }); 49 + await customerGroupPage.openCustomerGroupManagement();
52 }); 50 });
53 51
54 // 步骤2:点击新建按钮 52 // 步骤2:点击新建按钮
55 await allure.step('点击新建按钮', async () => { 53 await allure.step('点击新建按钮', async () => {
56 - await page.getByText('新建').click();  
57 - await page.waitForTimeout(500); 54 + await customerGroupPage.clickNewButton();
58 }); 55 });
59 56
60 // 步骤3:填写分组信息 57 // 步骤3:填写分组信息
61 await allure.step('填写分组信息', async () => { 58 await allure.step('填写分组信息', async () => {
62 - // 填写分组名称  
63 - await page.getByRole('textbox').nth(1).click();  
64 - await page.getByRole('textbox').nth(1).fill(groupName);  
65 -  
66 - // 填写备注/排序  
67 - await page.getByRole('textbox').nth(2).click();  
68 - await page.getByRole('textbox').nth(2).fill('自动化测试分组'); 59 + await customerGroupPage.fillGroupName(groupName);
69 }); 60 });
70 61
71 // 步骤4:保存分组 62 // 步骤4:保存分组
72 await allure.step('保存分组', async () => { 63 await allure.step('保存分组', async () => {
73 - await page.getByText('确定').click();  
74 - await page.waitForTimeout(1000); 64 + await customerGroupPage.clickConfirm();
75 }); 65 });
76 66
77 - // 步骤5:验证分组创建成功 - 使用正则匹配检查页面是否出现新增的内容 67 + // 步骤5:验证分组创建成功
78 await allure.step('验证分组创建成功', async () => { 68 await allure.step('验证分组创建成功', async () => {
79 - await page.waitForTimeout(1000); // 等待页面刷新  
80 - // 使用正则匹配分组名称  
81 - const isGroupVisible = await page.locator('uni-view').filter({ hasText: new RegExp(`^${groupName}$`) }).first().isVisible({ timeout: 5000 }).catch(() => false);  
82 - expect(isGroupVisible).toBeTruthy(); 69 + const isExists = await customerGroupPage.verifyGroupExists(groupName);
  70 + expect(isExists).toBeTruthy();
83 }); 71 });
84 }); 72 });
85 73
86 test('修改客户分组', async ({ page }, testInfo) => { 74 test('修改客户分组', async ({ page }, testInfo) => {
  75 + const customerGroupPage = new CustomerGroupPage(page);
  76 +
87 // 添加allure元素 77 // 添加allure元素
88 await allure.epic('客户管理'); 78 await allure.epic('客户管理');
89 await allure.feature('客户分组'); 79 await allure.feature('客户分组');
@@ -92,71 +82,30 @@ test.describe(&#39;客户分组&#39;, () =&gt; { @@ -92,71 +82,30 @@ test.describe(&#39;客户分组&#39;, () =&gt; {
92 // 先生成一个分组用于修改 82 // 先生成一个分组用于修改
93 const originalGroupName = generateGroupName(); 83 const originalGroupName = generateGroupName();
94 const newGroupName = generateGroupName(); 84 const newGroupName = generateGroupName();
  85 + const newSortNumber = generateSortNumber();
95 console.log('原分组名称:', originalGroupName); 86 console.log('原分组名称:', originalGroupName);
96 console.log('新分组名称:', newGroupName); 87 console.log('新分组名称:', newGroupName);
97 88
98 - // 步骤1:进入客户分组页面并创建分组  
99 - await allure.step('进入客户分组页面并创建分组', async () => {  
100 - await page.goto('/');  
101 - await page.waitForLoadState('networkidle', { timeout: 30000 });  
102 - await page.getByText('更多 >').click();  
103 - await page.waitForTimeout(500);  
104 - await page.getByText('客户分组').first().click();  
105 - await page.waitForLoadState('networkidle', { timeout: 30000 });  
106 -  
107 - // 创建分组  
108 - await page.getByText('新建').click();  
109 - await page.waitForTimeout(500);  
110 - // 点击分组名称输入框  
111 - await page.locator('uni-view').filter({ hasText: /^分组名称\*$/ }).first().click();  
112 - await page.getByRole('textbox').nth(1).fill(originalGroupName);  
113 - await page.getByText('确定').click();  
114 - await page.waitForTimeout(1000); 89 + // 步骤1:先创建分组
  90 + await allure.step('创建待修改的分组', async () => {
  91 + await customerGroupPage.createGroup(originalGroupName);
115 }); 92 });
116 93
117 - // 步骤2:搜索并修改分组  
118 - await allure.step('搜索并修改分组', async () => {  
119 - // 在搜索框中输入原分组名称  
120 - await page.getByRole('textbox').click();  
121 - await page.getByRole('textbox').fill(originalGroupName);  
122 -  
123 - // 点击搜索结果中的分组  
124 - await page.locator('uni-view').filter({ hasText: new RegExp(`^${originalGroupName}$`) }).first().click();  
125 - await page.waitForTimeout(500);  
126 -  
127 - // 点击编辑按钮  
128 - await page.getByText('编辑').click();  
129 - await page.waitForTimeout(500);  
130 -  
131 - // 清空分组名称并填写新名称  
132 - await page.getByRole('textbox').nth(1).click();  
133 - await page.locator('.nut-input__clear-icon').first().click();  
134 - await page.getByRole('textbox').nth(1).click();  
135 - await page.getByRole('textbox').nth(1).fill(newGroupName);  
136 -  
137 - // 直接填写新排序号(新增时未填写,无需清除)  
138 - const newSortNumber = generateSortNumber();  
139 - await page.getByRole('textbox').nth(2).click();  
140 - await page.getByRole('textbox').nth(2).fill(newSortNumber);  
141 -  
142 - await page.getByText('确定').click();  
143 - await page.waitForTimeout(1000); 94 + // 步骤2:修改分组
  95 + await allure.step('修改分组', async () => {
  96 + await customerGroupPage.updateGroup(originalGroupName, newGroupName, newSortNumber);
144 }); 97 });
145 98
146 - // 步骤3:验证分组修改成功 - 使用搜索验证修改后的内容 99 + // 步骤3:验证分组修改成功
147 await allure.step('验证分组修改成功', async () => { 100 await allure.step('验证分组修改成功', async () => {
148 - // 在搜索框中输入新分组名称进行验证  
149 - await page.waitForTimeout(500);  
150 - await page.getByRole('textbox').click();  
151 - await page.getByRole('textbox').fill(newGroupName);  
152 - await page.waitForTimeout(500);  
153 -  
154 - const isNewGroupVisible = await page.locator('uni-view').filter({ hasText: new RegExp(`^${newGroupName}$`) }).first().isVisible({ timeout: 5000 }).catch(() => false);  
155 - expect(isNewGroupVisible).toBeTruthy(); 101 + const isExists = await customerGroupPage.verifyGroupExists(newGroupName);
  102 + expect(isExists).toBeTruthy();
156 }); 103 });
157 }); 104 });
158 105
159 test('删除客户分组', async ({ page }, testInfo) => { 106 test('删除客户分组', async ({ page }, testInfo) => {
  107 + const customerGroupPage = new CustomerGroupPage(page);
  108 +
160 // 添加allure元素 109 // 添加allure元素
161 await allure.epic('客户管理'); 110 await allure.epic('客户管理');
162 await allure.feature('客户分组'); 111 await allure.feature('客户分组');
@@ -166,53 +115,20 @@ test.describe(&#39;客户分组&#39;, () =&gt; { @@ -166,53 +115,20 @@ test.describe(&#39;客户分组&#39;, () =&gt; {
166 const groupName = generateGroupName(); 115 const groupName = generateGroupName();
167 console.log('待删除分组名称:', groupName); 116 console.log('待删除分组名称:', groupName);
168 117
169 - // 步骤1:进入客户分组页面并创建分组  
170 - await allure.step('进入客户分组页面并创建分组', async () => {  
171 - await page.goto('/');  
172 - await page.waitForLoadState('networkidle', { timeout: 30000 });  
173 - await page.getByText('更多 >').click();  
174 - await page.waitForTimeout(500);  
175 - await page.getByText('客户分组').first().click();  
176 - await page.waitForLoadState('networkidle', { timeout: 30000 });  
177 -  
178 - // 创建分组  
179 - await page.getByText('新建').click();  
180 - await page.waitForTimeout(500);  
181 - await page.getByRole('textbox').nth(1).click();  
182 - await page.getByRole('textbox').nth(1).fill(groupName);  
183 - await page.getByText('确定').click();  
184 - await page.waitForTimeout(1000); 118 + // 步骤1:先创建分组
  119 + await allure.step('创建待删除的分组', async () => {
  120 + await customerGroupPage.createGroup(groupName);
185 }); 121 });
186 122
187 - // 步骤2:搜索并删除分组  
188 - await allure.step('搜索并删除分组', async () => {  
189 - // 在搜索框中输入分组名称  
190 - await page.getByRole('textbox').click();  
191 - await page.getByRole('textbox').fill(groupName);  
192 -  
193 - // 搜索后自动选中,直接点击删除按钮  
194 - await page.getByText('删除').click();  
195 - await page.waitForTimeout(500);  
196 -  
197 - // 确认删除  
198 - await page.getByText('确定', { exact: true }).click();  
199 - // await page.waitForTimeout(1000);  
200 -  
201 - // 点击清除按钮  
202 - await page.locator('.nut-searchbar__search-icon').click();  
203 - // 清除后等待页面刷新  
204 - await page.waitForTimeout(1000); 123 + // 步骤2:删除分组
  124 + await allure.step('删除分组', async () => {
  125 + await customerGroupPage.deleteGroup(groupName);
205 }); 126 });
206 127
207 // 步骤3:验证分组删除成功 128 // 步骤3:验证分组删除成功
208 await allure.step('验证分组删除成功', async () => { 129 await allure.step('验证分组删除成功', async () => {
209 - // 搜索已删除的分组名称  
210 - await page.getByRole('textbox').fill(groupName);  
211 - await page.waitForTimeout(1000);  
212 -  
213 - // 验证搜索结果中是否还能找到该分组(找不到才是删除成功)  
214 - const count = await page.locator('uni-view').filter({ hasText: new RegExp(`^${groupName}$`) }).count();  
215 - expect(count).toBe(0); 130 + const isNotExists = await customerGroupPage.verifyGroupNotExists(groupName);
  131 + expect(isNotExists).toBeTruthy();
216 }); 132 });
217 }); 133 });
218 }); 134 });