graphql - 将 ImageSharp 作为字段添加到 MarkdownRemark 节点(不是 frontmatter)
问题描述
我正在尝试使用以下 graphQL 查询:
{
allMarkdownRemark(
limit: 1000
) {
edges {
node {
id
parent {
id
}
fields{
slug
hero {
childImageSharp {
fixed {
src
}
}
}
}
frontmatter {
template
}
}
}
}
}
该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),
})
}
}
我已经看到其他人在图像路径中做了类似的事情,frontmatter
但我不想使用frontmatter,因为它很容易让graphql查看文件路径而无需指定它。
但是,当我尝试上述方法时,出现以下错误:
字段 \"hero\" 不能有选择,因为类型 \"String\" 没有子字段。
有没有办法让我childImageSharp
认识这个领域?
解决方案
我再次回来(希望)一劳永逸地解决这个问题(请参阅我们的历史)。
这一次,我们将把英雄图像附加ImageSharp
到MarkdownRemark
节点上。你的方法是正确的,有一个警告:盖茨比似乎只识别相对路径,即以点开头的路径。
您可以在代码中轻松解决此问题:
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),
})
这应该可行,尽管我想提供一个替代的英雄搜索功能。我们可以得到一个文件列表dir
,fs.readdir
然后找到一个名为“英雄”的文件:
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}`,
})
}
}
这样我们就不用关心英雄图像的扩展名是什么,只要它存在。我使用String.prototype.includes
,但为了安全起见,您可能希望使用正则表达式传入允许的扩展名列表,例如/hero.(png|jpg|gif|svg)/
. (我认为您的解决方案更具可读性,但我更喜欢每个节点只访问一次文件系统。)
您还可以使用path.relative
来查找默认英雄图像的相对路径。
现在,这个 graphql 查询有效:
一个(小)问题
但是,这种方法有一个小问题:它破坏了 graphql 过滤器类型!当我尝试基于 查询和过滤时hero
,我收到此错误:
也许盖茨比忘了重新推断 的类型hero
,所以它不是 a File
,而是 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,
})
}
}
如果您不需要按英雄图像过滤内容,那么让 gatsby 处理节点类型更为可取。
如果您在尝试此操作时遇到问题,请告诉我。
推荐阅读
- php - 流明验证:数据库中不存在值的规则
- php - 如何正确提取 PHP 字符串中分隔的标点字符和符号?
- testing - 有没有办法激活依赖扩展来测试完整的扩展激活?
- python - 访问通过 Xlwt 库 (Python) 生成的 .dtsx 中的 Excel 文件会引发错误 CANNOTACQUIRECONNECTIONFROMCONNECTIONMANAGER
- javascript - 请求错误:使用 vue cli devserver 代理的 Uri 无效
- java - 如何为多个 BitTorrent 连接创建套接字线程
- r - R 中使用 bplpapi 的 Bloomberg 数据提取问题,数据未填充且返回的开始日期不正确
- django - 我无法理解为什么会出现此错误。什么是解决方案
- java - Firestore 凭据问题:Firebase 应用 [默认] 已存在
- c++ - 如何实现内联显式类的构造函数?