javascript - puppeteer 师 |等待所有 JavaScript 执行完毕

标签 javascript node.js google-chrome testing puppeteer

我尝试从多个页面截取屏幕截图,这些页面应该已完全加载(包括延迟加载的图像)以供以后比较。

我找到了 lazyimages_without_scroll_events.js example这很有帮助。

使用以下代码,屏幕截图看起来不错,但存在一些主要问题。

async function takeScreenshot(browser, viewport, route) {
  return browser.newPage().then(async (page) => {
    const fileName = `${viewport.directory}/${getFilename(route)}`;

    await page.setViewport({
      width: viewport.width,
      height: 500,
    });
    await page.goto(
        `${config.server.master}${route}.html`,
        {
          waitUntil: 'networkidle0',
        }
    );
    await page.evaluate(() => {
      /* global document,requestAnimationFrame */
      let lastScrollTop = document.scrollingElement.scrollTop;

      // Scroll to bottom of page until we can't scroll anymore.
      const scroll = () => {
        document.scrollingElement.scrollTop += 100;
        if (document.scrollingElement.scrollTop !== lastScrollTop) {
          lastScrollTop = document.scrollingElement.scrollTop;
          requestAnimationFrame(scroll);
        }
      };
      scroll();
    });
    await page.waitFor(5000);
    await page.screenshot({
      path: `screenshots/master/${fileName}.png`,
      fullPage: true,
    });

    await page.close();
    console.log(`Viewport "${viewport.name}", Route "${route}"`);
  });
}

问题:即使 page.waitFor()(超时)的值更高,有时页面上的所有前端相关 JavaScript 也未完全执行。

对于某些 JavaScript 可以更改前端的旧页面。 F.e.在一个遗留案例中是 jQuery.matchHeight

最佳情况:在理想情况下,Puppeteer 会等到所有 JavaScript 都被评估和执行。 这样的事情可能吗?


编辑
cody-g 的帮助下,我可以稍微改进脚本。

function jQueryMatchHeightIsProcessed() {
  return Array.from($('.match-height')).every((element) => {
    return element.style.height !== '';
  });
}

// Within takeScreenshot() after page.waitFor()
await page.waitForFunction(jQueryMatchHeightIsProcessed, {timeout: 0});

...但它远非完美。看来我必须为不同的前端脚本找到类似的解决方案,才能真正考虑目标页面上发生的一切。

在我的例子中,jQuery.matchHeight 的主要问题是它在不同的运行中处理不同的高度。可能是由图像延迟加载引起的。看来要等到可以用 Flexbox 代替了。 (^_^)°

其他需要解决的问题:

禁用动画:

await page.addStyleTag({
  content: `
    * {
      transition: none !important;
      animation: none !important;
    }
  `,
});

处理幻灯片:

function handleSwiperSlideshows() {
  Array.from($('.swiper-container')).forEach((element) => {
    if (typeof element.swiper !== 'undefined') {
      if (element.swiper.autoplaying) {
        element.swiper.stopAutoplay();
        element.swiper.slideTo(0);
      }
    }
  });
}

// Within takeScreenshot() after page.waitFor()
await page.evaluate(handleSwiperSlideshows);

但还是不够。我认为不可能对这些遗留页面进行可视化测试。

最佳答案

以下waitForFunction可能对你有用,你可以用它来等待任意函数的计算结果为真。如果您有权访问页面的代码,则可以设置窗口状态并使用它来通知 puppeteer 可以安全地继续,或者只是依赖某种其他就绪状态。 注意:此函数是一个轮询函数,并在可以指定的某个时间间隔重新评估。

const watchDog = page.waitForFunction('<your function to evaluate to true>');

例如,

const watchDog = page.waitForFunction('window.status === "ready"');
await watchDog;

在您的页面代码中,您只需将 window.status 设置为 ready

要在多个异步文件中使用多个看门狗,您可以这样做

索引.js

...import/require file1.js;
...import/require file2.js;
...code...

文件1.js:

var file1Flag=false; // global
...code...
file1Flag=true;

文件2.js:

var file2Flag=false; // global
...code...
file2Flag=true;

主要.js:

const watchDog = page.waitForFunction('file1Flag && file2Flag');
await watchDog;

关于javascript - puppeteer 师 |等待所有 JavaScript 执行完毕,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/53483578/

相关文章:

javascript - 带有 MxN 矩阵的 jQuery 动画

javascript - React Google map API initMap 不是函数,没有包

html - iframe 上的边框?不在 Chrome 中

javascript - 从 Xamarin C# 应用程序将图像上传到 Node JS 服务器上的 Multer

google-chrome - 是什么导致某些网站嵌入在 Google Chrome 的 iframe 中?

javascript - Chrome 中光标位置异常位于输入值的末尾

javascript - 在 Google Maps API 中使用地址而不是经度和纬度

JavaScript 用逗号分割数组

node.js - 如何在 Electron 应用程序的主进程中对组件进行单元测试?

javascript - 如何使机器人不响应自身,但仍然响应其他机器人?