GraphQL查询可在盖茨比页面中使用,但不能在类组件中使用

时间:2019-08-22 14:57:22

标签: reactjs graphql gatsby prismic.io

有几个类似的问题,但是没有一个问题让我真正理解在页面文件夹中使用的GraphQL在(类)组件内部。

我的项目结构如下:

-src
--components
---aboutBody
----index.js
--pages
---about.js

我有一个名为about(原始单一页面类型)的页面组件,并设置了一些组件来“填充”此页面(为了更好的可读性而进行了清理)。

class AboutPage extends Component {

  render() {
    return (
        <LayoutDefault>
          <AboutBody
            introHeadline={this.props.data.prismicAbout.data.intro_headline.text}
            introParagraph={this.props.data.prismicAbout.data.intro_paragraph.text}
          />
        </LayoutDefault>
    )
  }

}

export default AboutPage

这是我的查询的样子(两个文件都这样):

export const aboutQuery = graphql`
  query About {
    prismicAbout {
      data {

        # Intro Block
        intro_headline {
          text
        }
        intro_paragraph {
          text
        }
      }
    }
  }
`

(如果我在底部缺少括号,这是由于清理了SO的查询示例-如前所述,它在我的页面组件中起作用)。

我的graphql查询位于AboutPage页面组件的底部。它像魅力一样并按预期工作。

但是要稍微整理一下页面,我想创建适当的组件并将查询放入每个组件(例如aboutBodyaboutCarousel)中,再次清理一下:

class AboutBody extends Component {

  render() {

    return (
      <StyledIntro>
        <h3>About</h3>
        <h1>{this.props.data.prismicAbout.data.intro_headline.text}</h1>
      </StyledIntro>
    )
  }

}

export default AboutBody

然后我从about页面组件中删除了查询,并将其放入了AboutBody组件中(完全如上所示)。

但是,它总是返回错误Cannot read property 'prismicAbout' of undefined(我什至无法控制台记录数据,它总是返回相同的错误)。

我在两个文件中都使用了import { graphql } from "gatsby"

长话短说,如何在我的类组件中放置一个查询并仅呈现该组件而又不弄清楚页面组件中的道具,就像这样:

class AboutPage extends Component {

  render() {
    return (
        <LayoutDefault>
          <AboutBody />
        </LayoutDefault>
    )
  }

}

一些博客文章提到了GraphQL查询片段,但是不确定这是否是正确的用例,或者仅仅是一个愚蠢的初学者错误...

3 个答案:

答案 0 :(得分:1)

那是因为您不能在组件中使用像这样的graphql。

要在组件中使用graphql,您有两个选择:useStaticQuery函数或StaticQuery组件,均来自graphql

useStaticQuery

import React from "react"
import { useStaticQuery, graphql } from "gatsby"

const MyElement = () => {
  const data = useStaticQuery(graphql`
     query About {
       prismicAbout {
         data {
           intro_headline {
             text
           }
           intro_paragraph {
             text
           }
         }
       }
     }
   `)

   return (
      <StyledIntro>
        <h3>About</h3>
        <h1>{this.props.data.prismicAbout.data.intro_headline.text}</h1>
      </StyledIntro>
   )
}

export default MyElement

staticQuery

import React from 'react'
import { StaticQuery, graphql } from 'gatsby';

const MyElement = () => {
   return(
      <StaticQuery
            query About {
              prismicAbout {
                data {
                   intro_headline {
                       text
                       }
                       intro_paragraph {
                          text
                       }
                   }
                }
             }
         `}
         render={data => (
            <StyledIntro>
               <h3>About</h3>
               <h1>{this.props.data.prismicAbout.data.intro_headline.text}</h1>
            </StyledIntro>
         )}
      />
   )
}

export default MyElement

希望对您有帮助!

答案 1 :(得分:1)

您只能在页面组件中使用类似的查询。一种选择是只在页面中查询它,然后将数据作为道具传递到您的组件中。另一个是在组件中使用静态查询。

如果查询中包含变量,则不能使用静态查询。在这种情况下,您应该在页面中全部查询然后传递给它,或者您可以将与该组件相关的查询部分放在该组件文件中的某个片段中,然后在页面查询中使用该片段。

在组件中使用片段然后将数据传递到组件中的示例:

// MyComponent.js
import React from "react"
import { graphql } from 'gatsby'

const MyComponent = (props) => {

  const { myProp: { someData } }  = props

  return (
    <div>
        my awesome component
    </div>
  )
}

export default MyComponent

export const query = graphql`
  fragment MyAwesomeFragment on Site {
     someData {
         item
     }
  }
`
// MyPage.js
import React from "react"
import { graphql } from "gatsby"

import MyComponent from "../components/MyComponent"

export default ({ data }) => {
  return (
    <div>
      {/*
        You can pass all the data from the fragment
        back to the component that defined it
      */}
      <MyComponent myProp={data.site.someData} />
    </div>
  )
}
export const query = graphql`
  query {
    site {
      ...MyAwesomeFragment
    }
  }
`

详细了解如何在Gatsby docs.中使用片段

答案 2 :(得分:0)

如果您需要在基于类的组件中呈现查询。这对我有用:

x.value