gpt4 book ai didi

reactjs - 是否可以将 React 组件传递给 puppeteer?

转载 作者:行者123 更新时间:2023-12-03 13:35:08 25 4
gpt4 key购买 nike

我有一个带有一些 componentDidMount 逻辑的 React 组件:

export default class MyComponent {
componentDidMount() {
// some changes to DOM done here by a library
}

render() {
return (
<div>{props.data}</div>
);
}
}

是否可以通过 props 传递此组件,以便执行 componentDidMount() 中的所有内容,以某种方式进行 puppeteer 以获取屏幕截图?大致如下:

const browser = await puppeteer.launch({ headless: true });
const page = await browser.newPage();

const html = ReactDOMServer.renderToString(<MyComponent data='' />); <-- but this skips the componentDidMount logic
await page.setContent(html);
await page.screenshot({ path: 'screenshot.png' });

我知道我可以使用 page.goto(),但我有一些复杂的登录逻辑,我想使用这样的快捷方式来避免这些逻辑,而是将所有需要的 Prop 直接传递给组件?

最佳答案

我回答了这个问题here 。让我们在这里尝试同样的操作。

安装 babel、webpack 和 puppeteer 软件包。

{
"name": "react-puppeteer",
"version": "1.0.0",
"main": "index.js",
"license": "MIT",
"scripts": {
"compile": "webpack",
"build": "webpack -p",
"start": "webpack && node pup.js"
},
"devDependencies": {
"babel-core": "^6.26.0",
"babel-loader": "^7.1.2",
"babel-preset-env": "^1.6.1",
"babel-preset-es2015": "^6.24.1",
"babel-preset-react": "^6.24.1",
"react": "^16.2.0",
"react-dom": "^16.2.0",
"webpack": "^3.10.0",
"webpack-dev-middleware": "^2.0.3"
},
"dependencies": {
"puppeteer": "^0.13.0"
}
}

准备 webpack 配置,

const webpack = require('webpack');

const loaders = [
{
test: /\.jsx?$/,
exclude: /node_modules/,
loader: 'babel-loader',
query: {
presets: ['babel-preset-es2015', 'babel-preset-react'],
plugins: []
}
}
];

module.exports = {
entry: './entry.js',
output: {
path: __dirname,
filename: 'bundle.js',
libraryTarget: 'umd'
},
module: {
loaders: loaders
}
};

创建入口文件,在这个文件上,不是直接挂载元素,而是将其导出到窗口,以便我们稍后可以访问它。

import React from 'react';
import { render } from 'react-dom';

class Hello extends React.Component {
render() {
return <h1>Hello, {this.props.name}</h1>;
}
}

// accept a name for example and a domNode where to render
function renderIt(name, domNode) {
render(<Hello name={name} />, domNode);
}

window.renderIt = renderIt;

当我们运行webpack时,它将生成一个bundle.js文件。我们可以在 puppeteer 上使用它。

他们已经弃用了 puppeteer 上的 insertFile 函数,因此我们将恢复它。这是一个示例存储库,您可以通过 yarn 添加它。

https://github.com/entrptaher/puppeteer-inject-file

现在,让我们创建一个 puppeteer 脚本。

const puppeteer = require('puppeteer');
const injectFile = require('puppeteer-inject-file');

(async () => {
const browser = await puppeteer.launch({ headless: false });
const page = await browser.newPage();
await page.goto('https://github.com');
await injectFile(page, require.resolve('./bundle.js'));
await page.evaluate(() => {
renderIt("Someone", document.querySelector('div.jumbotron.jumbotron-codelines > div > div > div > h1'));
});
await page.screenshot({ path: 'example.png' });
await browser.close();
})();

当我们运行这个时,我们得到以下结果,

enter image description here

如果我们添加 componentDidMount() 调用,我们也可以做到这一点。但如果我们想做更多的修改,我们就必须让 puppeteer 脚本等待,这在其他问题中已经讨论过很多次了。

假设我们现在有一个状态,一旦组件加载,它就会返回一些东西。

class Hello extends React.Component {
state = {
jokes: null
};

componentDidMount() {
const self = this;
const jokesUrl = `http://api.icndb.com/jokes/random?firstName=John&amp;lastName=Doe`;
fetch(jokesUrl)
.then(data => data.json())
.then(data => {
self.setState({
jokes: data.value.joke
});
});
}

render() {
if(!!this.state.jokes){
return <p id='quote'>{this.state.jokes}</p>
}
return <h1>Hello, {this.props.name}</h1>;
}
}

在 puppeteer 上,我可以像这样等待元素,

  ...
await injectFile(page, require.resolve('./bundle.js'));
await page.evaluate(() => {
renderIt("Someone", document.querySelector('div'));
});
await page.waitFor('p#quote');
...

我们可能需要 babel-preset-stage-2 但我会把它留给你。这是结果,

enter image description here

你自己解决剩下的问题:) ...

关于reactjs - 是否可以将 React 组件传递给 puppeteer?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/48034767/

25 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com