我无法使用 Gatsby 动态创建页面
我在我的项目中使用 Gatsby。我正在尝试创建一个需要分页的页面。
我按照这个 https://www.gatsbyjs.org/docs/adding-pagination/ 和这个其他指南 https://nickymeuleman.netlify.app/blog/gatsby-pagination/ 以及如何做到这一点,但它不起作用。
我对
gatsby-node.js
有多个查询:
exports.createPages = async ({ graphql, actions, page }) => {
const { createPage } = actions
const shop = path.resolve("./src/templates/shop.js")
const productPageTemplate = path.resolve("./src/templates/ProductPage/index.js")
const singleProduct = await graphql(`
query {
allShopifyProduct {
edges {
node {
handle
}
}
}
}
`)
// This is the query which I need for the pagination
// Create shop pages
const products = await graphql(`
query allShopifyProduct($skip: Int!, $limit: Int!) {
allShopifyProduct(sort: { fields: [createdAt], order: DESC }
skip: $skip
limit: 5
) {
edges {
node {
id
title
handle
}
}
}
}
`)
const posts = products.data.allShopifyProduct.edges
const postsPerPage = 5
const numPages = Math.ceil(posts.length / postsPerPage)
Array.from({ length: numPages }).forEach((_, i) => {
const withPrefix = (pageNumber) =>
pageNumber === 1 ? `/shop` : `/shop/${pageNumber}`
const pageNumber = i + 1
createPage({
path: withPrefix(pageNumber),
component: shop,
context: {
limit: postsPerPage,
skip: i * postsPerPage,
current: pageNumber,
total: numPages,
hasNext: pageNumber < numPages,
nextPath: withPrefix(pageNumber + 1),
hasPrev: i > 0,
prevPath: withPrefix(pageNumber - 1),
},
})
})
// Adding the single product stuff to show my multiple queries
singleProduct.data.allShopifyProduct.edges.forEach(({ node }) => {
createPage({
path: `/product/${node.handle}/`,
component: productPageTemplate,
context: {
// Data passed to context is available
// in page queries as GraphQL variables.
handle: node.handle,
},
})
})
})
这是我的React组件:
import React from "react"
import { graphql, Link } from "gatsby"
// Components
import ProductGrid from "@components/ProductGrid/productGrid"
const Shop = ({ data: { allShopifyProduct }, pageContext }) => {
return (
<div className="product-grid">
<ProductGrid allShopifyProduct={allShopifyProduct}
/>
{!pageContext.hasPrev && (
<Link to={pageContext.prevPage} rel="prev">
← Previous Page
</Link>
)}
{!pageContext.hasNext && (
<Link to={pageContext.nextPage} rel="next">
Next Page →
</Link>
)}
</div>
)
}
export default Shop
export const query = graphql`
query allShopifyProduct($skip: Int!, $limit: Int!) {
allShopifyProduct(
sort: { fields: [createdAt], order: DESC }
skip: $skip
limit: $limit
) {
edges {
node {
id
title
handle
}
}
}
}
`
它不断抛出错误,例如这些:
ERROR #85927 GRAPHQL
There was an error in your GraphQL query:
Variable "$skip" is never used in operation "allShopifyProduct".
See if $skip has a typo or allShopifyProduct doesn't actually require this variable.
File: gatsby-node.js:74:26
ERROR #85927 GRAPHQL
There was an error in your GraphQL query:
Variable "$limit" is never used in operation "allShopifyProduct".
See if $limit has a typo or allShopifyProduct doesn't actually require this variable.
File: gatsby-node.js:74:26
ERROR #11321 PLUGIN
"gatsby-node.js" threw an error while running the createPages lifecycle:
Cannot read property 'allShopifyProduct' of undefined
108 | `)
109 |
> 110 | const posts = products.data.allShopifyProduct.edges
| ^
111 |
File: gatsby-node.js:110:31
TypeError: Cannot read property 'allShopifyProduct' of undefined
- gatsby-node.js:110 Object.exports.createPages
/Users/marcelo/Work/gatsby-on-demand/gatsby-node.js:110:31
failed createPages - 0.113s
如果我在 GraphiQL 界面上运行这些精确的查询,一切都会完美运行:
有什么想法我可能在哪里失败了?
您必须在
createPage
API 内的上下文中提供这些变量:
createPage({
path: `/product/${node.handle}/`,
component: productPageTemplate,
context: {
skip: 0 // or any variable
limit: 5 // or any variable
handle: node.handle, // is it used? If don't, you can remove it
},
})
由于您在查询中使用了不可空的
skip
和
limit
变量(标有感叹号
!
):
query allShopifyProduct($skip: Int!, $limit: Int!)
它们需要存在,因此您需要以与在 GraphQL 查询操场中显示的方式相同的方式提供它们(通过上下文)。
您的
handle
变量似乎未被使用,至少在提供的代码中是如此,在这种情况下,您可以将其删除。
有关 GraphQL 架构和类型 的更多信息。
一开始...传递查询中使用的参数/值的方法(作为变量)...您应该将它们(值)传递给“循环”查询中的变量:
let myLoopingSkip = 0; // starting skip
// Create shop pages
const products = await graphql(`
query allShopifyProduct($skip: Int!, $limit: Int!) {
allShopifyProduct(sort: { fields: [createdAt], order: DESC }
skip: $skip
limit: $limit
) {
edges {
node {
id
title
handle
}
}
}
}
`, { skip: myLoopingSkip, limit: 5 } ); // variables used in query
然后,您可以构建一个“外部”循环,遍历整个(查询+分页列表和单个产品的 createPages)块,将当前
myLoopingSkip
值传递给查询变量
skip
。
两种可能的循环场景 :
- 一次查询所有产品(所有数据);
- 仅查询当前迭代所需的数据段(5 项记录集/块)。
第一个选项很简单,但在大数据集上可能会耗费资源 ,它可能会在某个时刻崩溃。
第二个(IMHO 更好)选项更可靠,但需要额外/单独的查询循环前
numPages
(所有产品的数量)。条件
next page
也需要它。
在现实生活中,浏览 10 页后的情况并不常见……实际上,即使谷歌在达到某个页面限制后也会停止显示结果……但产品页面必须链接到某个地方才能访问/抓取 - 在我看来,最好按类别列出较短的分页列表。
如果您真的不需要
numPages
,您可以简单地通过添加 5(您的“限制”/
postsPerPage
)来继续循环,只要查询的数据(结果)包含 5 条记录即可。在这种情况下,“外循环”可能看起来像:
const postsPerPage = 5;
let myLoopingSkip = 0; // starting skip
do {
// read only data you need in current iteration
let products = await graphql(PAGINATED_PRODUCTS_QUERY,
{ skip: myLoopingSkip, limit: postsPerPage } );
// paginated list
createPage({
path: withPrefix(pageNumber),
component: shop,
context: {
limit: postsPerPage,
skip: i * postsPerPage,
...
// but you can just pass fetched data
// as is done for product page
// no need for query inside component
// just loop over `data` prop to create grid view
//
// createPage({
// path: withPrefix(pageNumber),
// component: shop,
// context: {
// data: products.data.allShopifyProduct.edges,
// loop for single products pages
products.data.allShopifyProduct.edges.map( (node) => {
createPage({
path: `/product/${node.handle}/`,
component: productPageTemplate,
context: {
// Data passed to context is available
// in page queries as GraphQL variables.
handle: node.handle,
// ... but we already have all data here
// ... again, no query needed
// data: node
}
});
myLoopingSkip += postsPerPage;
} while( products.data.allShopifyProduct.edges.length===postsPerPage )
// or use total page condition