You cannot select more than 25 topics
			Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
		
		
		
		
		
			
		
			
				
	
	
		
			234 lines
		
	
	
		
			7.1 KiB
		
	
	
	
		
			TypeScript
		
	
			
		
		
	
	
			234 lines
		
	
	
		
			7.1 KiB
		
	
	
	
		
			TypeScript
		
	
| /* eslint-disable no-useless-escape */
 | |
| /* eslint-disable import/no-extraneous-dependencies */
 | |
| /* eslint-disable no-await-in-loop */
 | |
| import { ElementHandle, Page } from '@playwright/test';
 | |
| import { sleepFor } from '../../../session/utils/Promise';
 | |
| import { loaderType, Strategy } from '../types/testing';
 | |
| 
 | |
| // WAIT FOR FUNCTIONS
 | |
| 
 | |
| export async function waitForTestIdWithText(window: Page, dataTestId: string, text?: string) {
 | |
|   let builtSelector = `css=[data-testid=${dataTestId}]`;
 | |
|   if (text) {
 | |
|     // " =>  \\\"
 | |
|     /* prettier-ignore */
 | |
| 
 | |
|     const escapedText = text.replace(/"/g, '\\\"');
 | |
| 
 | |
|     builtSelector += `:has-text("${escapedText}")`;
 | |
|     console.warn('builtSelector:', builtSelector);
 | |
|     // console.warn('Text is tiny bubble: ', escapedText);
 | |
|   }
 | |
|   // console.info('looking for selector', builtSelector);
 | |
|   const found = await window.waitForSelector(builtSelector, { timeout: 55000 });
 | |
|   // console.info('found selector', builtSelector);
 | |
| 
 | |
|   return found;
 | |
| }
 | |
| 
 | |
| export async function waitForElement(
 | |
|   window: Page,
 | |
|   strategy: Strategy,
 | |
|   selector: string,
 | |
|   maxWaitMs?: number
 | |
| ) {
 | |
|   const builtSelector = `css=[${strategy}=${selector}]`;
 | |
| 
 | |
|   return window.waitForSelector(builtSelector, { timeout: maxWaitMs });
 | |
| }
 | |
| 
 | |
| export async function waitForTextMessage(window: Page, text: string, maxWait?: number) {
 | |
|   let builtSelector = `css=[data-testid=control-message]:has-text("${text}")`;
 | |
|   if (text) {
 | |
|     // " =>  \\\"
 | |
|     /* prettier-ignore */
 | |
| 
 | |
|     const escapedText = text.replace(/"/g, '\\\"');
 | |
| 
 | |
|     builtSelector += `:has-text("${escapedText}")`;
 | |
|     console.warn('builtSelector:', builtSelector);
 | |
|     // console.warn('Text is tiny bubble: ', escapedText);
 | |
|   }
 | |
|   const el = await window.waitForSelector(builtSelector, { timeout: maxWait });
 | |
|   console.info(`Text message found. Text: , ${text}`);
 | |
|   return el;
 | |
| }
 | |
| 
 | |
| export async function waitForControlMessageWithText(window: Page, text: string) {
 | |
|   return waitForTestIdWithText(window, 'control-message', text);
 | |
| }
 | |
| 
 | |
| export async function waitForMatchingText(window: Page, text: string, maxWait?: number) {
 | |
|   const builtSelector = `css=:has-text("${text}")`;
 | |
|   const maxTimeout = maxWait ?? 55000;
 | |
|   console.info(`waitForMatchingText: ${text}`);
 | |
| 
 | |
|   await window.waitForSelector(builtSelector, { timeout: maxTimeout });
 | |
| 
 | |
|   console.info(`got matchingText: ${text}`);
 | |
| }
 | |
| 
 | |
| export async function waitForMatchingPlaceholder(
 | |
|   window: Page,
 | |
|   dataTestId: string,
 | |
|   placeholder: string,
 | |
|   maxWait: number = 30000
 | |
| ) {
 | |
|   let found = false;
 | |
|   const start = Date.now();
 | |
|   console.info(`waitForMatchingPlaceholder: ${placeholder} with datatestId: ${dataTestId}`);
 | |
| 
 | |
|   do {
 | |
|     try {
 | |
|       const elem = await waitForElement(window, 'data-testid', dataTestId);
 | |
|       const elemPlaceholder = await elem.getAttribute('placeholder');
 | |
|       if (elemPlaceholder === placeholder) {
 | |
|         console.info(
 | |
|           `waitForMatchingPlaceholder foudn matching element with placeholder: "${placeholder}"`
 | |
|         );
 | |
| 
 | |
|         found = true;
 | |
|       }
 | |
|     } catch (e) {
 | |
|       await sleepFor(1000, true);
 | |
|       console.info(`waitForMatchingPlaceholder failed with ${e.message}, retrying in 1s`);
 | |
|     }
 | |
|   } while (!found && Date.now() - start <= maxWait);
 | |
| 
 | |
|   if (!found) {
 | |
|     throw new Error(`Failed to find datatestid:"${dataTestId}" with placeholder: "${placeholder}"`);
 | |
|   }
 | |
| }
 | |
| export async function waitForLoadingAnimationToFinish(
 | |
|   window: Page,
 | |
|   loader: loaderType,
 | |
|   maxWait?: number
 | |
| ) {
 | |
|   let loadingAnimation: ElementHandle<SVGElement | HTMLElement> | undefined;
 | |
| 
 | |
|   await waitForElement(window, 'data-testid', `${loader}`, maxWait);
 | |
| 
 | |
|   do {
 | |
|     try {
 | |
|       loadingAnimation = await waitForElement(window, 'data-testid', `${loader}`, 100);
 | |
|       await sleepFor(500);
 | |
|       console.info(`${loader} was found, waiting for it to be gone`);
 | |
|     } catch (e) {
 | |
|       loadingAnimation = undefined;
 | |
|     }
 | |
|   } while (loadingAnimation);
 | |
|   console.info('Loading animation has finished');
 | |
| }
 | |
| 
 | |
| // ACTIONS
 | |
| 
 | |
| export async function clickOnElement(
 | |
|   window: Page,
 | |
|   strategy: Strategy,
 | |
|   selector: string,
 | |
|   maxWait?: number
 | |
| ) {
 | |
|   const builtSelector = `css=[${strategy}=${selector}]`;
 | |
|   await window.waitForSelector(builtSelector, { timeout: maxWait });
 | |
|   await window.click(builtSelector);
 | |
| }
 | |
| 
 | |
| export async function clickOnMatchingText(window: Page, text: string, rightButton = false) {
 | |
|   console.info(`clickOnMatchingText: "${text}"`);
 | |
|   return window.click(`"${text}"`, rightButton ? { button: 'right' } : undefined);
 | |
| }
 | |
| 
 | |
| export async function clickOnTestIdWithText(
 | |
|   window: Page,
 | |
|   dataTestId: string,
 | |
|   text?: string,
 | |
|   rightButton?: boolean,
 | |
|   maxWait?: number
 | |
| ) {
 | |
|   console.info(`clickOnTestIdWithText with testId:${dataTestId} and text:${text || 'none'}`);
 | |
| 
 | |
|   const builtSelector = !text
 | |
|     ? `css=[data-testid=${dataTestId}]`
 | |
|     : `css=[data-testid=${dataTestId}]:has-text("${text}")`;
 | |
| 
 | |
|   await window.waitForSelector(builtSelector, { timeout: maxWait });
 | |
|   return window.click(builtSelector, rightButton ? { button: 'right' } : undefined);
 | |
| }
 | |
| 
 | |
| export function getMessageTextContentNow() {
 | |
|   return `Test message timestamp: ${Date.now()}`;
 | |
| }
 | |
| 
 | |
| export async function typeIntoInput(window: Page, dataTestId: string, text: string) {
 | |
|   console.info(`typeIntoInput testId: ${dataTestId} : "${text}"`);
 | |
|   const builtSelector = `css=[data-testid=${dataTestId}]`;
 | |
|   return window.fill(builtSelector, text);
 | |
| }
 | |
| 
 | |
| export async function typeIntoInputSlow(window: Page, dataTestId: string, text: string) {
 | |
|   console.info(`typeIntoInput testId: ${dataTestId} : "${text}"`);
 | |
|   const builtSelector = `css=[data-testid=${dataTestId}]`;
 | |
|   await window.waitForSelector(builtSelector);
 | |
|   return window.type(builtSelector, text, { delay: 100 });
 | |
| }
 | |
| 
 | |
| export async function hasTextElementBeenDeleted(window: Page, text: string, maxWait?: number) {
 | |
|   const fakeError = `Matching text: ${text} has been found... oops`;
 | |
|   try {
 | |
|     await waitForMatchingText(window, text, maxWait);
 | |
|     throw new Error(fakeError);
 | |
|   } catch (e) {
 | |
|     if (e.message === fakeError) {
 | |
|       throw e;
 | |
|     }
 | |
|   }
 | |
|   console.info('Element has not been found, congratulations', text);
 | |
| }
 | |
| 
 | |
| export async function doesTextIncludeString(window: Page, dataTestId: string, text: string) {
 | |
|   const element = await waitForTestIdWithText(window, dataTestId);
 | |
|   const el = await element.innerText();
 | |
| 
 | |
|   const builtSelector = el.includes(text);
 | |
|   if (builtSelector) {
 | |
|     console.info('Text found:', text);
 | |
|   } else {
 | |
|     throw new Error(`Text not found: "${text}"`);
 | |
|   }
 | |
| }
 | |
| 
 | |
| export async function hasElementBeenDeleted(
 | |
|   window: Page,
 | |
|   strategy: Strategy,
 | |
|   selector: string,
 | |
|   maxWait?: number
 | |
| ) {
 | |
|   const fakeError = `Element ${selector} has been found... oops`;
 | |
|   try {
 | |
|     await waitForElement(window, strategy, selector, maxWait);
 | |
|     throw new Error(fakeError);
 | |
|   } catch (e) {
 | |
|     if (e.message === fakeError) {
 | |
|       throw e;
 | |
|     }
 | |
|   }
 | |
|   console.info(`${selector} has not been found, congrats`);
 | |
| }
 | |
| 
 | |
| export async function hasElementPoppedUpThatShouldnt(
 | |
|   window: Page,
 | |
|   strategy: Strategy,
 | |
|   selector: string,
 | |
|   text?: string
 | |
| ) {
 | |
|   const builtSelector = !text
 | |
|     ? `css=[${strategy}=${selector}]`
 | |
|     : `css=[${strategy}=${selector}]:has-text("${text.replace(/"/g, '\\"')}")`;
 | |
| 
 | |
|   const fakeError = `Found ${selector}, oops..`;
 | |
|   const elVisible = await window.isVisible(builtSelector);
 | |
|   if (elVisible === true) {
 | |
|     throw new Error(fakeError);
 | |
|   }
 | |
| }
 |