详解使用Next.js构建服务端渲染应用(5)
运行npm run dev
后项目和之前一样可以运行,接下来我们需要添加路由将被伪装过的url和真实的url匹配起来,在server.js中添加:
...... const server = express(); server.get('/p/:id', (req, res) => { const actualPage = '/post'; const queryParams = { title: req.params.id }; app.render(req, res, actualPage, queryParams); }); ......
这样我们就把被伪装过的url和真实的url映射起来,并且query参数也进行了映射。重启项目之后就可以刷新详情页而不会报错了。但是有一个小问题,前端路由打开的页面和后端路由打开的页面title不一样,这是因为后端路由传过去的是id,而前端路由页面显示的是title。这个问题在实际项目中可以避免,因为在实际项目中我们一般会通过id获取到title,然后再展示。作为Demo我们偷个小懒,直接将id作为后端路由页面的title。
之前我们的展示数据都是静态的,接下来我们实现从远程服务获取数据并展示。
远程数据获取
next.js提供了一个标准的获取远程数据的接口:getInitialProps
,通过getInitialProps
我们可以获取到远程数据并赋值给页面的props。getInitialProps
即可以用在服务端也可以用在前端。接下来我们写个小Demo展示它的用法。我们打算从TVMaze API 获取到一些电视节目的信息并展示到我的网站上。首先,我们安装isomorphic-unfetch,它是基于fetch实现的一个网络请求库:
npm install --save isomorphic-unfetch
然后我们修改index.js如下:
import Link from 'next/link'; import Layout from '../components/Layout'; import fetch from 'isomorphic-unfetch'; const Index = (props) => ( <Layout> <h1>Marvel TV Shows</h1> <ul> {props.shows.map(({ show }) => { return ( <li key={show.id}> <Link as={`/p/${show.id}`} href={`/post?id=${show.id}`}> <a>{show.name}</a> </Link> </li> ); })} </ul> </Layout> ); Index.getInitialProps = async function () { const res = await fetch('https://api.tvmaze.com/search/shows?q=marvel'); const data = await res.json(); return { shows: data } } export default Index;
以上代码的逻辑应该很清晰了,我们在getInitialProps
中获取到电视节目的数据并返回,这样在Index的props就可以获取到节目数据,再遍历渲染成节目列表。
运行项目之后,页面完美展示:
接下来我们来实现详情页,首先我们把/p/:id
的路由修改为: