如何使用 Nextjs 和 Puppeteer 捕获网页屏幕截图

如何使用 nextjs 和 puppeteer 捕获网页屏幕截图

以编程方式捕获网页屏幕截图对于生成预览、创建基于图像的报告等非常有用。在本指南中,我们将构建一个 next.js api 路由,该路由采用 url 并生成 png 屏幕截图。我们的设置使用 puppeteer 和 chrome-aws-lambda 来利用无头 chrome 浏览器,使其多功能且可用于生产。

我们将首先设置一个新的 next.js 项目,并逐步浏览代码以了解 api 如何捕获屏幕截图。

先决条件

设置 next.js 应用使用 puppeteer 配置 api 路由为捕获接口创建react组件puppeteer 的本地与部署配置说明

开始新的 next.js 项目

创建一个新的 next.js 应用程序:

npx create-next-app@latest capture-image-appcd capture-image-app

安装必要的依赖项:

npm install puppeteer puppeteer-core chrome-aws-lambda busboy

第2步:创建生成屏幕截图的api路由

现在,我们将设置一个 api 端点来根据提供的 url 捕获并返回屏幕截图。

在pages/api文件夹中,创建一个名为generate-png.ts的新文件并添加以下代码:

import { nextapirequest, nextapiresponse } from "next";import busboy, { busboy } from "busboy"; // use busboy for multipart parsingimport chromium from "chrome-aws-lambda";import puppeteercore from "puppeteer-core"; // import puppeteer-core directlyimport puppeteer from "puppeteer"; // import puppeteer directly// conditional import for puppeteer based on the environmentconst puppeteermodule = process.env.node_env === "production" ? puppeteercore : puppeteer;export const config = {  api: {    bodyparser: false, // disable default body parsing to handle raw binary data (blob)  },};const delay = (ms: number): promise => new promise((resolve) => settimeout(resolve, ms));export default async function handler(  req: nextapirequest,  res: nextapiresponse): promise {  try {    if (req.method === "post") {      const bb: busboy = busboy({ headers: req.headers });      let width: number = 1920; // default width      let height: number = 0; // default height      let delaytime: number = 6000;      const buffers: buffer[] = [];      bb.on("file", (_name: string, file: nodejs.readablestream) => {        file.on("data", (data: buffer) => buffers.push(data));      });      bb.on("field", (name: string, value: string) => {        if (name === "width") width = parseint(value, 10) || 1920;        if (name === "height") height = parseint(value, 10) || 0;        if (name === "delay") delaytime = parseint(value, 10) || 6000;      });      bb.on("finish", async () => {        const blobbuffer: buffer = buffer.concat(buffers);        const htmlcontent: string = blobbuffer.tostring("utf-8");        const browser = await puppeteermodule.launch({          args: ["--start-maximized"],          executablepath: process.env.node_env === "production"            ? await chromium.executablepath || "/usr/bin/chromium-browser"            : undefined,  // no custom executable path needed for local          headless: true,        });        const page = await browser.newpage();        // load the html content directly        await page.setcontent(htmlcontent, { waituntil: "networkidle0" });        //@ts-expect-error todo        const bodyheight = await page.evaluate(() => {          return document.body.scrollheight; // get the full scrollable height of the body        });        await page.setviewport({          width: number(width),          height: height || bodyheight, // use the provided height or fallback to the full body height          devicescalefactor: 2,        });        await delay(delaytime);        const screenshotbuffer = await page.screenshot({          fullpage: !height,          type: "png",          omitbackground: false,        });        await browser.close();        res.setheader("content-type", "image/png");        res.setheader(          "content-disposition",          "attachment; filename=screenshot.png"        );        res.status(200).end(screenshotbuffer);      });      req.pipe(bb); // pipe the request stream to busboy    } else {      res.setheader("allow", ["post"]);      res.status(405).end(`method ${req.method} not allowed`);    }  } catch (error) {    console.error("error", error);    res.status(500).end("internal server error");  }}

*说明:为本地环境和生产环境选择 puppeteer
*

在此代码中,我们为 puppeteer 设置了动态导入:

本地开发:如果 node_env 不是生产环境,它会使用 puppeteer,它设置起来更简单,并且不需要 chrome-aws-lambda。

生产:对于无服务器部署,环境将检测 node_env 作为生产并加载 puppeteer-core 以及 chrome-aws-lambda,这使得它可以在 aws lambda 和其他类似环境中工作。在此设置中,chrome-aws-lambda 提供正确的 chromium 路径,确保与无服务器提供商的兼容性。

第 3 步:为 ui 创建一个简单的 react 组件

在这里,我们将创建一个简单的表单,让用户输入网页捕获的值。此表单将触发生成功能以捕获并下载 pdf 格式的屏幕截图。

import { useState } from "react";export default function ScreenCaptureComponent() {  const [isProcessing, setProcessing] = useState(false);  const [width, setWidth] = useState("1920");  const [height, setHeight] = useState("1000");  const [delay, setDelay] = useState("6000");  // Function to clone HTML and prepare for capture  function takeScreenshot() {    const clonedElement = document.body.cloneNode(true) as HTMLElement;    const blob = new Blob([clonedElement.outerHTML], { type: "text/html" });    return blob;  }  // Function to capture screenshot by sending cloned HTML to API  async function generateCapture() {    setProcessing(true);    const htmlBlob = takeScreenshot();    if (!htmlBlob) {      setProcessing(false);      return;    }    try {      const formData = new FormData();      formData.append("file", htmlBlob);      formData.append("width", width);      formData.append("height", height);      formData.append("delay", delay);      const response = await fetch("/api/generate-png", {        method: "POST",        body: formData,      });      if (!response.ok) throw new Error("Capture failed");      const blob = await response.blob();      const downloadUrl = URL.createObjectURL(blob);      const link = document.createElement("a");      link.href = downloadUrl;      link.download = "capture.png";      link.click();      URL.revokeObjectURL(downloadUrl);    } catch (error) {      console.error("Failed to capture screenshot", error);    } finally {      setProcessing(false);    }  }  return (    

Webpage Screenshot Capture

{ e.preventDefault(); generateCapture(); }} style={{ display: "flex", flexDirection: "column", alignItems: "center", marginBottom: "16px", }} > setWidth(e.target.value)} style={{ width: "100%", padding: "8px", marginBottom: "16px", borderRadius: "4px", border: "1px solid #ccc", outline: "none", }} > 1920 (Full HD) 1366 (Laptop) 1280 (Desktop) 1024 (Tablet Landscape) 768 (Tablet Portrait) 375 (Mobile) setHeight(e.target.value)} required style={{ width: "100%", padding: "8px", marginBottom: "16px", borderRadius: "4px", border: "1px solid #ccc", outline: "none", }} /> setDelay(e.target.value)} required style={{ width: "100%", padding: "8px", marginBottom: "16px", borderRadius: "4px", border: "1px solid #ccc", outline: "none", }} /> {/* Example HTML Element to Capture */}

Content to Capture

This is an example of the HTML content that will be captured.

);}

结论

本教程涵盖在 next.js 中设置网页捕获工具、使用 puppeteer 处理屏幕截图以及创建交互式前端组件。请记住在本地使用 puppeteer 并在生产中切换到 puppeteer-core,以减少捆绑包大小并优化无服务器环境。快乐编码!

以上就是如何使用 Nextjs 和 Puppeteer 捕获网页屏幕截图的详细内容,更多请关注创想鸟其它相关文章!

版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。
如发现本站有涉嫌抄袭侵权/违法违规的内容, 请发送邮件至 chuangxiangniao@163.com 举报,一经查实,本站将立刻删除。
发布者:程序猿,转转请注明出处:https://www.chuangxiangniao.com/p/1495426.html

(0)
打赏 微信扫一扫 微信扫一扫 支付宝扫一扫 支付宝扫一扫
上一篇 2025年12月19日 17:25:38
下一篇 2025年12月19日 17:25:47

相关推荐

  • 如何使用 Jquery Mobiscroll 实现移动端日期滑动切换?

    滑动切换日期 您希望移动端日期滑动切换实现方式。具体描述是,一行上显示三个日期,每个日期对应一个星期。点击左边的日期时,左边的日期移动到中间,原中间的日期移动到右边,最右边的日期隐藏;点击右边的日期时,右边的日期移动到中间,原来的中间日期移动到左边,最左边的日期隐藏。 解决方案 推荐使用 jquer…

    2025年12月19日
    000
  • 如何局部更改页面图片,无需刷新整个页面?

    如何局部更改页面图片 本问题涉及如何修改页面上一个特定区域的图像,而无需重新加载整个页面。具体场景如下: 您有一个包含下拉选择器的 ,其中列出了不同选项。在另一个 中,有一个图像。图像应根据下拉列表当前选定的值更改。 解决方法: 使用 javascript 此问题可以用 javascript 轻松解…

    2025年12月19日
    000
  • Angular – linkedSignal 简介

    angular 19 即将面世,它带来了许多令人兴奋的新功能。最值得注意的新增内容之一是 linkedsignal 原语,它有望彻底改变我们在 angular 应用程序中处理反应式编程的方式。 重置模式的问题 传统上,在 angular 中实现重置模式涉及使用计算()信号。这种方法虽然有效,但也有局…

    2025年12月19日
    000
  • 如何使用 Cordova 插件判断手机是否安装了微信或 QQ?

    借助插件判断手机是否安装应用 您希望通过 javascript 判断手机上是否安装了微信或 qq,以隐藏相应的快捷登录按钮。以下是借助插件实现这一目标的方法: 在您的 cordova 项目中,可以集成该插件: cordova plugin add cordova-plugin-appinstalle…

    2025年12月19日
    000
  • Nextjs 中的分布式跟踪

    随着现代应用程序变得越来越分布式,特别是随着微服务和无服务器架构的兴起,监控和调试这些系统变得更加复杂。分布式跟踪可帮助开发人员在请求通过各种服务时跟踪请求,从而清楚地了解性能瓶颈、错误和延迟问题。使用 next.js(一个强大的 react 框架)时,实现分布式跟踪可以提高应用程序的可观察性并实现…

    2025年12月19日
    000
  • 如何使用 Nextjs 创建玩家标签生成器应用

    创建玩家标签生成器应用程序可以是一个有趣的实践项目,可让您在构建游戏玩家可能实际使用的东西的同时发挥您的 next.js 技能。 玩家标签生成器制作起来非常简单,并提供了一种使用组件、表单和一些简单随机化的好方法。 到此结束,您将拥有一个可以运行的应用程序,它可以根据一些输入首选项为游戏玩家生成很酷…

    2025年12月19日
    000
  • 移动端日期选择如何实现左右滑动切换效果?

    jquery 日期左右滑动切换 移动端常见日期切换的需求,可以方便用户快速选择日期。如何实现这种左右滑动切换日期的效果呢? 解决方案 有一个很不错的 jquery 插件,推荐使用: jquery mobiscroll 这个插件可以实现丰富的日期选择功能,包括左右滑动切换日期。以下是使用步骤: 引入 …

    2025年12月19日
    000
  • jQuery实现左右滑动切换日期:如何使用jQuery实现一行三个日期和星期的顺次滑动切换?

    jquery实现左右滑动切换日期 如何通过jquery实现左右滑动切换日期的交互效果,让一行上的三个日期和星期随着用户滑动而顺次切换位置? 答案 可以使用jquery的animate()方法实现这一效果。 首先,为每个日期和星期元素添加一个id或类名,以便使用jquery对其进行操作。然后,编写以下…

    2025年12月19日
    000
  • 如何使用 jsPDF 在 React 中从 JSON 数据创建 PDF

    本文将展示如何在 js/react 中从 json 数据创建 pdf。 作为开发人员,我们必须将 pdf 生成集成到应用程序中。因此,在本文中,我们将讨论使用 jspdf 创建 pdf。 那么,让我们开始吧。 我们将专门为本文使用 react 环境。我假设你熟悉 javascript/react 并…

    2025年12月19日
    000
  • 针对开源软件项目的免费人工智能代码审查

    如果您参与过开源软件,您就会知道代码审查的重要性。它们不仅仅是捕捉错误,还确保代码质量、安全性和可维护性,帮助每个贡献者无缝协作。但让我们面对现实吧,代码审查非常耗时。手动审查每个拉取请求 (PR) 可能会减慢开发速度,尤其是在资源有限的开源项目中。 进入 Bito 的 AI 代码审查代理——一种自…

    2025年12月19日
    000
  • 如何成功实施战略

    如何成功实施战略:战略金字塔和 TeamStation AI 的方法 您是否知道 90% 的公司难以有效实施其战略?如果您想知道其他 10% 的人如何取得成功,可以考虑一个简单而强大的工具:策略金字塔。 战略金字塔由 Wendy McGuiness 于 2011 年开发,提供了一种结构化方法来定义和…

    2025年12月19日
    000
  • Forget about axios, this new tool makes automatic data fetching a breeze!

    震惊?这个表单提交策略比react-query还要强! 嘿,我的前端开发者们!今天要跟大家分享一个超级有用的东西——表单提交策略。说实话,你们一定对表单提交的各个细节很头疼吧?别担心,我最近发现了一个对我帮助很大的工具,它使表单提交变得简单而高效。今天就跟大家分享一下这个神奇的助手! alovajs…

    2025年12月19日
    000
  • 如何通过集群提高 Nodejs 应用程序的性能

    我们开发者面临的最大困境之一是如何为最终用户提供最佳性能。我们希望我们的系统灵活、高效且不易出现故障。但是,在开发过程中,我们并不总是意识到一些可以在这方面产生重大影响的策略。 想象一下以下场景:您有一个托管在云提供商(例如 AWS)上的应用程序,使用 EC2 实例在 Node.js 中运行 API…

    2025年12月19日
    000
  • Array – JavaScript Challenges

    您可以在 repo github 上找到这篇文章中的所有代码。 阵列相关的挑战 数组 /** * @return {array} */function arrayof(arr) { return [].slice.call(arguments);}// usage exampleconst arra…

    2025年12月19日
    000
  • 如何实现用户登录过期后的自动重新登录和权限控制?

    有关登录过期、重新登录及权限控制的设计 您提出了一个有关用户会话管理和权限控制的需求。以下是如何设计此解决方案: 验证登录凭据:用户登录时,服务器返回一个包含过期时间的令牌 (token)。该令牌在服务器端存储。客户端令牌验证:在用户浏览网站时,客户端(浏览器)会定期向服务器发送令牌进行验证。令牌过…

    2025年12月19日
    000
  • 反应疲劳:为什么一些开发人员正在继续前进

    别误会我的意思——我喜欢 React。我从 2021 年开始学习它,当时 Hooks 风靡一时,React 正在兴起。与我在大学必须学习的 Java 样板相比,JavaScript 和 React 让人耳目一新。我深入研究了 Scrimba 的 React 前端路径,其中构建迷你项目既有趣又信息丰富…

    2025年12月19日
    000
  • 如何实现用户登录过期自动跳转、重新登录和权限控制?

    如何处理登录过期、重新登录和权限控制 在开发项目时,我们经常会遇到用户登录后需要持续验证其身份并控制其权限的情况。当用户登录时,后台一般会返回一个 token,其中包含用户的过期时间。对于本问题提出的需求,我们需要设计一个机制,当用户的 token 过期时,无论他们正在操作页面的哪个部分,都能跳转到…

    2025年12月19日
    000
  • 如何关闭 Bootstrap 左侧导航栏并使右侧内容全屏显示?

    关闭 bootstrap 左侧导航栏 问题:如何使用 bootstrap 框架隐藏左侧导航栏,并使右侧内容全屏显示? 回答: bootstrap 中没有现成的功能可以关闭左侧导航栏。不过,您可以使用自定义事件手动实现此效果。 手动方法 在导航栏中添加一个关闭按钮。 为该按钮添加一个单击事件处理程序,…

    2025年12月19日
    000
  • Echarts柱状图显示后台数据,x轴坐标混乱如何解决?

    echarts从后台获取数据显示在柱状图上,数据混乱 在使用echarts的柱状图显示从后台获取的数据时,如果x轴的坐标显示混乱,可能是转换方式有误。 修改以下代码: foreach (pub_paramdetail i in _fys){ list.add(i.pdetail_name);} 将l…

    2025年12月19日
    000
  • 掌握 JavaScript 中的循环:综合指南

    循环是编程的基础:使我们能够用最少的代码执行重复性任务。无论您是刚刚入门的初学者,还是希望精炼知识的经验丰富的开发人员,理解循环都将大大增强您编写高效、干净且有趣的代码的能力。 在本指南中,我们将深入研究不同类型的循环、它们在流行编程语言中的语法,以及有关何时以及如何有效使用它们的一些提示。 什么是…

    2025年12月19日
    000

发表回复

登录后才能评论
关注微信