将 ImageSharp 作为字段添加到 MarkdownRemark 节点(不是 frontmatter)

Ale*_*igh 3 graphql gatsby

我正在尝试执行以下 graphQL 查询:

 {
      allMarkdownRemark(
        limit: 1000
      ) {
        edges {
          node {
            id
            parent {
              id
            }
            fields{
              slug
              hero {
                childImageSharp {
                  fixed {
                    src
                  }
                }
              }
            }
            frontmatter {
              template
            }
          }
        }
      }
    }
Run Code Online (Sandbox Code Playgroud)

hero字段当前使用以下代码返回图像的路径:

exports.onCreateNode = ({ node, actions, getNode }) => {
  const { createNodeField } = actions

  // Add slug to MarkdownRemark node
  if (node.internal.type === 'MarkdownRemark') {
    const value = createFilePath({ node, getNode, basePath: 'library' })
    const { dir } = getNode(node.parent)
    const getHero = (d) => {
      let hero = `${__dirname}/src/images/no-hero.gif`
      if (fs.existsSync(`${d}/hero.jpg`)) hero = `${d}/hero.jpg`
      if (fs.existsSync(`${d}/hero.png`)) hero = `${d}/hero.png`
      if (fs.existsSync(`${d}/hero.gif`)) hero = `${d}/hero.gif`
      return hero
    }
    createNodeField({
      node,
      name: 'slug',
      value,
    })

    createNodeField({
      node,
      name: 'hero',
      value: getHero(dir),
    })
  }
}
Run Code Online (Sandbox Code Playgroud)

我见过其他人对 中的图像路径做了类似的事情frontmatter,但我不想使用 frontmatter,因为很容易让 graphql 查看文件路径而无需指定它。

但是,当我尝试上述操作时,出现以下错误:

字段“英雄”不能有选择,因为类型“字符串”没有子字段。

有什么方法可以让我childImageSharp认识这个领域吗?

Der*_*yen 5

我再次回来(希望)一劳永逸地解决这个问题(请在此处查看我们的历史)。

这次,我们将把英雄图像附加ImageSharp到节点上MarkdownRemark。您的方法是正确的,但有 1 个警告:盖茨比似乎只识别相对路径,即以点开头的路径。

您可以在代码中轻松解决此问题:

    const getHero = (d) => {
      let hero = `${__dirname}/src/images/no-hero.gif`

  -   if (fs.existsSync(`${d}/hero.jpg`)) hero = `${d}/hero.jpg`
  -   if (fs.existsSync(`${d}/hero.png`)) hero = `${d}/hero.png`
  -   if (fs.existsSync(`${d}/hero.gif`)) hero = `${d}/hero.gif`

  +   if (fs.existsSync(`${d}/hero.jpg`)) hero = `./hero.jpg`
  +   if (fs.existsSync(`${d}/hero.png`)) hero = `./hero.png`
  +   if (fs.existsSync(`${d}/hero.gif`)) hero = `./hero.gif`

      return hero
    }
    createNodeField({
      node,
      name: 'hero',
      value: getHero(dir),
    })
Run Code Online (Sandbox Code Playgroud)

这应该可行,尽管我想提供替代的英雄搜索功能。dir我们可以获取with中的文件列表fs.readdir,然后找到名为“hero”的文件:

exports.onCreateNode = async ({
  node, actions,
}) => {
  const { createNodeField } = actions

  if (node.internal.type === 'MarkdownRemark') {
    const { dir } = path.parse(node.fileAbsolutePath)

    const heroImage = await new Promise((res, rej) => {

      // get a list of files in `dir`
      fs.readdir(dir, (err, files) => {
        if (err) rej(err)

        // if there's a file named `hero`, return it
        res(files.find(file => file.includes('hero')))
      })
    })

    // path.relative will return a (surprise!) a relative path from arg 1 to arg 2.
    // you can use this to set up your default hero
    const heroPath = heroImage 
      ? `./${heroImage}` 
      : path.relative(dir, 'src/images/default-hero.jpg')

    // create a node with relative path
    createNodeField({
      node,
      name: 'hero',
      value: `./${heroImage}`,
    })
  }
}
Run Code Online (Sandbox Code Playgroud)

这样我们就不在乎英雄图像的扩展名是什么,只要它存在即可。我使用String.prototype.includes,但为了安全起见,您可能想使用正则表达式传递允许的扩展名列表,例如/hero.(png|jpg|gif|svg)/. (我认为您的解决方案更具可读性,但我更喜欢每个节点仅访问文件系统一次。)

您还可以用来path.relative查找默认英雄图像的相对路径。

现在,这个 graphql 查询可以工作:

在此输入图像描述


一个(小)问题

然而,这种方法有一个小问题:它破坏了 graphql 过滤器类型!当我尝试基于 查询和过滤时hero,出现以下错误:

在此输入图像描述

也许盖茨比忘记了重新推断 的类型hero,所以File它仍然是 a ,而不是 a String。如果您需要过滤器工作,这很烦人。

这里有一个解决方法:我们不用要求 Gatsby 链接文件,而是自己做。

exports.onCreateNode = async ({
  node, actions, getNode, getNodesByType,
}) => {
  const { createNodeField } = actions

  // Add slug to MarkdownRemark node
  if (node.internal.type === 'MarkdownRemark') {
    const { dir } = path.parse(node.fileAbsolutePath)
    const heroImage = await new Promise((res, rej) => {
      fs.readdir(dir, (err, files) => {
        if (err) rej(err)
        res(files.find(file => file.includes('hero')))
      })
    })

    // substitute with a default image if there's no hero image
    const heroPath = heroImage ? path.join(dir, heroImage) : path.resolve(__dirname, 'src/images/default-hero.jpg')

    // get all file nodes
    const fileNodes = getNodesByType('File')

    // find the hero image's node
    const heroNode = fileNodes.find(fileNode => fileNode.absolutePath === heroPath)
    createNodeField({
      node,
      name: 'hero___NODE',
      value: heroNode.id,
    })
  }
}
Run Code Online (Sandbox Code Playgroud)

现在我们可以hero再次过滤该字段: 在此输入图像描述

如果您不需要按英雄图像过滤内容,那么让 gatsby 处理节点类型会更好。

如果您在尝试此操作时遇到问题,请告诉我。