我需要在第三方 html 标记内构建我的 Next.js 应用程序。
标记如下:
标题.txt
<html>
<head>
<title>Some title</title>
</head>
<body>
<header>Some header</header>
<div>
页脚.txt
</div>
<footer>Some footer</footer>
</body>
</html>
此文件动态生成到文件夹中。当我渲染 next.js 应用程序时,我需要将它们包装在我的应用程序周围。
我使用名为:html-react-parser 的包创建了一个工作示例
我从 _document.js 中的文件中解析标记,并且正在寻找一个自定义元素 id,我将其替换为下一个 js 应用程序,如下所示:
const header = fs.readFileSync(path.resolve(ROOT + '/resources', 'header.txt'), 'utf8');
const footer = fs.readFileSync(path.resolve(ROOT + '/resources', 'footer.txt'), 'utf8');
const shell = `
${header}
<main id="react-app"></main>
${footer}
`;
// later in the render method od _document.js:
render() {
return (
<React.Fragment>
{parse(shell, {
replace: domNode => {
if (domNode.attribs && domNode.attribs.id === 'react-app') {
return (
<React.Fragment>
<Main/>
<NextScript/>
</React.Fragment>
)
}
}
})}
</React.Fragment>
)
}
也许解决方案是使用angerouslySetInnerHTML,但在这种情况下我无法注入(inject) and 。
// it fails because it wont treat the components as normal html
// <Main/><NextScript/> are not evaluated
const header = fs.readFileSync(path.resolve(ROOT + '/resources', 'header.txt'), 'utf8');
const footer = fs.readFileSync(path.resolve(ROOT + '/resources', 'footer.txt'), 'utf8');
const shell = `
${header}
<React.Fragment>
<Main/>
<NextScript/>
</React.Fragment>
${footer}
`;
// later in the render method od _document.js:
render(<html dangerouslySetInnerHTML={{__html: shell}}/>)
如果有人知道我如何设法将我的下一个应用程序包装在来自文件的标记周围,请给我一些建议。
最佳答案
您可以通过custom server来实现这一点,
const express = require('express')
const next = require('next')
const header = fs.readFileSync(path.resolve(ROOT + '/resources', 'header.txt'), 'utf8');
const footer = fs.readFileSync(path.resolve(ROOT + '/resources', 'footer.txt'), 'utf8');
const port = parseInt(process.env.PORT, 10) || 3000
const dev = process.env.NODE_ENV !== 'production'
const app = next({ dev })
const handle = app.getRequestHandler()
app.prepare().then(() => {
const server = express()
server.all('*', async (req, res) => {
const nextResponse = await handle(req, res);
return mergeHTML(header, footer, nextResponse); // You will need to "merge" cause nextResponse has `head` & `body` as well.
})
server.listen(port, err => {
if (err) throw err
console.log(`> Ready on http://localhost:${port}`)
})
})
mergeHTML
应该合并 header
中的 head 以及下一个的 head 和正文。您可以使用cheerio
来帮助你。
关于javascript - Next.js 在 html 标记中渲染应用程序作为字符串,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/60357068/