DEV Community

liwenzhi
liwenzhi

Posted on

Using image in Gatsby application by a clear way

Gatsby has many features to win the developers' love and prevails over other mainstream static site generators. One of those features is gatsby-image component. It solved the image optimization problem with the specified dimension scope you defined, progressively and responsively presenting to the webpage gives user a comfortable browsing experience.

Steps to use gatsby-image

Like the official doc said, there are three steps needed to implement a gatsby image in Gatsby website:

  • install gatsby-image component and two other build dependency plugins: gatsby-transformer-sharp and gatsby-plugin-sharp
  • define image source directory in gatsby-source-filesystem plugin as well as above two plugin in gatsby-config.js plugins section
  • import gatsby-image component to your gatsby page component and declare a image tag/instance with a fixed or fluid property whose value derived from graphql query.

In the step 2, you tell the Gatsby build tool, where to find the root directory of all the image files. Just like the doc example, the root directory here is src/images/ of current Gatsby project:

{
  resolve: `gatsby-source-filesystem`,
  options: {
    name: `images`,
    path: path.join(__dirname, `src`, `images`),
  },
},
Enter fullscreen mode Exit fullscreen mode

In the step 3, you tell gatsby-image component instance where to get the file: graphql query result.

export const query = graphql`
  query {
    file(relativePath: { eq: "blog/avatars/kyle-mathews.jpeg" }) {
      childImageSharp {
        # Specify the image processing specifications right in the query.
        # Makes it trivial to update as your page's design changes.
        fixed(width: 125, height: 125) {
          ...GatsbyImageSharpFixed
        }
      }
    }
  }
`
Enter fullscreen mode Exit fullscreen mode

Which directory to put image?

After seeing this graphql code snippet, you may wonder where's the blog/avatars/kyle-mathews.jpeg? Is it under project root or src/images? let's take a test from a blank gatsby project.

create gatsby project:

$ gatsby new using-image-in-gatsby
Enter fullscreen mode Exit fullscreen mode

place kyle-mathews.jpeg image under src/images/blog/avatars/.

replace original index.js content with the following code:

index page code

Note: Dont forget the data property in IndexPage properties.

Now, we can start the website without the need to modify the other stuff:

$ gatsby develop
Enter fullscreen mode Exit fullscreen mode

Visit the http://localhost:8000/ in your browser, you'll see the correct result.

kyle

This experiment verified our speculation, that is:

gatsy-image query(graphql) using relative path under the path defined in gatsby-source-filesystem as query condition!

How about the dynamic image source value?

Above example use static value blog/avatars/kyle-mathews.jpeg to query in graphql expression. Then, what if the query condition is from markdown file meta data?

Official doc about Working with image in Markdown posts and pages told us the placing the featuredImage in the same position of markdown file:

file structure:

├── src
  ├── pages
    ├── example-post.md
    ├── awesome-image.png
Enter fullscreen mode Exit fullscreen mode

example-post.md:

--------
title: My Amazing Post
featuredImage: ./awesome-image.png
--------

Content goes here!
Enter fullscreen mode Exit fullscreen mode

In this solution, it works though, but, if you have many posts which include many images in each post, the directory structure will grow big dramatically and end up with chaos.

How to restructure the images path in a reasonable way?

  • Step one: deine a new content source in gatsby-config.js
plugins: [
    {
      resolve: `gatsby-source-filesystem`,
      options: {
        path: `${__dirname}/content`,
        name: `content`,
      },
    },
    ...
]
Enter fullscreen mode Exit fullscreen mode
  • Step two: place all the images under content/assets

separate image files and .md files to different folder

├── content
  ├── posts
    ├── example-post.md
    ├── ....md
  ├── assets
      ├── fantastic-image.png
      ├── awesome-image.png
      |── ...
├── src
Enter fullscreen mode Exit fullscreen mode
  • Step three: reference the image using relative path in markdown file metadata.

example-post.md:

--------
title: My Amazing Post
featuredImage: ../assets/awesome-image.png
--------

Business intro is missing...
Enter fullscreen mode Exit fullscreen mode

How to use html img tag in a Gatsby application

In a normal web application, html img element can use relative or absolute path to assign src property. We use image in a native and static way like the official doc said, place image under static folder, display image in webpage as we expected. But, when we build and deploy the gatsby site the GitHub Pages, visit it under an URL pattern like this:

https://username.github.io/your-gatsby-website/

The images declared by img tag broken all!

Although all the gatsby-images built by $ gatsby build --prefx-paths works, but those native imgs don't work.

What's the solution?

  • One solution is refactor all the imgs to gatsby-image components
  • Another solution is add context prefix the img src property.

For example:

<img src="/ueofcweb/images/project-home.jpg" className="img-fluid" alt="ultronele screenshot" />
Enter fullscreen mode Exit fullscreen mode

ueofcweb is the github project name, images/project-home.jpg is actually under ueofcweb/static.

If you're refactoring a traditional website which use a large amount of img tags, the second option would be a cheap option. My production officail site is migrated from bootstrap/jquey stack, while I was adding it to the Gatsby showcase, I spend hours to figure out that the second approach is the best way to me.

Last note I can give in this post is:

Do not include folders under static folder, it's a BAD practice ending up with build error `childImageSharp' of null.
Enter fullscreen mode Exit fullscreen mode

Gatsby image is the best responsive image solution which I have met. It frees you from the tedious optimization work in the build phase, add good performance and excellent user experience to your website. It deserves you to spend time to delve into and use proficiently.

Top comments (1)

Collapse
 
conspire_agency profile image
Conspire Agency • Edited

Have you ever seen a Gatsby Image wrapper (think that's what you'd call it) used to dynamically make the graph ql queries. Gatsby Image is great and all, but when you're using it for multiple images in a component the graphql really can get messy bloating the component.

Would be nice if you could for example:

<ImageWrapper fluid='filename.jpg' /> or <ImageWrapper fixed='filename.jpg' />

With all the graphql static queries internalized in the wrapper component.