enterprise-mini.page.ts 10 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337
  1. import { TIMEOUTS } from '../../utils/timeouts';
  2. import { Page, Locator, expect } from '@playwright/test';
  3. /**
  4. * 企业小程序 H5 URL
  5. */
  6. const MINI_BASE_URL = process.env.E2E_BASE_URL || 'http://localhost:8080';
  7. const MINI_LOGIN_URL = `${MINI_BASE_URL}/mini`;
  8. /**
  9. * 企业小程序 Page Object
  10. *
  11. * 用于企业小程序 E2E 测试
  12. * H5 页面路径: /mini
  13. *
  14. * 主要功能:
  15. * - 小程序登录(手机号 + 密码)
  16. * - Token 管理
  17. * - 页面导航和验证
  18. *
  19. * @example
  20. * ```typescript
  21. * const miniPage = new EnterpriseMiniPage(page);
  22. * await miniPage.goto();
  23. * await miniPage.login('13800138000', 'password123');
  24. * await miniPage.expectLoginSuccess();
  25. * ```
  26. */
  27. export class EnterpriseMiniPage {
  28. readonly page: Page;
  29. // ===== 页面级选择器 =====
  30. /** 登录页面容器 */
  31. readonly loginPage: Locator;
  32. /** 页面标题 */
  33. readonly pageTitle: Locator;
  34. // ===== 登录表单选择器 =====
  35. /** 手机号输入框 */
  36. readonly phoneInput: Locator;
  37. /** 密码输入框 */
  38. readonly passwordInput: Locator;
  39. /** 登录按钮 */
  40. readonly loginButton: Locator;
  41. // ===== 主页选择器(登录后) =====
  42. /** 用户信息显示区域 */
  43. readonly userInfo: Locator;
  44. /** 退出登录按钮 */
  45. readonly logoutButton: Locator;
  46. constructor(page: Page) {
  47. this.page = page;
  48. // 初始化登录页面选择器
  49. // Taro 组件在 H5 渲染时会传递 data-testid 到 DOM (使用 taro-view-core 等组件)
  50. this.loginPage = page.getByTestId('mini-login-page');
  51. this.pageTitle = page.getByTestId('mini-page-title');
  52. // 登录表单选择器 - 使用 data-testid
  53. this.phoneInput = page.getByTestId('mini-phone-input');
  54. this.passwordInput = page.getByTestId('mini-password-input');
  55. this.loginButton = page.getByTestId('mini-login-button');
  56. // 主页选择器(登录后可用)
  57. this.userInfo = page.getByTestId('mini-user-info');
  58. // 退出登录按钮 - 使用 getByText 而非 getByRole
  59. this.logoutButton = page.getByText('退出登录');
  60. }
  61. // ===== 导航和基础验证 =====
  62. /**
  63. * 移除开发服务器的覆盖层 iframe(防止干扰测试)
  64. */
  65. private async removeDevOverlays(): Promise<void> {
  66. await this.page.evaluate(() => {
  67. // 移除 react-refresh-overlay 和 webpack-dev-server-client-overlay
  68. const overlays = document.querySelectorAll('#react-refresh-overlay, #webpack-dev-server-client-overlay');
  69. overlays.forEach(overlay => overlay.remove());
  70. // 移除 vConsole 开发者工具覆盖层
  71. const vConsole = document.querySelector('#__vconsole');
  72. if (vConsole) {
  73. vConsole.remove();
  74. }
  75. });
  76. }
  77. /**
  78. * 导航到企业小程序 H5 登录页面
  79. */
  80. async goto(): Promise<void> {
  81. await this.page.goto(MINI_LOGIN_URL);
  82. // 移除开发服务器的覆盖层
  83. await this.removeDevOverlays();
  84. // 使用 auto-waiting 机制,等待页面容器可见
  85. await this.expectToBeVisible();
  86. }
  87. /**
  88. * 验证登录页面关键元素可见
  89. */
  90. async expectToBeVisible(): Promise<void> {
  91. // 等待页面容器可见
  92. await this.loginPage.waitFor({ state: 'visible', timeout: TIMEOUTS.PAGE_LOAD });
  93. // 验证页面标题
  94. await this.pageTitle.waitFor({ state: 'visible', timeout: TIMEOUTS.ELEMENT_VISIBLE_SHORT });
  95. }
  96. // ===== 登录功能方法 =====
  97. /**
  98. * 填写手机号
  99. * @param phone 手机号(11位数字)
  100. *
  101. * 注意:使用 click + type 方法触发自然的用户输入事件
  102. * Taro Input 组件需要完整的事件流才能正确更新 react-hook-form 状态
  103. */
  104. async fillPhone(phone: string): Promise<void> {
  105. // 先移除覆盖层,确保输入可操作
  106. await this.removeDevOverlays();
  107. // 点击聚焦,然后清空(使用 Ctrl+A + Backspace 模拟用户操作)
  108. await this.phoneInput.click();
  109. // 等待元素聚焦
  110. await this.page.waitForTimeout(100);
  111. // 使用 type 方法输入,会自动覆盖现有内容
  112. await this.phoneInput.type(phone, { delay: 50 });
  113. // 等待表单验证更新
  114. await this.page.waitForTimeout(200);
  115. }
  116. /**
  117. * 填写密码
  118. * @param password 密码(6-20位)
  119. *
  120. * 注意:使用 click + type 方法触发自然的用户输入事件
  121. * Taro Input 组件需要完整的事件流才能正确更新 react-hook-form 状态
  122. */
  123. async fillPassword(password: string): Promise<void> {
  124. // 先移除覆盖层,确保输入可操作
  125. await this.removeDevOverlays();
  126. // 点击聚焦
  127. await this.passwordInput.click();
  128. // 等待元素聚焦
  129. await this.page.waitForTimeout(100);
  130. // 使用 type 方法输入
  131. await this.passwordInput.type(password, { delay: 50 });
  132. // 等待表单验证更新
  133. await this.page.waitForTimeout(200);
  134. }
  135. /**
  136. * 点击登录按钮
  137. */
  138. async clickLoginButton(): Promise<void> {
  139. // 使用 force: true 避免被开发服务器的覆盖层阻止
  140. await this.loginButton.click({ force: true });
  141. }
  142. /**
  143. * 执行登录操作(完整流程)
  144. * @param phone 手机号
  145. * @param password 密码
  146. */
  147. async login(phone: string, password: string): Promise<void> {
  148. await this.fillPhone(phone);
  149. await this.fillPassword(password);
  150. await this.clickLoginButton();
  151. }
  152. /**
  153. * 验证登录成功
  154. *
  155. * 登录成功后应该跳转到主页或显示用户信息
  156. */
  157. async expectLoginSuccess(): Promise<void> {
  158. // 使用 auto-waiting 机制,等待 URL 变化或用户信息显示
  159. // 小程序登录成功后会跳转到 dashboard 页面
  160. // 等待 URL 变化,使用 Promise.race 实现超时
  161. await this.page.waitForURL(
  162. url => url.pathname.includes('/dashboard') || url.pathname.includes('/pages/yongren/dashboard'),
  163. { timeout: TIMEOUTS.PAGE_LOAD }
  164. ).catch(() => {
  165. // 如果没有跳转,检查是否显示用户信息
  166. // 注意:此验证将在 Story 12.5 E2E 测试中完全实现
  167. // 当前仅提供基础结构
  168. });
  169. }
  170. /**
  171. * 验证登录失败(错误提示显示)
  172. * @param expectedErrorMessage 预期的错误消息(可选)
  173. */
  174. async expectLoginError(expectedErrorMessage?: string): Promise<void> {
  175. // 等待一下,让后端响应或前端验证生效
  176. await this.page.waitForTimeout(1000);
  177. // 验证仍然在登录页面(未跳转)
  178. const currentUrl = this.page.url();
  179. expect(currentUrl).toContain('/mini');
  180. // 验证登录页面容器仍然可见
  181. await expect(this.loginPage).toBeVisible();
  182. // 如果提供了预期的错误消息,尝试验证
  183. if (expectedErrorMessage) {
  184. // 尝试查找错误消息(可能在 Toast、Modal 或表单验证中)
  185. const errorElement = this.page.getByText(expectedErrorMessage, { exact: false }).first();
  186. await errorElement.isVisible().catch(() => false);
  187. // 不强制要求错误消息可见,因为后端可能不会返回错误
  188. }
  189. }
  190. // ===== Token 管理方法 =====
  191. /**
  192. * 获取当前存储的 token
  193. * @returns token 字符串,如果不存在则返回 null
  194. *
  195. * 注意:Taro.getStorageSync 在 H5 环境下映射到 localStorage
  196. * token 直接存储为字符串,不是 JSON 格式
  197. *
  198. * Taro H5 可能使用以下键名格式:
  199. * - 直接键名: 'enterprise_token'
  200. * - 带前缀: 'taro_app_storage_key'
  201. * - 或者其他变体
  202. */
  203. async getToken(): Promise<string | null> {
  204. const result = await this.page.evaluate(() => {
  205. // 获取所有 localStorage 键
  206. const keys = Object.keys(localStorage);
  207. const storage: Record<string, string> = {};
  208. keys.forEach(k => storage[k] = localStorage.getItem(k) || '');
  209. // 尝试各种可能的键名
  210. // 1. 直接键名
  211. const token = localStorage.getItem('enterprise_token');
  212. if (token) return token;
  213. // 2. 带前缀的键名(Taro 可能使用前缀)
  214. const prefixedKeys = keys.filter(k => k.includes('token') || k.includes('auth'));
  215. for (const key of prefixedKeys) {
  216. const value = localStorage.getItem(key);
  217. if (value && value.length > 20) { // JWT token 通常很长
  218. return value;
  219. }
  220. }
  221. // 3. 其他常见键名
  222. return (
  223. localStorage.getItem('token') ||
  224. localStorage.getItem('auth_token') ||
  225. sessionStorage.getItem('token') ||
  226. sessionStorage.getItem('auth_token') ||
  227. null
  228. );
  229. });
  230. return result;
  231. }
  232. /**
  233. * 设置 token(用于测试前置条件)
  234. * @param token token 字符串
  235. */
  236. async setToken(token: string): Promise<void> {
  237. await this.page.evaluate((t) => {
  238. localStorage.setItem('token', t);
  239. localStorage.setItem('auth_token', t);
  240. }, token);
  241. }
  242. /**
  243. * 清除所有认证相关的存储
  244. */
  245. async clearAuth(): Promise<void> {
  246. await this.page.evaluate(() => {
  247. // 清除企业小程序相关的认证数据
  248. localStorage.removeItem('enterprise_token');
  249. localStorage.removeItem('enterpriseUserInfo');
  250. // 清除其他常见 token 键
  251. localStorage.removeItem('token');
  252. localStorage.removeItem('auth_token');
  253. sessionStorage.removeItem('token');
  254. sessionStorage.removeItem('auth_token');
  255. });
  256. }
  257. // ===== 主页元素验证方法 =====
  258. /**
  259. * 验证主页元素可见(登录后)
  260. * 根据实际小程序主页结构调整
  261. */
  262. async expectHomePageVisible(): Promise<void> {
  263. // 使用 auto-waiting 机制,等待主页元素可见
  264. // 注意:此方法将在 Story 12.5 E2E 测试中使用,当前仅提供基础结构
  265. // 根据实际小程序主页的 data-testid 调整
  266. const dashboard = this.page.getByTestId('mini-dashboard');
  267. await dashboard.waitFor({ state: 'visible', timeout: TIMEOUTS.PAGE_LOAD });
  268. }
  269. /**
  270. * 获取用户信息显示的文本
  271. * @returns 用户信息文本
  272. */
  273. async getUserInfoText(): Promise<string | null> {
  274. const userInfo = this.userInfo;
  275. const count = await userInfo.count();
  276. if (count === 0) {
  277. return null;
  278. }
  279. return await userInfo.textContent();
  280. }
  281. // ===== 退出登录方法 =====
  282. /**
  283. * 退出登录
  284. */
  285. async logout(): Promise<void> {
  286. // 点击退出登录按钮
  287. await this.logoutButton.click();
  288. // 等待页面加载完成
  289. await this.page.waitForLoadState('domcontentloaded');
  290. }
  291. /**
  292. * 验证已退出登录(返回登录页面)
  293. */
  294. async expectLoggedOut(): Promise<void> {
  295. // 验证返回到登录页面
  296. await this.loginPage.waitFor({ state: 'visible', timeout: TIMEOUTS.PAGE_LOAD });
  297. }
  298. }