盖茨比创建页面

时间:2020-10-07 00:31:51

标签: javascript node.js graphql gatsby express-graphql

我正在尝试创建一个gatsby网站,在该网站上我发布了每个项目都带有标签的项目。我遵循了有关创建gatsby博客的基本教程,现在正在研究gatsby网站上记录的标签。在将变量名从path更改为slug以避免保留字的过程中,由于无法再创建Blogpost页面或标记页面,我似乎一团糟。每个帖子都包含在以帖子标题命名的文件夹中,该文件夹中包含一个index.md文件,该文件定义了其内容,日期,标签等。

当我尝试运行gatsby development时,输出以下错误

错误#11323

Your site's "gatsby-node.js" must set the page path when creating a page.

The page object passed to createPage:
{
    "slug": "/an-example-post",
    "component": "C:\\Users\\Monolith\\Documents\\programming\\webdev\\Gatsby-Portfolio\\src\\templates\\post.js",
    "context": {
        "slug": "/an-example-post"
    }
}

我的gatsby-node.js

const path = require('path');

exports.createPages = ({actions, graphql}) => {
     const {createPage} = actions;

     const postTemplate = path.resolve('src/templates/post.js');
     const tagTemplate = path.resolve("src/templates/tags.js")
     //const postTemplate = require.resolve('src/templates/post.js');
     //const tagTemplate = require.resolve("src/templates/tags.js")
     
     return graphql(`{
        allMarkdownRemark{
            edges{
                node {
                    html
                    id
                    frontmatter{
                        slug
                        title
                    }
                }
            }
        }
    
        tagsGroup: allMarkdownRemark(limit: 2000) {
            group(field: frontmatter___tags) {
              fieldValue
            }
          }

    }`)
    .then(res => {
        if(res.errors){
            return Promise.reject(res.errors);
        }


        
        res.data.allMarkdownRemark.edges.forEach( ({node}) => {
            createPage({
                slug: node.frontmatter.slug,
                component: postTemplate,
                context: {
                    slug:node.frontmatter.slug
                },
            })
        })

          // Extract tag data from query
        const tags = res.data.tagsGroup.group

        // Make tag pages
        tags.forEach(tag => {
            createPage({
                // path: `/tags/${_.kebabCase(tag.fieldValue)}/`,
                slug: `/tags/${(tag.fieldValue)}/`,
                component: tagTemplate,
                context: {
                    tag: tag.fieldValue,
                },
            })
        })

    }) 
}

如前所述,我担心将路径变量用作保留字可能是一个问题,但是我比忽略它更进一步地使用了它,所以现在它一直存在。

示例post.js

import React from 'react';
import { graphql } from 'gatsby';
import Layout from "../components/layout"

export default function Template({data}) {
    const {markdownRemark: post} = data;

    return(
        <Layout>
        <div>
            <h1 className="postTitle">{post.frontmatter.title}</h1>
            <div className="tagGroup">
                {post.frontmatter.tags.map((tag, index) => (
                    <div key={index}>
                        <h2 className = "tagStyle">{tag}</h2>
                    </div>
                ))}
            </div>
            <p>{post.frontmatter.description}</p>
            <p>{post.frontmatter.date}</p>            
            {/* <h2 className="tagStyle">{post.frontmatter.tags + " "}</h2> */}
            <div dangerouslySetInnerHTML={{__html: post.html}} />
        </div>
        </Layout>
    )
}


export const postQuery = graphql`
    #query BlogPostByPath($slug: String!) {
    query($slug: String!) {
        markdownRemark(frontmatter: { slug: {eq: $slug} }) {
            html
            frontmatter {
                slug
                title
                description
                date
                tags
            }
        }
    }
`

tags.js与gatsby的默认值非常相似,但内容略有改动。这是我正在使用的Graphql查询。

Tags.propTypes = {
  pageContext: PropTypes.shape({
    tag: PropTypes.string.isRequired,
  }),
  data: PropTypes.shape({
    allMarkdownRemark: PropTypes.shape({
      totalCount: PropTypes.number.isRequired,
      edges: PropTypes.arrayOf(
        PropTypes.shape({
          node: PropTypes.shape({
            frontmatter: PropTypes.shape({
              title: PropTypes.string.isRequired,
           //   slug: PropTypes.string.isRequired,
            }),
            fields: PropTypes.shape({
              slug: PropTypes.string.isRequired,
            }),
          }),
        }).isRequired
      ),
    }),
  }),
}

export default Tags

export const pageQuery = graphql`
  query($tag: String) {
    allMarkdownRemark(
      limit: 2000
      sort: { fields: [frontmatter___date], order: DESC }
      filter: { frontmatter: { tags: { in: [$tag] } } }
    ) {
      totalCount
      edges {
        node {
           fields {
             slug
           }
          frontmatter {
            title
          }
        }
      }
    }
  }
`

如果任何人有任何信息可以帮助我指出正确的方向,我将不胜感激。我已经花了几个小时浏览了盖茨比的文档,但尚未取得实质性进展。

1 个答案:

答案 0 :(得分:1)

您正在根据自己的情况传递tag,但期望slug

    createPage({
        slug: `/tags/${(tag.fieldValue)}/`,
        component: tagTemplate,
        context: {
            tag: tag.fieldValue,
        },
    })

并且:

   query($slug: String!) {}

上下文是一种将数据传递到组件模板以使用它来过滤数据的方法。理想情况下,应为唯一参数(例如slugid等)。这样,您应该在gatsby-node.js中提取所有帖子,按上下文传递唯一字段,然后在模板中使用该变量来获取每个帖子所需的所有数据。

此外,您必须为函数提供path,而不是slug

您应该将createPage函数更改为:

    createPage({
        path: `/tags/${tag.fieldValue}/`,
        component: tagTemplate,
        context: {
            slug: tag.fieldValue,
        },
    })

请记住,尽管命名不同,fieldValue还是应该是个子。

我建议阅读Gatsby的教程:https://www.gatsbyjs.com/tutorial/part-seven/