From 9df849ffae108df492ef49cbc4353ea3bbfb15de Mon Sep 17 00:00:00 2001 From: balesniy Date: Mon, 10 Jun 2024 20:23:00 +0200 Subject: [PATCH] update tests --- engine_scripts/childrenDecrease.js | 94 ++++++++++++++ engine_scripts/childrenIncrease.js | 94 ++++++++++++++ engine_scripts/clickAndHoverHelper.cjs | 89 +++++++++++++ engine_scripts/contentOverflow.js | 94 ++++++++++++++ engine_scripts/fill-email.js | 11 ++ engine_scripts/fillInput.js | 20 +++ engine_scripts/hide-bg.js | 18 +++ engine_scripts/highlight-required.js | 21 +++ engine_scripts/imgDecrease.js | 94 ++++++++++++++ engine_scripts/imgIncrease.js | 96 ++++++++++++++ engine_scripts/imgLandscape.js | 96 ++++++++++++++ engine_scripts/imgPortrait.js | 96 ++++++++++++++ engine_scripts/interaction.js | 62 +++++++++ engine_scripts/interceptImages.js | 37 ++++++ engine_scripts/jsDisable.js | 10 ++ engine_scripts/markup-interaction-active.js | 134 +++++++++++++++++++ engine_scripts/markup-interaction-hover.js | 136 ++++++++++++++++++++ engine_scripts/onBefore.js | 3 + engine_scripts/onReady.cjs | 36 ++++++ engine_scripts/overrideCSS.js | 14 ++ engine_scripts/textDecrease.js | 94 ++++++++++++++ engine_scripts/textIncrease.js | 95 ++++++++++++++ engine_scripts/textStylesOnly.js | 59 +++++++++ run-test.js | 102 +++++++-------- test-config/backstop-html-03.config.js | 2 +- test-config/backstop-html-04.config.js | 2 +- test-config/backstop-img-01.config.js | 2 +- test-config/backstop-test-03.config.js | 2 +- test-config/backstop-test-04.config.js | 2 +- test-config/backstop-test-05.config.js | 2 +- test-config/backstop-test-06.config.js | 4 +- test-config/backstop-test-07.config.js | 2 +- test-config/backstop-test-08.config.js | 2 +- test-config/backstop-test-ff.config.js | 2 +- test-config/backstop-test-logo.config.js | 2 +- test-config/backstop-test-menu.config.js | 2 +- test-config/backstop-test-swiper.config.js | 2 +- 37 files changed, 1567 insertions(+), 66 deletions(-) create mode 100644 engine_scripts/childrenDecrease.js create mode 100644 engine_scripts/childrenIncrease.js create mode 100644 engine_scripts/clickAndHoverHelper.cjs create mode 100644 engine_scripts/contentOverflow.js create mode 100644 engine_scripts/fill-email.js create mode 100644 engine_scripts/fillInput.js create mode 100644 engine_scripts/hide-bg.js create mode 100644 engine_scripts/highlight-required.js create mode 100644 engine_scripts/imgDecrease.js create mode 100644 engine_scripts/imgIncrease.js create mode 100644 engine_scripts/imgLandscape.js create mode 100644 engine_scripts/imgPortrait.js create mode 100644 engine_scripts/interaction.js create mode 100644 engine_scripts/interceptImages.js create mode 100644 engine_scripts/jsDisable.js create mode 100644 engine_scripts/markup-interaction-active.js create mode 100644 engine_scripts/markup-interaction-hover.js create mode 100644 engine_scripts/onBefore.js create mode 100644 engine_scripts/onReady.cjs create mode 100644 engine_scripts/overrideCSS.js create mode 100644 engine_scripts/textDecrease.js create mode 100644 engine_scripts/textIncrease.js create mode 100644 engine_scripts/textStylesOnly.js diff --git a/engine_scripts/childrenDecrease.js b/engine_scripts/childrenDecrease.js new file mode 100644 index 00000000..99f165b0 --- /dev/null +++ b/engine_scripts/childrenDecrease.js @@ -0,0 +1,94 @@ +const modifyTextAmount = (multiplier, selectors) => { + const $els = [...document.querySelectorAll(selectors.join(","))].flatMap((el) => [...el.childNodes]); + for (const $el of $els) { + if ($el.nodeType !== $el.TEXT_NODE) continue; + if (!$el.textContent) continue; + + const text = $el.textContent; + + // Умножим строку на меньшее целое число множителя + const intCnt = Math.floor(multiplier); + $el.textContent = Array.from({ length: intCnt }) + .map(() => text) + .join(""); + + // Добавим остаток строки из множителя + const tailMultiplier = multiplier % 1; + $el.textContent += text.slice(0, Math.floor(text.length * tailMultiplier)); + } +}; + +const modifyChildrenAmount = (multiplier, selectors) => { + const $els = [...document.querySelectorAll(selectors.join(","))]; + const getRandomIntBetween = (min, max, seed = 0.5) => { + min = Math.ceil(min); + max = Math.floor(max); + return Math.floor(seed * (max - min + 1) + min); + }; + for (const $el of $els) { + if ($el.nodeType !== $el.ELEMENT_NODE) { + throw new Error( + `По одному из селекторов «${selectors.join("», «")}» найден элемент с неверным nodeType равным ${$el.nodeType}` + ); + } + if ($el.childElementCount < 2) continue; + const newLength = Math.ceil($el.childElementCount * multiplier); + + if (newLength === $el.childElementCount) continue; + while ($el.childElementCount !== newLength) { + const idx = getRandomIntBetween(0, $el.childElementCount - 1); + const $child = $el.children[idx]; + if (newLength < $el.childElementCount) { + $el.removeChild($child); + } else if (newLength > $el.childElementCount) { + $el.appendChild($child.cloneNode(true)); + } + } + } +}; + +const modifyImages = (isBig, selectors) => { + const $els = [...document.querySelectorAll(selectors.join(","))]; + for (const $el of $els) { + if ($el.tagName !== "IMG") { + throw new Error( + `По одному из селекторов «${selectors.join("», «")}» найден элемент с неверным tagName равным ${ + $el.tagName + }, должен быть IMG` + ); + } + // 1500x1500 + const largeImgSrc = + ""; + // 15x15 + const smallImgSrc = + ""; + + $el.src = isBig ? largeImgSrc : smallImgSrc; + $el.width = isBig ? 1500 : 15; + } +}; + +const defaultTextIncreaseMultiplier = 2.1; +const defaultTextDecreaseMultiplier = 0.4; + +const defaultChildrenIncreaseMultiplier = 1.5; +const defaultChildrenDecreaseMultiplier = 0.6; + +const defaultTextSelectors = ["a", "button", "h1", "h2", "h3", "h4", "h5", "h6", "p", "span", "label", "legend", "li"]; +const defaultChildrenSelectors = ["fieldset", "ul", "ol", "dl"]; +const defaultImgSelectors = ["img"]; + +module.exports = async (page, scenario, vp) => { + console.log('SCENARIO > ' + scenario.label); + // add more ready handlers here... + await page.waitForFunction(() => { + return document.fonts.ready.then(() => { + console.log('Fonts loaded'); + return true; + }); + }); + + await page.evaluate(modifyChildrenAmount, scenario.multiplier || defaultChildrenDecreaseMultiplier, defaultChildrenSelectors) + +}; diff --git a/engine_scripts/childrenIncrease.js b/engine_scripts/childrenIncrease.js new file mode 100644 index 00000000..f13ff75b --- /dev/null +++ b/engine_scripts/childrenIncrease.js @@ -0,0 +1,94 @@ +const modifyTextAmount = (multiplier, selectors) => { + const $els = [...document.querySelectorAll(selectors.join(","))].flatMap((el) => [...el.childNodes]); + for (const $el of $els) { + if ($el.nodeType !== $el.TEXT_NODE) continue; + if (!$el.textContent) continue; + + const text = $el.textContent; + + // Умножим строку на меньшее целое число множителя + const intCnt = Math.floor(multiplier); + $el.textContent = Array.from({ length: intCnt }) + .map(() => text) + .join(""); + + // Добавим остаток строки из множителя + const tailMultiplier = multiplier % 1; + $el.textContent += text.slice(0, Math.floor(text.length * tailMultiplier)); + } +}; + +const modifyChildrenAmount = (multiplier, selectors) => { + const $els = [...document.querySelectorAll(selectors.join(","))]; + const getRandomIntBetween = (min, max, seed = 0.5) => { + min = Math.ceil(min); + max = Math.floor(max); + return Math.floor(seed * (max - min + 1) + min); + }; + for (const $el of $els) { + if ($el.nodeType !== $el.ELEMENT_NODE) { + throw new Error( + `По одному из селекторов «${selectors.join("», «")}» найден элемент с неверным nodeType равным ${$el.nodeType}` + ); + } + if ($el.childElementCount < 2) continue; + const newLength = Math.ceil($el.childElementCount * multiplier); + + if (newLength === $el.childElementCount) continue; + while ($el.childElementCount !== newLength) { + const idx = getRandomIntBetween(0, $el.childElementCount - 1); + const $child = $el.children[idx]; + if (newLength < $el.childElementCount) { + $el.removeChild($child); + } else if (newLength > $el.childElementCount) { + $el.appendChild($child.cloneNode(true)); + } + } + } +}; + +const modifyImages = (isBig, selectors) => { + const $els = [...document.querySelectorAll(selectors.join(","))]; + for (const $el of $els) { + if ($el.tagName !== "IMG") { + throw new Error( + `По одному из селекторов «${selectors.join("», «")}» найден элемент с неверным tagName равным ${ + $el.tagName + }, должен быть IMG` + ); + } + // 1500x1500 + const largeImgSrc = + ""; + // 15x15 + const smallImgSrc = + ""; + + $el.src = isBig ? largeImgSrc : smallImgSrc; + $el.width = isBig ? 1500 : 15; + } +}; + +const defaultTextIncreaseMultiplier = 2.1; +const defaultTextDecreaseMultiplier = 0.4; + +const defaultChildrenIncreaseMultiplier = 1.5; +const defaultChildrenDecreaseMultiplier = 0.5; + +const defaultTextSelectors = ["a", "button", "h1", "h2", "h3", "h4", "h5", "h6", "p", "span", "label", "legend", "li"]; +const defaultChildrenSelectors = ["fieldset", "ul", "ol", "dl"]; +const defaultImgSelectors = ["img"]; + +module.exports = async (page, scenario, vp) => { + console.log('SCENARIO > ' + scenario.label); + // add more ready handlers here... + await page.waitForFunction(() => { + return document.fonts.ready.then(() => { + console.log('Fonts loaded'); + return true; + }); + }); + + await page.evaluate(modifyChildrenAmount, scenario.multiplier || defaultChildrenIncreaseMultiplier, defaultChildrenSelectors) + +}; diff --git a/engine_scripts/clickAndHoverHelper.cjs b/engine_scripts/clickAndHoverHelper.cjs new file mode 100644 index 00000000..b32b5b99 --- /dev/null +++ b/engine_scripts/clickAndHoverHelper.cjs @@ -0,0 +1,89 @@ +module.exports = async (page, scenario) => { + const hoverSelector = scenario.hoverSelectors || scenario.hoverSelector; + const clickSelector = scenario.clickSelectors || scenario.clickSelector; + const activeSelector = scenario.activeSelectors || scenario.activeSelector; + const disableSelector = scenario.disableSelectors || scenario.disableSelector; + const focusSelector = scenario.focusSelectors || scenario.focusSelector; + const keyPressSelector = scenario.keyPressSelectors || scenario.keyPressSelector; + const scrollToSelector = scenario.scrollToSelector; + const postInteractionWait = scenario.postInteractionWait; // ms [int] + + const interactiveElsSelector = "a, button, input[type='radio'], input[type='checkbox'], label"; + const content = ` + const preventer = (e) => e.preventDefault(); + const els = document.querySelectorAll("${interactiveElsSelector}"); + const mouseEvts = ["mouseup", "mousedown", "click"]; + + els.forEach((el) => { + mouseEvts.forEach((me) => el.addEventListener(me, preventer)); + }); + `; + + // await page.addScriptTag({ content }); + + if (disableSelector) { + await Promise.all( + [].concat(disableSelector).map(async (selector) => { + await page + .evaluate((sel) => { + document.querySelectorAll(sel).forEach(s => { + if (s.tagName === 'LABEL') { + s = s.control + } + s.disabled = true; + s.classList.add('disabled'); + }); + }, selector); + }) + ); + } + + if (keyPressSelector) { + for (const keyPressSelectorItem of [].concat(keyPressSelector)) { + await page.waitFor(keyPressSelectorItem.selector); + await page.type(keyPressSelectorItem.selector, keyPressSelectorItem.keyPress); + } + } + + if (hoverSelector) { + for (const hoverSelectorIndex of [].concat(hoverSelector)) { + await page.waitForSelector(hoverSelectorIndex); + await page.hover(hoverSelectorIndex); + } + } + + if (activeSelector) { + for (const activeSelectorIndex of [].concat(activeSelector)) { + await page.waitForSelector(activeSelectorIndex); + await page.hover(activeSelectorIndex); + await page.mouse.down(); + } + } + + if (clickSelector) { + for (const clickSelectorIndex of [].concat(clickSelector)) { + await page.waitForSelector(clickSelectorIndex); + await page.click(clickSelectorIndex); + } + } + + if (focusSelector) { + for (const focusSelectorIndex of [].concat(focusSelector)) { + await page.waitForSelector(focusSelectorIndex); + await page.focus(focusSelectorIndex); + } + } + + if (postInteractionWait) { + await page.waitForTimeout(postInteractionWait); + } + + if (scrollToSelector) { + await page.waitForSelector(scrollToSelector); + await page.evaluate(scrollToSelector => { + document.querySelector(scrollToSelector).scrollIntoView({ + behavior: "instant", block: "center", inline: "center" + }); + }, scrollToSelector); + } +}; diff --git a/engine_scripts/contentOverflow.js b/engine_scripts/contentOverflow.js new file mode 100644 index 00000000..12e4c547 --- /dev/null +++ b/engine_scripts/contentOverflow.js @@ -0,0 +1,94 @@ +const modifyTextAmount = (multiplier, selectors) => { + const $els = [...document.querySelectorAll(selectors.join(","))].flatMap((el) => [...el.childNodes]); + for (const $el of $els) { + if ($el.nodeType !== $el.TEXT_NODE) continue; + if (!$el.textContent) continue; + + const text = $el.textContent; + + // Умножим строку на меньшее целое число множителя + const intCnt = Math.floor(multiplier); + $el.textContent = Array.from({ length: intCnt }) + .map(() => text) + .join(""); + + // Добавим остаток строки из множителя + const tailMultiplier = multiplier % 1; + $el.textContent += text.slice(0, Math.floor(text.length * tailMultiplier)); + } +}; + +const modifyChildrenAmount = (multiplier, selectors) => { + const $els = [...document.querySelectorAll(selectors.join(","))]; + const getRandomIntBetween = (min, max, seed = 0.5) => { + min = Math.ceil(min); + max = Math.floor(max); + return Math.floor(seed * (max - min + 1) + min); + }; + for (const $el of $els) { + if ($el.nodeType !== $el.ELEMENT_NODE) { + throw new Error( + `По одному из селекторов «${selectors.join("», «")}» найден элемент с неверным nodeType равным ${$el.nodeType}` + ); + } + if ($el.childElementCount < 2) continue; + const newLength = Math.ceil($el.childElementCount * multiplier); + + if (newLength === $el.childElementCount) continue; + while ($el.childElementCount !== newLength) { + const idx = getRandomIntBetween(0, $el.childElementCount - 1); + const $child = $el.children[idx]; + if (newLength < $el.childElementCount) { + $el.removeChild($child); + } else if (newLength > $el.childElementCount) { + $el.appendChild($child.cloneNode(true)); + } + } + } +}; + +const modifyImages = (isBig, selectors) => { + const $els = [...document.querySelectorAll(selectors.join(","))]; + for (const $el of $els) { + if ($el.tagName !== "IMG") { + throw new Error( + `По одному из селекторов «${selectors.join("», «")}» найден элемент с неверным tagName равным ${ + $el.tagName + }, должен быть IMG` + ); + } + // 1500x1500 + const largeImgSrc = + ""; + // 15x15 + const smallImgSrc = + ""; + + $el.src = isBig ? largeImgSrc : smallImgSrc; + $el.width = isBig ? 1500 : 15; + } +}; + +const defaultTextIncreaseMultiplier = 2.1; +const defaultTextDecreaseMultiplier = 0.4; + +const defaultChildrenIncreaseMultiplier = 1.5; +const defaultChildrenDecreaseMultiplier = 0.5; + +const defaultTextSelectors = ["a", "button", "h1", "h2", "h3", "h4", "h5", "h6", "p", "span", "label", "legend", "li"]; +const defaultChildrenSelectors = ["section", "article", "ul", "ol", "dl"]; +const defaultImgSelectors = ["img"]; + +module.exports = async (page, scenario, vp) => { + console.log('SCENARIO > ' + scenario.label); + // add more ready handlers here... + await page.waitForFunction(() => { + return document.fonts.ready.then(() => { + console.log('Fonts loaded'); + return true; + }); + }); + + await page.evaluate(modifyTextAmount, defaultTextIncreaseMultiplier, defaultTextSelectors) + +}; diff --git a/engine_scripts/fill-email.js b/engine_scripts/fill-email.js new file mode 100644 index 00000000..1b503395 --- /dev/null +++ b/engine_scripts/fill-email.js @@ -0,0 +1,11 @@ +module.exports = async (page, scenario, vp, isReference) => { + console.log('SCENARIO > ' + scenario.label); + await page.waitForFunction(() => { + return document.fonts.ready.then(() => { + console.log('Fonts loaded'); + return true; + }); + }); + await page.type(scenario.field, scenario.email); + await require('./clickAndHoverHelper')(page, scenario); +}; diff --git a/engine_scripts/fillInput.js b/engine_scripts/fillInput.js new file mode 100644 index 00000000..7314e84e --- /dev/null +++ b/engine_scripts/fillInput.js @@ -0,0 +1,20 @@ +module.exports = async (page, scenario, vp) => { + console.log('SCENARIO > ' + scenario.label); + // add more ready handlers here... + await page.waitForFunction(() => { + return document.fonts.ready.then(() => { + console.log('Fonts loaded'); + return true; + }); + }); + await page.evaluate(() => { + const $textEls = [...document.querySelectorAll('input[type="text"],input[type="email"]')]; + $textEls.forEach((el) => { + el.value = 'Lorem ipsum dolor sit amet, consectetur adipisicing elit.'; + }); + const $numberEls = [...document.querySelectorAll('input[type="number"]')]; + $numberEls.forEach((el) => { + el.value = 999999999999999; + }); + }); +}; diff --git a/engine_scripts/hide-bg.js b/engine_scripts/hide-bg.js new file mode 100644 index 00000000..26706c3e --- /dev/null +++ b/engine_scripts/hide-bg.js @@ -0,0 +1,18 @@ +module.exports = async (page, scenario, vp) => { + console.log('SCENARIO > ' + scenario.label); + + // add more ready handlers here... + await page.waitForFunction(() => { + return document.fonts.ready.then(() => { + console.log('Fonts loaded'); + return true; + }); + }); + const BACKSTOP_TEST_CSS_OVERRIDE = ` + *, *::before, *::after { + background-image: unset !important; + } + `; + + await require('./overrideCSS')(page, scenario, BACKSTOP_TEST_CSS_OVERRIDE); +}; diff --git a/engine_scripts/highlight-required.js b/engine_scripts/highlight-required.js new file mode 100644 index 00000000..5545fd30 --- /dev/null +++ b/engine_scripts/highlight-required.js @@ -0,0 +1,21 @@ +module.exports = async (page, scenario, vp) => { + console.log('SCENARIO > ' + scenario.label); + + const BACKSTOP_TEST_CSS_OVERRIDE = ` + input[required] { + outline: solid 10px red !important; + outline-offset: -10px !important; + } + `; + + await require('./overrideCSS')(page, scenario, BACKSTOP_TEST_CSS_OVERRIDE); + + // add more ready handlers here... + await page.waitForFunction(() => { + return document.fonts.ready.then(() => { + console.log('Fonts loaded'); + return true; + }); + }); + +}; diff --git a/engine_scripts/imgDecrease.js b/engine_scripts/imgDecrease.js new file mode 100644 index 00000000..ce7bae6b --- /dev/null +++ b/engine_scripts/imgDecrease.js @@ -0,0 +1,94 @@ +const modifyTextAmount = (multiplier, selectors) => { + const $els = [...document.querySelectorAll(selectors.join(","))].flatMap((el) => [...el.childNodes]); + for (const $el of $els) { + if ($el.nodeType !== $el.TEXT_NODE) continue; + if (!$el.textContent) continue; + + const text = $el.textContent; + + // Умножим строку на меньшее целое число множителя + const intCnt = Math.floor(multiplier); + $el.textContent = Array.from({ length: intCnt }) + .map(() => text) + .join(""); + + // Добавим остаток строки из множителя + const tailMultiplier = multiplier % 1; + $el.textContent += text.slice(0, Math.floor(text.length * tailMultiplier)); + } +}; + +const modifyChildrenAmount = (multiplier, selectors) => { + const $els = [...document.querySelectorAll(selectors.join(","))]; + const getRandomIntBetween = (min, max, seed = 0.5) => { + min = Math.ceil(min); + max = Math.floor(max); + return Math.floor(seed * (max - min + 1) + min); + }; + for (const $el of $els) { + if ($el.nodeType !== $el.ELEMENT_NODE) { + throw new Error( + `По одному из селекторов «${selectors.join("», «")}» найден элемент с неверным nodeType равным ${$el.nodeType}` + ); + } + if ($el.childElementCount < 2) continue; + const newLength = Math.ceil($el.childElementCount * multiplier); + + if (newLength === $el.childElementCount) continue; + while ($el.childElementCount !== newLength) { + const idx = getRandomIntBetween(0, $el.childElementCount - 1); + const $child = $el.children[idx]; + if (newLength < $el.childElementCount) { + $el.removeChild($child); + } else if (newLength > $el.childElementCount) { + $el.appendChild($child.cloneNode(true)); + } + } + } +}; + +const modifyImages = (isBig, selectors) => { + const $els = [...document.querySelectorAll(selectors.join(","))]; + for (const $el of $els) { + if ($el.tagName !== "IMG") { + throw new Error( + `По одному из селекторов «${selectors.join("», «")}» найден элемент с неверным tagName равным ${ + $el.tagName + }, должен быть IMG` + ); + } + // 1500x1500 + const largeImgSrc = + ""; + // 15x15 + const smallImgSrc = + ""; + + $el.src = isBig ? largeImgSrc : smallImgSrc; + $el.width = isBig ? 1500 : 15; + } +}; + +const defaultTextIncreaseMultiplier = 2.1; +const defaultTextDecreaseMultiplier = 0.4; + +const defaultChildrenIncreaseMultiplier = 1.5; +const defaultChildrenDecreaseMultiplier = 0.5; + +const defaultTextSelectors = ["a", "button", "h1", "h2", "h3", "h4", "h5", "h6", "p", "span", "label", "legend", "li"]; +const defaultChildrenSelectors = ["section", "article", "ul", "ol", "dl"]; +const defaultImgSelectors = ["img"]; + +module.exports = async (page, scenario, vp) => { + console.log('SCENARIO > ' + scenario.label); + // add more ready handlers here... + await page.waitForFunction(() => { + return document.fonts.ready.then(() => { + console.log('Fonts loaded'); + return true; + }); + }); + + await page.evaluate(modifyImages, false, defaultImgSelectors) + +}; diff --git a/engine_scripts/imgIncrease.js b/engine_scripts/imgIncrease.js new file mode 100644 index 00000000..971235ef --- /dev/null +++ b/engine_scripts/imgIncrease.js @@ -0,0 +1,96 @@ +const modifyTextAmount = (multiplier, selectors) => { + const $els = [...document.querySelectorAll(selectors.join(","))].flatMap((el) => [...el.childNodes]); + for (const $el of $els) { + if ($el.nodeType !== $el.TEXT_NODE) continue; + if (!$el.textContent) continue; + + const text = $el.textContent; + + // Умножим строку на меньшее целое число множителя + const intCnt = Math.floor(multiplier); + $el.textContent = Array.from({ length: intCnt }) + .map(() => text) + .join(""); + + // Добавим остаток строки из множителя + const tailMultiplier = multiplier % 1; + $el.textContent += text.slice(0, Math.floor(text.length * tailMultiplier)); + } +}; + +const modifyChildrenAmount = (multiplier, selectors) => { + const $els = [...document.querySelectorAll(selectors.join(","))]; + const getRandomIntBetween = (min, max, seed = 0.5) => { + min = Math.ceil(min); + max = Math.floor(max); + return Math.floor(seed * (max - min + 1) + min); + }; + for (const $el of $els) { + if ($el.nodeType !== $el.ELEMENT_NODE) { + throw new Error( + `По одному из селекторов «${selectors.join("», «")}» найден элемент с неверным nodeType равным ${$el.nodeType}` + ); + } + if ($el.childElementCount < 2) continue; + const newLength = Math.ceil($el.childElementCount * multiplier); + + if (newLength === $el.childElementCount) continue; + while ($el.childElementCount !== newLength) { + const idx = getRandomIntBetween(0, $el.childElementCount - 1); + const $child = $el.children[idx]; + if (newLength < $el.childElementCount) { + $el.removeChild($child); + } else if (newLength > $el.childElementCount) { + $el.appendChild($child.cloneNode(true)); + } + } + } +}; + +const modifyImages = (isBig, selectors) => { + const $els = [...document.querySelectorAll(selectors.join(","))]; + for (const $el of $els) { + if ($el.tagName !== "IMG") { + throw new Error( + `По одному из селекторов «${selectors.join("», «")}» найден элемент с неверным tagName равным ${ + $el.tagName + }, должен быть IMG` + ); + } + // 1500x1500 + const largeImgSrc = + ""; + // 15x15 + const smallImgSrc = + ""; + const portraitImgSrc = + "" + const landscapeImgSrc = + "" + $el.src = isBig ? largeImgSrc : smallImgSrc; + $el.width = isBig ? 1500 : 15; + } +}; + +const defaultTextIncreaseMultiplier = 2.1; +const defaultTextDecreaseMultiplier = 0.4; + +const defaultChildrenIncreaseMultiplier = 1.5; +const defaultChildrenDecreaseMultiplier = 0.5; + +const defaultTextSelectors = ["a", "button", "h1", "h2", "h3", "h4", "h5", "h6", "p", "span", "label", "legend", "li"]; +const defaultChildrenSelectors = ["section", "article", "ul", "ol", "dl"]; +const defaultImgSelectors = ["img"]; + +module.exports = async (page, scenario, vp) => { + console.log('SCENARIO > ' + scenario.label); + // add more ready handlers here... + await page.waitForFunction(() => { + return document.fonts.ready.then(() => { + console.log('Fonts loaded'); + return true; + }); + }); + + await page.evaluate(modifyImages, true, defaultImgSelectors) +}; diff --git a/engine_scripts/imgLandscape.js b/engine_scripts/imgLandscape.js new file mode 100644 index 00000000..fefb2ff1 --- /dev/null +++ b/engine_scripts/imgLandscape.js @@ -0,0 +1,96 @@ +const modifyTextAmount = (multiplier, selectors) => { + const $els = [...document.querySelectorAll(selectors.join(","))].flatMap((el) => [...el.childNodes]); + for (const $el of $els) { + if ($el.nodeType !== $el.TEXT_NODE) continue; + if (!$el.textContent) continue; + + const text = $el.textContent; + + // Умножим строку на меньшее целое число множителя + const intCnt = Math.floor(multiplier); + $el.textContent = Array.from({ length: intCnt }) + .map(() => text) + .join(""); + + // Добавим остаток строки из множителя + const tailMultiplier = multiplier % 1; + $el.textContent += text.slice(0, Math.floor(text.length * tailMultiplier)); + } +}; + +const modifyChildrenAmount = (multiplier, selectors) => { + const $els = [...document.querySelectorAll(selectors.join(","))]; + const getRandomIntBetween = (min, max, seed = 0.5) => { + min = Math.ceil(min); + max = Math.floor(max); + return Math.floor(seed * (max - min + 1) + min); + }; + for (const $el of $els) { + if ($el.nodeType !== $el.ELEMENT_NODE) { + throw new Error( + `По одному из селекторов «${selectors.join("», «")}» найден элемент с неверным nodeType равным ${$el.nodeType}` + ); + } + if ($el.childElementCount < 2) continue; + const newLength = Math.ceil($el.childElementCount * multiplier); + + if (newLength === $el.childElementCount) continue; + while ($el.childElementCount !== newLength) { + const idx = getRandomIntBetween(0, $el.childElementCount - 1); + const $child = $el.children[idx]; + if (newLength < $el.childElementCount) { + $el.removeChild($child); + } else if (newLength > $el.childElementCount) { + $el.appendChild($child.cloneNode(true)); + } + } + } +}; + +const modifyImages = (isLandscape, selectors) => { + const $els = [...document.querySelectorAll(selectors.join(","))]; + for (const $el of $els) { + if ($el.tagName !== "IMG") { + throw new Error( + `По одному из селекторов «${selectors.join("», «")}» найден элемент с неверным tagName равным ${ + $el.tagName + }, должен быть IMG` + ); + } + // 1500x1500 + const largeImgSrc = + ""; + // 15x15 + const smallImgSrc = + ""; + const portraitImgSrc = + "" + const landscapeImgSrc = + "" + $el.src = isLandscape ? landscapeImgSrc : portraitImgSrc; + $el.width = isLandscape ? 500 : 300; + } +}; + +const defaultTextIncreaseMultiplier = 2.1; +const defaultTextDecreaseMultiplier = 0.4; + +const defaultChildrenIncreaseMultiplier = 1.5; +const defaultChildrenDecreaseMultiplier = 0.5; + +const defaultTextSelectors = ["a", "button", "h1", "h2", "h3", "h4", "h5", "h6", "p", "span", "label", "legend", "li"]; +const defaultChildrenSelectors = ["section", "article", "ul", "ol", "dl"]; +const defaultImgSelectors = ["img"]; + +module.exports = async (page, scenario, vp) => { + console.log('SCENARIO > ' + scenario.label); + // add more ready handlers here... + await page.waitForFunction(() => { + return document.fonts.ready.then(() => { + console.log('Fonts loaded'); + return true; + }); + }); + + await page.evaluate(modifyImages, true, defaultImgSelectors) +}; diff --git a/engine_scripts/imgPortrait.js b/engine_scripts/imgPortrait.js new file mode 100644 index 00000000..7231d684 --- /dev/null +++ b/engine_scripts/imgPortrait.js @@ -0,0 +1,96 @@ +const modifyTextAmount = (multiplier, selectors) => { + const $els = [...document.querySelectorAll(selectors.join(","))].flatMap((el) => [...el.childNodes]); + for (const $el of $els) { + if ($el.nodeType !== $el.TEXT_NODE) continue; + if (!$el.textContent) continue; + + const text = $el.textContent; + + // Умножим строку на меньшее целое число множителя + const intCnt = Math.floor(multiplier); + $el.textContent = Array.from({ length: intCnt }) + .map(() => text) + .join(""); + + // Добавим остаток строки из множителя + const tailMultiplier = multiplier % 1; + $el.textContent += text.slice(0, Math.floor(text.length * tailMultiplier)); + } +}; + +const modifyChildrenAmount = (multiplier, selectors) => { + const $els = [...document.querySelectorAll(selectors.join(","))]; + const getRandomIntBetween = (min, max, seed = 0.5) => { + min = Math.ceil(min); + max = Math.floor(max); + return Math.floor(seed * (max - min + 1) + min); + }; + for (const $el of $els) { + if ($el.nodeType !== $el.ELEMENT_NODE) { + throw new Error( + `По одному из селекторов «${selectors.join("», «")}» найден элемент с неверным nodeType равным ${$el.nodeType}` + ); + } + if ($el.childElementCount < 2) continue; + const newLength = Math.ceil($el.childElementCount * multiplier); + + if (newLength === $el.childElementCount) continue; + while ($el.childElementCount !== newLength) { + const idx = getRandomIntBetween(0, $el.childElementCount - 1); + const $child = $el.children[idx]; + if (newLength < $el.childElementCount) { + $el.removeChild($child); + } else if (newLength > $el.childElementCount) { + $el.appendChild($child.cloneNode(true)); + } + } + } +}; + +const modifyImages = (isLandscape, selectors) => { + const $els = [...document.querySelectorAll(selectors.join(","))]; + for (const $el of $els) { + if ($el.tagName !== "IMG") { + throw new Error( + `По одному из селекторов «${selectors.join("», «")}» найден элемент с неверным tagName равным ${ + $el.tagName + }, должен быть IMG` + ); + } + // 1500x1500 + const largeImgSrc = + ""; + // 15x15 + const smallImgSrc = + ""; + const portraitImgSrc = + "" + const landscapeImgSrc = + "" + $el.src = isLandscape ? landscapeImgSrc : portraitImgSrc; + $el.width = isLandscape ? 500 : 300; + } +}; + +const defaultTextIncreaseMultiplier = 2.1; +const defaultTextDecreaseMultiplier = 0.4; + +const defaultChildrenIncreaseMultiplier = 1.5; +const defaultChildrenDecreaseMultiplier = 0.5; + +const defaultTextSelectors = ["a", "button", "h1", "h2", "h3", "h4", "h5", "h6", "p", "span", "label", "legend", "li"]; +const defaultChildrenSelectors = ["section", "article", "ul", "ol", "dl"]; +const defaultImgSelectors = ["img"]; + +module.exports = async (page, scenario, vp) => { + console.log('SCENARIO > ' + scenario.label); + // add more ready handlers here... + await page.waitForFunction(() => { + return document.fonts.ready.then(() => { + console.log('Fonts loaded'); + return true; + }); + }); + + await page.evaluate(modifyImages, false, defaultImgSelectors) +}; diff --git a/engine_scripts/interaction.js b/engine_scripts/interaction.js new file mode 100644 index 00000000..d1eb5520 --- /dev/null +++ b/engine_scripts/interaction.js @@ -0,0 +1,62 @@ +module.exports = async (page, scenario, vp, isReference) => { + console.log('SCENARIO > ' + scenario.label); + await page.waitForFunction(() => { + return document.fonts.ready.then(() => { + console.log('Fonts loaded'); + return true; + }); + }); + if (scenario.content) { + function xpathPrepare(xpath, searchString) { + return xpath.replace("$u", searchString.toUpperCase()) + .replace("$l", searchString.toLowerCase()) + .replace("$s", searchString.toLowerCase()); + } + + const ancestor = `*[@data-test="${scenario.section}"]` + + const [elementHandle] = await page.$x(`//${ancestor}//${xpathPrepare("*[text()[contains(translate(., '$u', '$l'), '$s')]]", scenario.content)}`); + if (!elementHandle) { + throw new Error('Element not found with text "' + scenario.content + '"'); + } + const selector = await page.evaluate((element, text) => { + const closest = element.closest('a, button, label, select'); + if (!closest) { + return null; + } + closest.dataset.testText = text + return closest.tagName + '[data-test-text="' + text + '"]' + (closest.classList.length ? '.' + Array.from(closest.classList).join('.') : '') + }, elementHandle, scenario.content) + if (!selector) { + throw new Error('Selector not found for text "' + scenario.content + '"'); + } + + // const parentSelector = await page.evaluate((element) => { + // const closest = element.closest('a, button, label, select'); + // let parent = closest.parentElement || document.body; + // while (parent.clientWidth < (closest.clientWidth + 10) || parent.clientHeight < (closest.clientHeight + 10)) { + // parent = parent.parentElement + // } + // parent.dataset.testText = 'parent' + // return parent.tagName + '[data-test-text="parent"]' + '.' + Array.from(parent.classList).join('.') + // }, elementHandle) + scenario.selectors = [`[data-test="${scenario.section}"] ${selector}`] + + // scenario.scrollToSelector = selector + console.log({ selector }) + + if (scenario.hoverSelectors || scenario.hoverSelector) { + scenario.hoverSelectors = [selector] + } + if (scenario.activeSelectors || scenario.activeSelector) { + scenario.activeSelectors = [selector] + } + if (scenario.focusSelectors || scenario.focusSelector) { + scenario.focusSelectors = [selector] + } + if (scenario.disableSelectors || scenario.disableSelector) { + scenario.disableSelectors = [selector] + } + } + await require('./clickAndHoverHelper')(page, scenario); +}; diff --git a/engine_scripts/interceptImages.js b/engine_scripts/interceptImages.js new file mode 100644 index 00000000..2b02be91 --- /dev/null +++ b/engine_scripts/interceptImages.js @@ -0,0 +1,37 @@ +/** + * INTERCEPT IMAGES + * Listen to all requests. If a request matches IMAGE_URL_RE + * then stub the image with data from IMAGE_STUB_URL + * + * Use this in an onBefore script E.G. + ``` + module.exports = async function(page, scenario) { + require('./interceptImages')(page, scenario); + } + ``` + * + */ + +const fs = require('fs'); +const path = require('path'); + +const IMAGE_URL_RE = /\.gif|\.jpg|\.png/i; +const IMAGE_STUB_URL = path.resolve(__dirname, '../imageStub.jpg'); +const IMAGE_DATA_BUFFER = fs.readFileSync(IMAGE_STUB_URL); +const HEADERS_STUB = {}; + +module.exports = async function (page, scenario) { + const intercept = async (request, targetUrl) => { + if (IMAGE_URL_RE.test(request.url())) { + await request.respond({ + body: IMAGE_DATA_BUFFER, + headers: HEADERS_STUB, + status: 200 + }); + } else { + request.continue(); + } + }; + await page.setRequestInterception(true); + page.on('request', intercept); +}; diff --git a/engine_scripts/jsDisable.js b/engine_scripts/jsDisable.js new file mode 100644 index 00000000..4820f42f --- /dev/null +++ b/engine_scripts/jsDisable.js @@ -0,0 +1,10 @@ +module.exports = async (page, scenario, vp) => { + await page.setRequestInterception(true) + //check resourceType is script + page.on('request', request => { + if (request.resourceType() === 'script') + request.abort(); + else + request.continue(); + }) +}; diff --git a/engine_scripts/markup-interaction-active.js b/engine_scripts/markup-interaction-active.js new file mode 100644 index 00000000..dfb5fb42 --- /dev/null +++ b/engine_scripts/markup-interaction-active.js @@ -0,0 +1,134 @@ +module.exports = async (page, { section = 'body' }, vp) => { + await page.waitForFunction(() => { + return document.fonts.ready.then(() => { + console.log('Fonts loaded'); + return true; + }); + }); + const interactiveElsSelector = `${section} :is(a, button, label, input[type='radio'], input[type='checkbox'])`; + const content = ` + const preventer = (e) => e.preventDefault(); + const els = document.querySelectorAll("${interactiveElsSelector}"); + const mouseEvts = ["mouseup", "mousedown", "click"]; + + els.forEach((el) => { + mouseEvts.forEach((me) => el.addEventListener(me, preventer)); + }); + `; + await page.addScriptTag({ content }); + + // 1. Получаем список интерактивных элементов: a, button, input[radio, checkbox] + const elts = await page.$$(interactiveElsSelector); + + const eq = (l, r) => { + if (l === null && r === null) { + return true; + } + if (l === null || r === null) { + return false; + } + + return l.x === r.x && l.y === r.y && l.width === r.width && l.height === r.height; + }; + const diff = (l, r) => { + if (l === null && r === null) { + return true; + } + if (l === null || r === null) { + return false; + } + return { + x: l.x - r.x, + y: l.y - r.y, + width: l.width - r.width, + height: l.height - r.height + } + } + const results = []; + for await (const el of elts) { + const isVisibleHandle = await page.evaluateHandle((e) => { + const style = window.getComputedStyle(e); + return (style && style.display !== 'none' && + style.visibility !== 'hidden' && style.opacity !== '0' && !e.classList.contains('visually-hidden')); + }, el); + const visible = await isVisibleHandle.jsonValue(); + const box = await el.boxModel(); + if (!visible || !box) { + continue; + } + + await el.evaluate((element) => { + element.scrollIntoView({block: 'center', inline: 'center', behavior: 'instant'}); + }) + const bb = await el.boundingBox(); + if (bb.width === 0 || bb.height === 0) { + // await el.evaluate((h) => h.style.visibility = 'visible'); + continue; + } + + // JSHandle следующего элемента + const nh = await el.evaluateHandle((e) => e.nextElementSibling || e.parentElement, el); + // ElementHandle + const ne = nh.asElement(); + const nb = ne ? await ne.boundingBox() : null; + const code = await el.evaluate((h) => h.outerHTML); + if (!nb) { + console.log("no next element", code); + } + + // эмулируем page.mouse.move(координаты элемента), + // el.hover() тоже подойдёт + await el.hover(); + + const bbHover = await el.boundingBox(); + const nbHover = ne ? await ne.boundingBox() : null; + + // затем page.mouse.down() + await page.mouse.down(); + + const bbActive = await el.boundingBox(); + const nbActive = ne ? await ne.boundingBox() : null; + + // чтобы можно было кликнуть следующий элемент, поднимаем палец с кнопки + await page.mouse.up(); + + // 3. После каждой эмуляции + // 3.1. Смотрим размеры и позицию самого элемента + // 3.2. Смотрим размеры и позицию следующего элемента (надеюсь, этого хватит) + // Если размеры какого-то элемента меняются, репортим + if (!eq(bbActive, bbHover)) { + results.push({ + code, + diff: diff(bbActive, bbHover), + title: "В состоянии «:hover» позиция/размеры меняются" + }); + await el.evaluate((h) => h.style.visibility = 'visible'); + } else if (!eq(nbActive, nbHover)) { + results.push({ + code, + diff: diff(nbActive, nbHover), + title: "В состоянии «:hover» позиция/размеры следующего элемента меняются" + }); + await el.evaluate((h) => h.style.visibility = 'visible'); + } else if (!eq(bb, bbActive)) { + results.push({ + code, + diff: diff(bb, bbActive), + title: "В состоянии «:active» позиция/размеры меняются" + }); + await el.evaluate((h) => h.style.visibility = 'visible'); + } else if (!eq(nb, nbActive)) { + results.push({ + code, + diff: diff(nb, nbActive), + title: "В состоянии «:active» позиция/размеры следующего элемента меняются" + }); + await el.evaluate((h) => h.style.visibility = 'visible'); + } + + nh.dispose(); + } + if (results.length) { + // throw new Error(results.map((r) => r.title + '\n' + r.code + '\n' + JSON.stringify(r.diff)).join('\n')); + } +} diff --git a/engine_scripts/markup-interaction-hover.js b/engine_scripts/markup-interaction-hover.js new file mode 100644 index 00000000..900b40cc --- /dev/null +++ b/engine_scripts/markup-interaction-hover.js @@ -0,0 +1,136 @@ +module.exports = async (page, { section = 'body' }, scenario) => { + await page.waitForFunction(() => { + return document.fonts.ready.then(() => { + console.log('Fonts loaded'); + return true; + }); + }); + const interactiveElsSelector = `${section} :is(a, button, label, input[type='radio'], input[type='checkbox'])`; + const content = ` + const preventer = (e) => e.preventDefault(); + const els = document.querySelectorAll("${interactiveElsSelector}"); + const mouseEvts = ["mouseup", "mousedown", "click"]; + + els.forEach((el) => { + mouseEvts.forEach((me) => el.addEventListener(me, preventer)); + }); + `; + + await page.addScriptTag({ content }); + + // 1. Получаем список интерактивных элементов: a, button, input[radio, checkbox] + const elts = await page.$$(interactiveElsSelector); + + const eq = (l, r) => { + if (l === null && r === null) { + return true; + } + if (l === null || r === null) { + return false; + } + + return l.x === r.x && l.y === r.y && l.width === r.width && l.height === r.height; + }; + const diff = (l, r) => { + if (l === null && r === null) { + return true; + } + if (l === null || r === null) { + return false; + } + return { + x: l.x - r.x, + y: l.y - r.y, + width: l.width - r.width, + height: l.height - r.height + } + } + const results = []; + for await (const el of elts) { + const isVisibleHandle = await page.evaluateHandle((e) => { + const style = window.getComputedStyle(e); + return (style && style.display !== 'none' && + style.visibility !== 'hidden' && style.opacity !== '0' && !e.classList.contains('visually-hidden')); + }, el); + const visible = await isVisibleHandle.jsonValue(); + const box = await el.boxModel(); + if (!visible || !box) { + continue; + } + + await el.evaluate((element) => { + element.scrollIntoView({block: 'center', inline: 'center', behavior: 'instant'}); + }) + const bb = await el.boundingBox(); + if (bb.width === 0 || bb.height === 0) { + // await el.evaluate((h) => h.style.visibility = 'visible'); + continue; + } + + // JSHandle следующего элемента + const nh = await el.evaluateHandle((e) => e.nextElementSibling || e.parentElement, el); + // ElementHandle + const ne = nh.asElement(); + const nb = ne ? await ne.boundingBox() : null; + const code = await el.evaluate((h) => h.outerHTML); + if (!nb) { + console.log("no next element", code); + } + + // эмулируем page.mouse.move(координаты элемента), + // el.hover() тоже подойдёт + await el.hover(); + + const bbHover = await el.boundingBox(); + const nbHover = ne ? await ne.boundingBox() : null; + + // затем page.mouse.down() + // await page.mouse.down(); + // + // const bbActive = await el.boundingBox(); + // const nbActive = ne ? await ne.boundingBox() : null; + + // чтобы можно было кликнуть следующий элемент, поднимаем палец с кнопки + // await page.mouse.up(); + + // 3. После каждой эмуляции + // 3.1. Смотрим размеры и позицию самого элемента + // 3.2. Смотрим размеры и позицию следующего элемента (надеюсь, этого хватит) + // Если размеры какого-то элемента меняются, репортим + if (!eq(bb, bbHover)) { + results.push({ + code, + diff: diff(bb, bbHover), + title: "В состоянии «:hover» позиция/размеры меняются" + }); + await el.evaluate((h) => h.style.visibility = 'visible'); + } else if (!eq(nb, nbHover)) { + results.push({ + code, + diff: diff(nb, nbHover), + title: "В состоянии «:hover» позиция/размеры следующего элемента меняются" + }); + await el.evaluate((h) => h.style.visibility = 'visible'); + } + /*else if (!eq(bb, bbActive)) { + results.push({ + code, + diff: diff(bb, bbActive), + title: "В состоянии «:active» позиция/размеры меняются" + }); + await el.evaluate((h) => h.style.visibility = 'visible'); + } else if (!eq(nb, nbActive)) { + results.push({ + code, + diff: diff(nb, nbActive), + title: "В состоянии «:active» позиция/размеры следующего элемента меняются" + }); + await el.evaluate((h) => h.style.visibility = 'visible'); + }*/ + + nh.dispose(); + } + if (results.length) { + // throw new Error(results.map((r) => r.title + '\n' + r.code + '\n' + JSON.stringify(r.diff)).join('\n')); + } +} diff --git a/engine_scripts/onBefore.js b/engine_scripts/onBefore.js new file mode 100644 index 00000000..c0ed9cb2 --- /dev/null +++ b/engine_scripts/onBefore.js @@ -0,0 +1,3 @@ +module.exports = async (page, scenario, vp) => { + await require('./interceptImages.cjs')(page, scenario); +}; diff --git a/engine_scripts/onReady.cjs b/engine_scripts/onReady.cjs new file mode 100644 index 00000000..0e42875e --- /dev/null +++ b/engine_scripts/onReady.cjs @@ -0,0 +1,36 @@ +module.exports = async (page, scenario, vp) => { + console.log('SCENARIO > ' + scenario.label); + + // add more ready handlers here... + await page.waitForFunction(() => { + return document.fonts.ready.then(() => { + console.log('Fonts loaded'); + return true; + }); + }); + // await page.waitForSelector('.leaflet-marker-icon'); + // await page.waitForSelector('.leaflet-tile-loaded'); + + await require('./clickAndHoverHelper.cjs')(page, scenario); + + if (scenario.content) { + const [elementHandle] = await page.$x(`//*[contains(text(), '${scenario.content}')]`); + const selector = await page.evaluate((element) => { + return element.tagName + '.' + element.className + }, elementHandle) + console.log(selector) + scenario.selectors = [selector] + } + if (scenario.showSelectors) { + await Promise.all( + scenario.showSelectors.map(async (selector) => { + await page + .evaluate((sel) => { + document.querySelectorAll(sel).forEach(s => { + s.style.visibility = 'visible'; + }); + }, selector); + }) + ); + } +}; diff --git a/engine_scripts/overrideCSS.js b/engine_scripts/overrideCSS.js new file mode 100644 index 00000000..c5f9da2f --- /dev/null +++ b/engine_scripts/overrideCSS.js @@ -0,0 +1,14 @@ +module.exports = async (page, scenario, BACKSTOP_TEST_CSS_OVERRIDE = 'html {background-image: none;}') => { + // inject arbitrary css to override styles + await page.evaluate(`window._styleData = '${BACKSTOP_TEST_CSS_OVERRIDE.replace(/\n/g, '')}';`); + + await page.evaluate(() => { + const style = document.createElement('style'); + style.type = 'text/css'; + const styleNode = document.createTextNode(window._styleData); + style.appendChild(styleNode); + document.head.appendChild(style); + }); + + console.log('BACKSTOP_TEST_CSS_OVERRIDE injected for: ' + scenario.label); +}; diff --git a/engine_scripts/textDecrease.js b/engine_scripts/textDecrease.js new file mode 100644 index 00000000..693b0cf7 --- /dev/null +++ b/engine_scripts/textDecrease.js @@ -0,0 +1,94 @@ +const modifyTextAmount = (multiplier, selectors) => { + const $els = [...document.querySelectorAll(selectors.join(","))].flatMap((el) => [...el.childNodes]); + for (const $el of $els) { + if ($el.nodeType !== $el.TEXT_NODE) continue; + if (!$el.textContent) continue; + + const text = $el.textContent; + + // Умножим строку на меньшее целое число множителя + const intCnt = Math.floor(multiplier); + $el.textContent = Array.from({ length: intCnt }) + .map(() => text) + .join(""); + + // Добавим остаток строки из множителя + const tailMultiplier = multiplier % 1; + $el.textContent += text.slice(0, Math.floor(text.length * tailMultiplier)); + } +}; + +const modifyChildrenAmount = (multiplier, selectors) => { + const $els = [...document.querySelectorAll(selectors.join(","))]; + const getRandomIntBetween = (min, max, seed = 0.5) => { + min = Math.ceil(min); + max = Math.floor(max); + return Math.floor(seed * (max - min + 1) + min); + }; + for (const $el of $els) { + if ($el.nodeType !== $el.ELEMENT_NODE) { + throw new Error( + `По одному из селекторов «${selectors.join("», «")}» найден элемент с неверным nodeType равным ${$el.nodeType}` + ); + } + if ($el.childElementCount < 2) continue; + const newLength = Math.ceil($el.childElementCount * multiplier); + + if (newLength === $el.childElementCount) continue; + while ($el.childElementCount !== newLength) { + const idx = getRandomIntBetween(0, $el.childElementCount - 1); + const $child = $el.children[idx]; + if (newLength < $el.childElementCount) { + $el.removeChild($child); + } else if (newLength > $el.childElementCount) { + $el.appendChild($child.cloneNode(true)); + } + } + } +}; + +const modifyImages = (isBig, selectors) => { + const $els = [...document.querySelectorAll(selectors.join(","))]; + for (const $el of $els) { + if ($el.tagName !== "IMG") { + throw new Error( + `По одному из селекторов «${selectors.join("», «")}» найден элемент с неверным tagName равным ${ + $el.tagName + }, должен быть IMG` + ); + } + // 1500x1500 + const largeImgSrc = + ""; + // 15x15 + const smallImgSrc = + ""; + + $el.src = isBig ? largeImgSrc : smallImgSrc; + $el.width = isBig ? 1500 : 15; + } +}; + +const defaultTextIncreaseMultiplier = 2.1; +const defaultTextDecreaseMultiplier = 0.4; + +const defaultChildrenIncreaseMultiplier = 1.5; +const defaultChildrenDecreaseMultiplier = 0.5; + +const defaultTextSelectors = ["a", "button", "h1", "h2", "h3", "h4", "h5", "h6", "p", "span", "label", "legend", "li"]; +const defaultChildrenSelectors = ["section", "article", "ul", "ol", "dl"]; +const defaultImgSelectors = ["img"]; + +module.exports = async (page, scenario, vp) => { + console.log('SCENARIO > ' + scenario.label); + // add more ready handlers here... + await page.waitForFunction(() => { + return document.fonts.ready.then(() => { + console.log('Fonts loaded'); + return true; + }); + }); + + await page.evaluate(modifyTextAmount, defaultTextDecreaseMultiplier, defaultTextSelectors) + +}; diff --git a/engine_scripts/textIncrease.js b/engine_scripts/textIncrease.js new file mode 100644 index 00000000..42b4368e --- /dev/null +++ b/engine_scripts/textIncrease.js @@ -0,0 +1,95 @@ +const modifyTextAmount = (multiplier, selectors) => { + const $els = [...document.querySelectorAll(selectors.join(","))].flatMap((el) => [...el.childNodes]); + for (const $el of $els) { + if ($el.nodeType !== $el.TEXT_NODE) continue; + if (!$el.textContent) continue; + + const text = $el.textContent.trim(); + if (!text) continue; + + // Умножим строку на меньшее целое число множителя + const intCnt = Math.floor(multiplier); + $el.textContent = Array.from({ length: intCnt }) + .map(() => text) + .join(", "); + + // Добавим остаток строки из множителя + const tailMultiplier = multiplier % 1; + $el.textContent += text.slice(0, Math.floor(text.length * tailMultiplier)); + } +}; + +const modifyChildrenAmount = (multiplier, selectors) => { + const $els = [...document.querySelectorAll(selectors.join(","))]; + const getRandomIntBetween = (min, max, seed = 0.5) => { + min = Math.ceil(min); + max = Math.floor(max); + return Math.floor(seed * (max - min + 1) + min); + }; + for (const $el of $els) { + if ($el.nodeType !== $el.ELEMENT_NODE) { + throw new Error( + `По одному из селекторов «${selectors.join("», «")}» найден элемент с неверным nodeType равным ${$el.nodeType}` + ); + } + if ($el.childElementCount < 2) continue; + const newLength = Math.ceil($el.childElementCount * multiplier); + + if (newLength === $el.childElementCount) continue; + while ($el.childElementCount !== newLength) { + const idx = getRandomIntBetween(0, $el.childElementCount - 1); + const $child = $el.children[idx]; + if (newLength < $el.childElementCount) { + $el.removeChild($child); + } else if (newLength > $el.childElementCount) { + $el.appendChild($child.cloneNode(true)); + } + } + } +}; + +const modifyImages = (isBig, selectors) => { + const $els = [...document.querySelectorAll(selectors.join(","))]; + for (const $el of $els) { + if ($el.tagName !== "IMG") { + throw new Error( + `По одному из селекторов «${selectors.join("», «")}» найден элемент с неверным tagName равным ${ + $el.tagName + }, должен быть IMG` + ); + } + // 1500x1500 + const largeImgSrc = + ""; + // 15x15 + const smallImgSrc = + ""; + + $el.src = isBig ? largeImgSrc : smallImgSrc; + $el.width = isBig ? 1500 : 15; + } +}; + +const defaultTextIncreaseMultiplier = 2.1; +const defaultTextDecreaseMultiplier = 0.4; + +const defaultChildrenIncreaseMultiplier = 1.5; +const defaultChildrenDecreaseMultiplier = 0.5; + +const defaultTextSelectors = ["a", "button", "h1", "h2", "h3", "h4", "h5", "h6", "p", "span", "label", "legend", "li"]; +const defaultChildrenSelectors = ["section", "article", "ul", "ol", "dl"]; +const defaultImgSelectors = ["img"]; + +module.exports = async (page, scenario, vp) => { + console.log('SCENARIO > ' + scenario.label); + // add more ready handlers here... + await page.waitForFunction(() => { + return document.fonts.ready.then(() => { + console.log('Fonts loaded'); + return true; + }); + }); + + await page.evaluate(modifyTextAmount, defaultTextIncreaseMultiplier, defaultTextSelectors) + +}; diff --git a/engine_scripts/textStylesOnly.js b/engine_scripts/textStylesOnly.js new file mode 100644 index 00000000..0da3e432 --- /dev/null +++ b/engine_scripts/textStylesOnly.js @@ -0,0 +1,59 @@ +module.exports = async (page, scenario, vp) => { + console.log('SCENARIO > ' + scenario.label); + + const BACKSTOP_TEST_CSS_OVERRIDE = ` + *:where(:not(.visually-hidden, :root, head, script)) { + background-image: unset !important; + transform: unset !important; + margin: unset !important; + padding: unset !important; + border: unset !important; + border-radius: unset !important; + width: auto !important; + max-width: unset !important; + height: auto !important; + min-height: unset !important; + min-width: unset !important; + text-align: unset !important; + display: inline-block !important; + position: static !important; + } + img, svg, *::before, *::after { + display: none !important; + } + `; + + await require('./overrideCSS')(page, scenario, BACKSTOP_TEST_CSS_OVERRIDE); + + // add more ready handlers here... + await page.waitForFunction(() => { + return document.fonts.ready.then(() => { + console.log('Fonts loaded'); + return true; + }); + }); + await page.evaluate(() => { + document.querySelectorAll('br').forEach(s => { + s.remove(); + }) + }); + if (scenario.content) { + function xpathPrepare(xpath, searchString) { + return xpath.replace("$u", searchString.toUpperCase()) + .replace("$l", searchString.toLowerCase()) + .replace("$s", searchString.toLowerCase()); + } + const selectors = [] + for (const text of [].concat(scenario.content)) { + const [elementHandle] = await page.$x(`//${xpathPrepare("*[text()[contains(translate(., '$u', '$l'), '$s')]]", text)}`); + if (!elementHandle) { + throw new Error('Element not found with text "' + text + '"'); + } + await page.evaluate((element, text) => { + element.dataset.testText = `${text}` + }, elementHandle, text) + selectors.push(`[data-test-text="${text}"]`) + } + scenario.selectors = selectors + } +}; diff --git a/run-test.js b/run-test.js index 369d258e..d810ae2c 100644 --- a/run-test.js +++ b/run-test.js @@ -1,56 +1,54 @@ -const fs = require('node:fs/promises'); -const backstop = require('htmlacademy-backstopjs'); -const fontsConfig = require('./test-config/backstop-test-05.config'); -const ppConfig = require('./test-config/backstop-test-06.config'); -const styleguideConfig = require('./test-config/backstop-test-07.config'); -const interactionConfig = require('./test-config/backstop-test-08.config'); +import fs from 'node:fs/promises'; +import backstop from 'htmlacademy-backstopjs'; +import fontsConfig from './test-config/backstop-test-05.config.js'; +import ppConfig from './test-config/backstop-test-06.config.js'; +import styleguideConfig from './test-config/backstop-test-07.config.js'; +import interactionConfig from './test-config/backstop-test-08.config.js'; -(async () => { - let passedSelectors - try { - await backstop('test', { config: ppConfig }); - console.log('All blocks passed') - } catch (e) { - console.log('mismatch blocks detected') - } finally { - const reportFile = await fs.readFile('./backstop_data/json_report/jsonReport.json', 'utf8') - const report = JSON.parse(reportFile) - const passed = report.tests - .filter(({ status }) => status === 'pass') - .map(({ - pair: { - viewportLabel, - selector, - }, - }) => ({ selector, viewportLabel })) - .reduce((acc, { viewportLabel, selector }) => { - acc[selector] = acc[selector]?.add(viewportLabel) ?? new Set([viewportLabel]); - return acc; - }, {}) - passedSelectors = Object.entries(passed) - .filter(([, set]) => set.size === 3) - .map(([selector]) => { - let pattern = /data-test="(\w+)"/; - let match = selector.match(pattern); - return (match[1]); - }).join('|') || 'nothing good found' - } - const config05 = { - ...fontsConfig, - scenarios: fontsConfig.scenarios.filter(({ label }) => !!label.match(passedSelectors)), - } - await fs.writeFile('./test-config/backstop-test-05.config.json', JSON.stringify(config05, null, 2), 'utf8') +let passedSelectors +try { + await backstop('test', { config: ppConfig }); + console.log('All blocks passed') +} catch (e) { + console.log('mismatch blocks detected') +} finally { + const reportFile = await fs.readFile('./backstop_data/json_report/jsonReport.json', 'utf8') + const report = JSON.parse(reportFile) + const passed = report.tests + .filter(({ status }) => status === 'pass') + .map(({ + pair: { + viewportLabel, + selector, + }, + }) => ({ selector, viewportLabel })) + .reduce((acc, { viewportLabel, selector }) => { + acc[selector] = acc[selector]?.add(viewportLabel) ?? new Set([viewportLabel]); + return acc; + }, {}) + passedSelectors = Object.entries(passed) + .filter(([, set]) => set.size === 3) + .map(([selector]) => { + let pattern = /data-test="(\w+)"/; + let match = selector.match(pattern); + return (match[1]); + }).join('|') || 'nothing good found' +} +const config05 = { + ...fontsConfig, + scenarios: fontsConfig.scenarios.filter(({ label }) => !!label.match(passedSelectors)), +} +await fs.writeFile('./test-config/backstop-test-05.config.json', JSON.stringify(config05, null, 2), 'utf8') - const config08 = { - ...interactionConfig, - scenarios: interactionConfig.scenarios.filter(({ label }) => !!label.match(passedSelectors)), - } - await fs.writeFile('./test-config/backstop-test-08.config.json', JSON.stringify(config08, null, 2), 'utf8') +const config08 = { + ...interactionConfig, + scenarios: interactionConfig.scenarios.filter(({ label }) => !!label.match(passedSelectors)), +} +await fs.writeFile('./test-config/backstop-test-08.config.json', JSON.stringify(config08, null, 2), 'utf8') - const config07 = { - ...styleguideConfig, - scenarios: styleguideConfig.scenarios.filter(({ label }) => !!label.match(passedSelectors)), - } - await fs.writeFile('./test-config/backstop-test-07.config.json', JSON.stringify(config07, null, 2), 'utf8') +const config07 = { + ...styleguideConfig, + scenarios: styleguideConfig.scenarios.filter(({ label }) => !!label.match(passedSelectors)), +} +await fs.writeFile('./test-config/backstop-test-07.config.json', JSON.stringify(config07, null, 2), 'utf8') -})() diff --git a/test-config/backstop-html-03.config.js b/test-config/backstop-html-03.config.js index e5393a7c..3efc9c35 100644 --- a/test-config/backstop-html-03.config.js +++ b/test-config/backstop-html-03.config.js @@ -1,4 +1,4 @@ -module.exports = { +export default { "id": "drink2go html-03", "viewports": [ { diff --git a/test-config/backstop-html-04.config.js b/test-config/backstop-html-04.config.js index 74e07c30..7e9f3a18 100644 --- a/test-config/backstop-html-04.config.js +++ b/test-config/backstop-html-04.config.js @@ -3,7 +3,7 @@ * todo прокликать контролы формы * */ -module.exports = { +export default { "id": "drink2go form", "viewports": [ { diff --git a/test-config/backstop-img-01.config.js b/test-config/backstop-img-01.config.js index baaade30..06281bf6 100644 --- a/test-config/backstop-img-01.config.js +++ b/test-config/backstop-img-01.config.js @@ -1,4 +1,4 @@ -module.exports = { +export default { "id": "drink2go img", "viewports": [ { diff --git a/test-config/backstop-test-03.config.js b/test-config/backstop-test-03.config.js index c95d4d41..0ea95ea6 100644 --- a/test-config/backstop-test-03.config.js +++ b/test-config/backstop-test-03.config.js @@ -18,7 +18,7 @@ */ -module.exports = { +export default { "id": "drink2go test-03", "viewports": [ { diff --git a/test-config/backstop-test-04.config.js b/test-config/backstop-test-04.config.js index f6c98b94..ba6cb0eb 100644 --- a/test-config/backstop-test-04.config.js +++ b/test-config/backstop-test-04.config.js @@ -19,7 +19,7 @@ */ const indexSections = ['header', 'main', 'hero', 'features', 'catalog', 'map', 'footer']; -module.exports = { +export default { "id": "drink2go test-04", "viewports": [ { diff --git a/test-config/backstop-test-05.config.js b/test-config/backstop-test-05.config.js index 761621fc..5d10ac51 100644 --- a/test-config/backstop-test-05.config.js +++ b/test-config/backstop-test-05.config.js @@ -17,7 +17,7 @@ const indexSections = [ { section: 'catalog', content: ['Каталог кофейных напитков', 'Цена', 'Наличие молока', 'Неважно', 'Страна произрастания', 'Бразилия', 'Эфиопия', 'Перу', 'Применить', 'Сбросить', 'Сортировка', 'Кофе без кофеина из Эфиопии', 'В корзину'] }, { section: 'footer', content: ['Способы оплаты', 'Медиа', 'Санкт-Петербург', 'Разработано', 'HTML Academy'] } ]; -module.exports = { +export default { "id": "drink2go test-05", "viewports": [ { diff --git a/test-config/backstop-test-06.config.js b/test-config/backstop-test-06.config.js index 5e765ae5..3212f13e 100644 --- a/test-config/backstop-test-06.config.js +++ b/test-config/backstop-test-06.config.js @@ -7,7 +7,7 @@ const indexSections = [ { section: 'footer', misMatchThreshold: 3 } ]; -module.exports = { +export default { "id": "drink2go test-06", "viewports": [ { @@ -26,7 +26,7 @@ module.exports = { "height": 800, }, ], - "onReadyScript": "onReady.js", + "onReadyScript": "onReady.cjs", "resembleOutputOptions": { "ignoreAntialiasing": true, "errorType": "movementDifferenceIntensity", diff --git a/test-config/backstop-test-07.config.js b/test-config/backstop-test-07.config.js index 62131551..7506959a 100644 --- a/test-config/backstop-test-07.config.js +++ b/test-config/backstop-test-07.config.js @@ -20,7 +20,7 @@ const indexSections = [ ] } ].flatMap(item => item.content.map(content => ({ ...item, content }))); -module.exports = { +export default { "id": "drink2go test-07", "viewports": [ { diff --git a/test-config/backstop-test-08.config.js b/test-config/backstop-test-08.config.js index 0823c455..92497999 100644 --- a/test-config/backstop-test-08.config.js +++ b/test-config/backstop-test-08.config.js @@ -11,7 +11,7 @@ const indexSections = [ { section: 'footer' } ]; -module.exports = { +export default { "id": "drink2go test-08", "viewports": [{ "label": "desktop", diff --git a/test-config/backstop-test-ff.config.js b/test-config/backstop-test-ff.config.js index d93b047d..959bbb7d 100644 --- a/test-config/backstop-test-ff.config.js +++ b/test-config/backstop-test-ff.config.js @@ -1,6 +1,6 @@ const indexSections = ['header', 'main', 'hero', 'features', 'catalog', 'map', 'footer']; -module.exports = { +export default { "id": "drink2go FF", "viewports": [ { diff --git a/test-config/backstop-test-logo.config.js b/test-config/backstop-test-logo.config.js index d64188f8..1b03279f 100644 --- a/test-config/backstop-test-logo.config.js +++ b/test-config/backstop-test-logo.config.js @@ -1,5 +1,5 @@ -module.exports = { +export default { "id": "drink2go Logo", "viewports": [ { diff --git a/test-config/backstop-test-menu.config.js b/test-config/backstop-test-menu.config.js index 6c61111e..fbbe473a 100644 --- a/test-config/backstop-test-menu.config.js +++ b/test-config/backstop-test-menu.config.js @@ -1,4 +1,4 @@ -module.exports = { +export default { "id": "drink2go menu", "viewports": [ { diff --git a/test-config/backstop-test-swiper.config.js b/test-config/backstop-test-swiper.config.js index 16b89fee..d1bf4b3d 100644 --- a/test-config/backstop-test-swiper.config.js +++ b/test-config/backstop-test-swiper.config.js @@ -1,4 +1,4 @@ -module.exports = { +export default { "id": "drink2go swiper", "viewports": [ {