creevey
Version:
creevey is a tool for automated visual testing, that tightly integrated with storybook
300 lines (260 loc) • 43.8 kB
JavaScript
;
Object.defineProperty(exports, "__esModule", {
value: true
});
exports.getBrowser = getBrowser;
exports.switchStory = switchStory;
var _https = _interopRequireDefault(require("https"));
var _pngjs = require("pngjs");
var _seleniumWebdriver = require("selenium-webdriver");
function _interopRequireDefault(obj) { return obj && obj.__esModule ? obj : { default: obj }; }
const LOCALHOST_REGEXP = /(localhost|127\.0\.0\.1)/gi;
const TESTKONTUR_REGEXP = /testkontur/gi;
function getRealIp() {
return new Promise((resolve, reject) => _https.default.get('https://fake.testkontur.ru/ip', res => {
if (res.statusCode !== 200) {
return reject(new Error(`Couldn't resolve real ip for \`localhost\`. Status code: ${res.statusCode}`));
}
let data = '';
res.setEncoding('utf8');
res.on('data', chunk => data += chunk);
res.on('end', () => resolve(data));
}));
}
async function resetMousePosition(browser) {
const isChrome = (await browser.getCapabilities()).get('browserName') == 'chrome';
const {
top,
left,
width,
height
} = await browser.executeScript(function () {
/* eslint-disable no-var */
// NOTE On storybook >= 4.x already reset scroll
window.scrollTo(0, 0);
var bodyRect = document.body.getBoundingClientRect();
return {
top: bodyRect.top,
left: bodyRect.left,
width: bodyRect.width,
height: bodyRect.height
};
/* eslint-enable no-var */
});
if (isChrome) {
// NOTE Bridge mode not support move mouse relative viewport
await browser.actions({
bridge: true
}).move({
origin: browser.findElement(_seleniumWebdriver.By.css('body')),
x: Math.ceil(-1 * width / 2) - left,
y: Math.ceil(-1 * height / 2) - top
}).perform();
} else {
// NOTE Firefox for some reason moving by 0 x 0 move cursor in bottom left corner :sad:
// NOTE IE don't emit move events until force window focus or connect by RDP on virtual machine
await browser.actions().move({
origin: _seleniumWebdriver.Origin.VIEWPORT,
x: 0,
y: 1
}).perform();
}
}
async function resizeViewport(browser, viewport) {
const windowRect = await browser.manage().window().getRect();
const {
innerWidth,
innerHeight
} = await browser.executeScript(function () {
return {
innerWidth: window.innerWidth,
innerHeight: window.innerHeight
};
});
const dWidth = windowRect.width - innerWidth;
const dHeight = windowRect.height - innerHeight;
await browser.manage().window().setRect({
width: viewport.width + dWidth,
height: viewport.height + dHeight
});
}
function disableAnimations(browser) {
const disableAnimationsStyles = `
*,
*:hover,
*::before,
*::after {
animation-delay: -0.0001ms !important;
animation-duration: 0s !important;
animation-play-state: paused !important;
cursor: none !important;
caret-color: transparent !important;
transition: 0s !important;
}
`;
return browser.executeScript(function (stylesheet) {
/* eslint-disable no-var */
var style = document.createElement('style');
var textNode = document.createTextNode(stylesheet);
style.setAttribute('type', 'text/css');
style.appendChild(textNode);
document.head.appendChild(style);
/* eslint-enable no-var */
}, disableAnimationsStyles);
}
const getScrollBarWidth = (() => {
let scrollBarWidth = null;
return async browser => {
if (scrollBarWidth != null) return Promise.resolve(scrollBarWidth);
scrollBarWidth = await browser.executeScript(function () {
// eslint-disable-next-line no-var
var div = document.createElement('div');
div.innerHTML = 'a'; // NOTE: In IE clientWidth is 0 if this div is empty.
div.style.overflowY = 'scroll';
document.body.appendChild(div); // eslint-disable-next-line no-var
var widthDiff = div.offsetWidth - div.clientWidth;
document.body.removeChild(div);
return widthDiff;
});
return scrollBarWidth;
};
})();
async function takeCompositeScreenshot(browser, windowRect, elementRect) {
const screens = [];
const browserName = (await browser.getCapabilities()).get('browserName'); // NOTE Firefox and Safari take viewport screenshot without scrollbars
const isScreenshotWithoutScrollBar = browserName == 'firefox' || browserName == 'Safari';
const scrollBarWidth = await getScrollBarWidth(browser); // NOTE Sometimes viewport has been scrolled somewhere
const normalizedElementRect = {
left: elementRect.left - windowRect.x,
right: elementRect.right - windowRect.x,
top: elementRect.top - windowRect.y,
bottom: elementRect.bottom - windowRect.y
};
const isFitHorizontally = windowRect.width >= elementRect.width + normalizedElementRect.left;
const isFitVertically = windowRect.height >= elementRect.height + normalizedElementRect.top;
const viewportWidth = windowRect.width - (isFitVertically ? 0 : scrollBarWidth);
const viewportHeight = windowRect.height - (isFitHorizontally ? 0 : scrollBarWidth);
const cols = Math.ceil(elementRect.width / viewportWidth);
const rows = Math.ceil(elementRect.height / viewportHeight);
const xOffset = Math.round(isFitHorizontally ? normalizedElementRect.left : Math.max(0, cols * viewportWidth - elementRect.width));
const yOffset = Math.round(isFitVertically ? normalizedElementRect.top : Math.max(0, rows * viewportHeight - elementRect.height));
for (let row = 0; row < rows; row += 1) {
for (let col = 0; col < cols; col += 1) {
const dx = Math.min(viewportWidth * col + normalizedElementRect.left, Math.max(0, normalizedElementRect.right - viewportWidth));
const dy = Math.min(viewportHeight * row + normalizedElementRect.top, Math.max(0, normalizedElementRect.bottom - viewportHeight));
await browser.executeScript(function (x, y) {
window.scrollTo(x, y);
}, dx, dy);
screens.push((await browser.takeScreenshot()));
}
}
const images = screens.map(s => Buffer.from(s, 'base64')).map(b => _pngjs.PNG.sync.read(b));
const compositeImage = new _pngjs.PNG({
width: Math.round(elementRect.width),
height: Math.round(elementRect.height)
});
for (let y = 0; y < compositeImage.height; y += 1) {
for (let x = 0; x < compositeImage.width; x += 1) {
const col = Math.floor(x / viewportWidth);
const row = Math.floor(y / viewportHeight);
const isLastCol = cols - col == 1;
const isLastRow = rows - row == 1;
const scrollOffset = isFitVertically || isScreenshotWithoutScrollBar ? 0 : scrollBarWidth;
const i = (y * compositeImage.width + x) * 4;
const j = // NOTE compositeImage(x, y) => image(x, y)
(y % viewportHeight * (viewportWidth + scrollOffset) + x % viewportWidth) * 4 + ( // NOTE Offset for last row/col image
isLastRow ? yOffset * (viewportWidth + scrollOffset) * 4 : 0) + (isLastCol ? xOffset * 4 : 0);
const image = images[row * cols + col];
compositeImage.data[i + 0] = image.data[j + 0];
compositeImage.data[i + 1] = image.data[j + 1];
compositeImage.data[i + 2] = image.data[j + 2];
compositeImage.data[i + 3] = image.data[j + 3];
}
}
return _pngjs.PNG.sync.write(compositeImage).toString('base64');
}
async function takeScreenshot(browser, captureElement) {
if (!captureElement) return browser.takeScreenshot();
const {
elementRect,
windowRect
} = await browser.executeScript(function (selector) {
var _document$querySelect;
window.scrollTo(0, 0);
return {
elementRect: (_document$querySelect = document.querySelector(selector)) === null || _document$querySelect === void 0 ? void 0 : _document$querySelect.getBoundingClientRect(),
windowRect: {
width: window.innerWidth,
height: window.innerHeight,
x: Math.round(window.scrollX),
y: Math.round(window.scrollY)
}
};
}, captureElement);
const isFitIntoViewport = elementRect.width + elementRect.left <= windowRect.width && elementRect.height + elementRect.top <= windowRect.height;
if (isFitIntoViewport) return browser.findElement(_seleniumWebdriver.By.css(captureElement)).takeScreenshot();
return takeCompositeScreenshot(browser, windowRect, elementRect);
}
async function selectStory(browser, storyId, kind, story) {
const errorMessage = await browser.executeAsyncScript(function (storyId, kind, name, callback) {
window.__CREEVEY_SELECT_STORY__(storyId, kind, name, callback);
}, storyId, kind, story);
if (errorMessage) throw new Error(errorMessage);
}
async function getBrowser(config, browserConfig) {
const {
gridUrl = config.gridUrl,
storybookUrl: address = config.storybookUrl,
limit,
testRegex,
viewport,
...capabilities
} = browserConfig;
void limit;
void testRegex;
let realAddress = address;
if (LOCALHOST_REGEXP.test(address) && TESTKONTUR_REGEXP.test(gridUrl)) {
realAddress = address.replace(LOCALHOST_REGEXP, (await getRealIp()));
}
const browser = await new _seleniumWebdriver.Builder().usingServer(gridUrl).withCapabilities(capabilities).build();
if (viewport) {
await resizeViewport(browser, viewport);
}
try {
await browser.get(`${realAddress}/iframe.html`);
await browser.wait(_seleniumWebdriver.until.elementLocated(_seleniumWebdriver.By.css('#root')), 10000);
} catch (_) {
throw new Error(`Can't load storybook root page by URL ${realAddress}/iframe.html`);
}
await disableAnimations(browser);
return browser;
}
async function switchStory() {
var _this$currentTest, _this$currentTest$ctx, _story$parameters$cre;
let testOrSuite = this.currentTest;
if (!testOrSuite) throw new Error("Can't switch story, because test context doesn't have 'currentTest' field");
this.testScope.length = 0;
this.testScope.push(this.browserName);
while ((_testOrSuite = testOrSuite) === null || _testOrSuite === void 0 ? void 0 : _testOrSuite.title) {
var _testOrSuite;
this.testScope.push(testOrSuite.title);
testOrSuite = testOrSuite.parent;
}
const story = (_this$currentTest = this.currentTest) === null || _this$currentTest === void 0 ? void 0 : (_this$currentTest$ctx = _this$currentTest.ctx) === null || _this$currentTest$ctx === void 0 ? void 0 : _this$currentTest$ctx.story;
if (!story) throw new Error(`Current test '${this.testScope.join('/')}' context doesn't have 'story' field`);
await resetMousePosition(this.browser);
await selectStory(this.browser, story.id, story.kind, story.name);
const {
captureElement
} = (_story$parameters$cre = story.parameters.creevey) !== null && _story$parameters$cre !== void 0 ? _story$parameters$cre : {};
if (captureElement) Object.defineProperty(this, 'captureElement', {
enumerable: true,
configurable: true,
get() {
return this.browser.findElement(_seleniumWebdriver.By.css(captureElement));
}
});else Reflect.deleteProperty(this, 'captureElement');
this.takeScreenshot = () => takeScreenshot(this.browser, captureElement);
this.testScope.reverse();
}
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../src/selenium.ts"],"names":["LOCALHOST_REGEXP","TESTKONTUR_REGEXP","getRealIp","Promise","resolve","reject","https","get","res","statusCode","Error","data","setEncoding","on","chunk","resetMousePosition","browser","isChrome","getCapabilities","top","left","width","height","executeScript","window","scrollTo","bodyRect","document","body","getBoundingClientRect","actions","bridge","move","origin","findElement","By","css","x","Math","ceil","y","perform","Origin","VIEWPORT","resizeViewport","viewport","windowRect","manage","getRect","innerWidth","innerHeight","dWidth","dHeight","setRect","disableAnimations","disableAnimationsStyles","stylesheet","style","createElement","textNode","createTextNode","setAttribute","appendChild","head","getScrollBarWidth","scrollBarWidth","div","innerHTML","overflowY","widthDiff","offsetWidth","clientWidth","removeChild","takeCompositeScreenshot","elementRect","screens","browserName","isScreenshotWithoutScrollBar","normalizedElementRect","right","bottom","isFitHorizontally","isFitVertically","viewportWidth","viewportHeight","cols","rows","xOffset","round","max","yOffset","row","col","dx","min","dy","push","takeScreenshot","images","map","s","Buffer","from","b","PNG","sync","read","compositeImage","floor","isLastCol","isLastRow","scrollOffset","i","j","image","write","toString","captureElement","selector","querySelector","scrollX","scrollY","isFitIntoViewport","selectStory","storyId","kind","story","errorMessage","executeAsyncScript","name","callback","__CREEVEY_SELECT_STORY__","getBrowser","config","browserConfig","gridUrl","storybookUrl","address","limit","testRegex","capabilities","realAddress","test","replace","Builder","usingServer","withCapabilities","build","wait","until","elementLocated","_","switchStory","testOrSuite","currentTest","testScope","length","title","parent","ctx","join","id","parameters","creevey","Object","defineProperty","enumerable","configurable","Reflect","deleteProperty","reverse"],"mappings":"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","sourcesContent":["import https from 'https';\nimport { PNG } from 'pngjs';\nimport { Context, Test, Suite } from 'mocha';\nimport { Builder, By, until, WebDriver, Origin } from 'selenium-webdriver';\nimport { Config, BrowserConfig, StoryInput } from './types';\n\ndeclare global {\n  interface Window {\n    __CREEVEY_RESTORE_SCROLL__?: () => void;\n  }\n}\n\nconst LOCALHOST_REGEXP = /(localhost|127\\.0\\.0\\.1)/gi;\nconst TESTKONTUR_REGEXP = /testkontur/gi;\n\nfunction getRealIp(): Promise<string> {\n  return new Promise((resolve, reject) =>\n    https.get('https://fake.testkontur.ru/ip', (res) => {\n      if (res.statusCode !== 200) {\n        return reject(new Error(`Couldn't resolve real ip for \\`localhost\\`. Status code: ${res.statusCode}`));\n      }\n\n      let data = '';\n\n      res.setEncoding('utf8');\n      res.on('data', (chunk) => (data += chunk));\n      res.on('end', () => resolve(data));\n    }),\n  );\n}\n\nasync function resetMousePosition(browser: WebDriver): Promise<void> {\n  const isChrome = (await browser.getCapabilities()).get('browserName') == 'chrome';\n  const { top, left, width, height } = await browser.executeScript(function () {\n    /* eslint-disable no-var */\n    // NOTE On storybook >= 4.x already reset scroll\n    window.scrollTo(0, 0);\n\n    var bodyRect = document.body.getBoundingClientRect();\n    return {\n      top: bodyRect.top,\n      left: bodyRect.left,\n      width: bodyRect.width,\n      height: bodyRect.height,\n    };\n    /* eslint-enable no-var */\n  });\n\n  if (isChrome) {\n    // NOTE Bridge mode not support move mouse relative viewport\n    await browser\n      .actions({ bridge: true })\n      .move({\n        origin: browser.findElement(By.css('body')),\n        x: Math.ceil((-1 * width) / 2) - left,\n        y: Math.ceil((-1 * height) / 2) - top,\n      })\n      .perform();\n  } else {\n    // NOTE Firefox for some reason moving by 0 x 0 move cursor in bottom left corner :sad:\n    // NOTE IE don't emit move events until force window focus or connect by RDP on virtual machine\n    await browser.actions().move({ origin: Origin.VIEWPORT, x: 0, y: 1 }).perform();\n  }\n}\n\nasync function resizeViewport(browser: WebDriver, viewport: { width: number; height: number }): Promise<void> {\n  const windowRect = await browser.manage().window().getRect();\n  const { innerWidth, innerHeight } = await browser.executeScript(function () {\n    return {\n      innerWidth: window.innerWidth,\n      innerHeight: window.innerHeight,\n    };\n  });\n  const dWidth = windowRect.width - innerWidth;\n  const dHeight = windowRect.height - innerHeight;\n  await browser\n    .manage()\n    .window()\n    .setRect({\n      width: viewport.width + dWidth,\n      height: viewport.height + dHeight,\n    });\n}\n\nfunction disableAnimations(browser: WebDriver): Promise<void> {\n  const disableAnimationsStyles = `\n*,\n*:hover,\n*::before,\n*::after {\n  animation-delay: -0.0001ms !important;\n  animation-duration: 0s !important;\n  animation-play-state: paused !important;\n  cursor: none !important;\n  caret-color: transparent !important;\n  transition: 0s !important;\n}\n`;\n  return browser.executeScript(function (stylesheet: string) {\n    /* eslint-disable no-var */\n    var style = document.createElement('style');\n    var textNode = document.createTextNode(stylesheet);\n    style.setAttribute('type', 'text/css');\n    style.appendChild(textNode);\n    document.head.appendChild(style);\n    /* eslint-enable no-var */\n  }, disableAnimationsStyles);\n}\n\nconst getScrollBarWidth: (browser: WebDriver) => Promise<number> = (() => {\n  let scrollBarWidth: number | null = null;\n\n  return async (browser: WebDriver): Promise<number> => {\n    if (scrollBarWidth != null) return Promise.resolve(scrollBarWidth);\n    scrollBarWidth = await browser.executeScript<number>(function () {\n      // eslint-disable-next-line no-var\n      var div = document.createElement('div');\n      div.innerHTML = 'a'; // NOTE: In IE clientWidth is 0 if this div is empty.\n      div.style.overflowY = 'scroll';\n      document.body.appendChild(div);\n      // eslint-disable-next-line no-var\n      var widthDiff = div.offsetWidth - div.clientWidth;\n      document.body.removeChild(div);\n\n      return widthDiff;\n    });\n    return scrollBarWidth;\n  };\n})();\n\nasync function takeCompositeScreenshot(\n  browser: WebDriver,\n  windowRect: { width: number; height: number; x: number; y: number },\n  elementRect: DOMRect,\n): Promise<string> {\n  const screens = [];\n  const browserName = (await browser.getCapabilities()).get('browserName');\n  // NOTE Firefox and Safari take viewport screenshot without scrollbars\n  const isScreenshotWithoutScrollBar = browserName == 'firefox' || browserName == 'Safari';\n  const scrollBarWidth = await getScrollBarWidth(browser);\n  // NOTE Sometimes viewport has been scrolled somewhere\n  const normalizedElementRect = {\n    left: elementRect.left - windowRect.x,\n    right: elementRect.right - windowRect.x,\n    top: elementRect.top - windowRect.y,\n    bottom: elementRect.bottom - windowRect.y,\n  };\n  const isFitHorizontally = windowRect.width >= elementRect.width + normalizedElementRect.left;\n  const isFitVertically = windowRect.height >= elementRect.height + normalizedElementRect.top;\n  const viewportWidth = windowRect.width - (isFitVertically ? 0 : scrollBarWidth);\n  const viewportHeight = windowRect.height - (isFitHorizontally ? 0 : scrollBarWidth);\n  const cols = Math.ceil(elementRect.width / viewportWidth);\n  const rows = Math.ceil(elementRect.height / viewportHeight);\n  const xOffset = Math.round(\n    isFitHorizontally ? normalizedElementRect.left : Math.max(0, cols * viewportWidth - elementRect.width),\n  );\n  const yOffset = Math.round(\n    isFitVertically ? normalizedElementRect.top : Math.max(0, rows * viewportHeight - elementRect.height),\n  );\n\n  for (let row = 0; row < rows; row += 1) {\n    for (let col = 0; col < cols; col += 1) {\n      const dx = Math.min(\n        viewportWidth * col + normalizedElementRect.left,\n        Math.max(0, normalizedElementRect.right - viewportWidth),\n      );\n      const dy = Math.min(\n        viewportHeight * row + normalizedElementRect.top,\n        Math.max(0, normalizedElementRect.bottom - viewportHeight),\n      );\n      await browser.executeScript(\n        function (x: number, y: number) {\n          window.scrollTo(x, y);\n        },\n        dx,\n        dy,\n      );\n      screens.push(await browser.takeScreenshot());\n    }\n  }\n\n  const images = screens.map((s) => Buffer.from(s, 'base64')).map((b) => PNG.sync.read(b));\n  const compositeImage = new PNG({ width: Math.round(elementRect.width), height: Math.round(elementRect.height) });\n\n  for (let y = 0; y < compositeImage.height; y += 1) {\n    for (let x = 0; x < compositeImage.width; x += 1) {\n      const col = Math.floor(x / viewportWidth);\n      const row = Math.floor(y / viewportHeight);\n      const isLastCol = cols - col == 1;\n      const isLastRow = rows - row == 1;\n      const scrollOffset = isFitVertically || isScreenshotWithoutScrollBar ? 0 : scrollBarWidth;\n      const i = (y * compositeImage.width + x) * 4;\n      const j =\n        // NOTE compositeImage(x, y) => image(x, y)\n        ((y % viewportHeight) * (viewportWidth + scrollOffset) + (x % viewportWidth)) * 4 +\n        // NOTE Offset for last row/col image\n        (isLastRow ? yOffset * (viewportWidth + scrollOffset) * 4 : 0) +\n        (isLastCol ? xOffset * 4 : 0);\n      const image = images[row * cols + col];\n      compositeImage.data[i + 0] = image.data[j + 0];\n      compositeImage.data[i + 1] = image.data[j + 1];\n      compositeImage.data[i + 2] = image.data[j + 2];\n      compositeImage.data[i + 3] = image.data[j + 3];\n    }\n  }\n  return PNG.sync.write(compositeImage).toString('base64');\n}\n\nasync function takeScreenshot(browser: WebDriver, captureElement?: string | null): Promise<string> {\n  if (!captureElement) return browser.takeScreenshot();\n\n  const { elementRect, windowRect } = await browser.executeScript(function (selector: string) {\n    window.scrollTo(0, 0);\n    return {\n      elementRect: document.querySelector(selector)?.getBoundingClientRect(),\n      windowRect: {\n        width: window.innerWidth,\n        height: window.innerHeight,\n        x: Math.round(window.scrollX),\n        y: Math.round(window.scrollY),\n      },\n    };\n  }, captureElement);\n\n  const isFitIntoViewport =\n    elementRect.width + elementRect.left <= windowRect.width &&\n    elementRect.height + elementRect.top <= windowRect.height;\n\n  if (isFitIntoViewport) return browser.findElement(By.css(captureElement)).takeScreenshot();\n\n  return takeCompositeScreenshot(browser, windowRect, elementRect);\n}\n\nasync function selectStory(browser: WebDriver, storyId: string, kind: string, story: string): Promise<void> {\n  const errorMessage = await browser.executeAsyncScript<string | undefined>(\n    function (storyId: string, kind: string, name: string, callback: (error?: string) => void) {\n      window.__CREEVEY_SELECT_STORY__(storyId, kind, name, callback);\n    },\n    storyId,\n    kind,\n    story,\n  );\n  if (errorMessage) throw new Error(errorMessage);\n}\n\nexport async function getBrowser(config: Config, browserConfig: BrowserConfig): Promise<WebDriver> {\n  const {\n    gridUrl = config.gridUrl,\n    storybookUrl: address = config.storybookUrl,\n    limit,\n    testRegex,\n    viewport,\n    ...capabilities\n  } = browserConfig;\n  void limit;\n  void testRegex;\n  let realAddress = address;\n  if (LOCALHOST_REGEXP.test(address) && TESTKONTUR_REGEXP.test(gridUrl)) {\n    realAddress = address.replace(LOCALHOST_REGEXP, await getRealIp());\n  }\n  const browser = await new Builder().usingServer(gridUrl).withCapabilities(capabilities).build();\n\n  if (viewport) {\n    await resizeViewport(browser, viewport);\n  }\n  try {\n    await browser.get(`${realAddress}/iframe.html`);\n    await browser.wait(until.elementLocated(By.css('#root')), 10000);\n  } catch (_) {\n    throw new Error(`Can't load storybook root page by URL ${realAddress}/iframe.html`);\n  }\n  await disableAnimations(browser);\n\n  return browser;\n}\n\nexport async function switchStory(this: Context): Promise<void> {\n  let testOrSuite: Test | Suite | undefined = this.currentTest;\n\n  if (!testOrSuite) throw new Error(\"Can't switch story, because test context doesn't have 'currentTest' field\");\n\n  this.testScope.length = 0;\n  this.testScope.push(this.browserName);\n  while (testOrSuite?.title) {\n    this.testScope.push(testOrSuite.title);\n    testOrSuite = testOrSuite.parent;\n  }\n  const story: StoryInput | undefined = this.currentTest?.ctx?.story;\n\n  if (!story) throw new Error(`Current test '${this.testScope.join('/')}' context doesn't have 'story' field`);\n\n  await resetMousePosition(this.browser);\n  await selectStory(this.browser, story.id, story.kind, story.name);\n\n  const { captureElement } = story.parameters.creevey ?? {};\n\n  if (captureElement)\n    Object.defineProperty(this, 'captureElement', {\n      enumerable: true,\n      configurable: true,\n      get() {\n        return this.browser.findElement(By.css(captureElement));\n      },\n    });\n  else Reflect.deleteProperty(this, 'captureElement');\n\n  this.takeScreenshot = () => takeScreenshot(this.browser, captureElement);\n\n  this.testScope.reverse();\n}\n"]}