The Gatsby Masterclass

Github Repositories

The Gatsby Masterclass Udemy course explains how to build blazing-fast, feature-rich, and overall stunning React apps with the Gatsby framework.

Table of contents

What I've learned

  • Build rich, fully-featured Gatsby sites / apps from project start to deployment on the web
  • Create stunning, blazing fast sites with Gatsby
  • Master GraphQL to get and manage data with your React apps
  • Utilize the best practices for building React projects.

1. Getting Started

1. What You Need for this Course

Microsoft Windows [Version 10.0.17763.195]
(c) 2018 Microsoft Corporation. All rights reserved.

C:\Windows\system32>cd C:\Work\Training\Pre\Gatsby

C:\Work\Training\Pre\Gatsby>npm i --global gatsby-cli
C:\Users\juan.pablo.perez\AppData\Roaming\npm\gatsby -> C:\Users\juan.pablo.perez\AppData\Roaming\npm\node_modules\gatsby-cli\lib\index.js
+ gatsby-cli@2.4.8
added 211 packages from 119 contributors in 30.836s

2. How to Get Help in this Course

  • To get help from the creator of the course we need to put @Reed at the begining of the title

  • If we want to get help for the other students we don't have to include @Reed

2. Creating our Gatsby Project

3. Using the Gatsby Docs

4. Creating our Project with the Gatsby CLI


C:\Work\Training\Pre\Gatsby>gatsby -v
2.4.8
  • Create the new gatsby-garb project
C:\Work\Training\Pre\Gatsby>gatsby new gatsby-garb
info Creating new site from git: https://github.com/gatsbyjs/gatsby-starter-default.git
Cloning into 'gatsby-garb'...
remote: Enumerating objects: 10, done.
remote: Counting objects: 100% (10/10), done.
remote: Compressing objects: 100% (9/9), done.
remote: Total 1082 (delta 1), reused 5 (delta 1), pack-reused 1072 eceiving objects: 100% (1082/1082), 1.75 MiB | 851.00Receivi
ng objects: 100% (1082/1082), 2.25 MiB | 915.00 KiB/s, done.
Resolving deltas: 100% (611/611), done.
success Created starter directory layout
info Installing packages...
yarn install v1.13.0
warning package-lock.json found. Your project contains lock files generated by tools other than Yarn. It is advised not to mix package managers in order to avoid resolution inconsistencies caused by unsynchronized lock files. To clear this warning, remove package-lock.json.
[1/4] Resolving packages...
[2/4] Fetching packages...
info fsevents@1.2.4: The platform "win32" is incompatible with this module.
info "fsevents@1.2.4" is an optional dependency and failed compatibility check. Excluding it from installation.
[3/4] Linking dependencies...
[4/4] Building fresh packages...
Done in 132.16s.
  • Open the folde with Visual Studio Code
C:\Work\Training\Pre\Gatsby>cd gatsby-garb

C:\Work\Training\Pre\Gatsby\gatsby-garb>code .

package.json

{
  "name": "gatsby-starter-default",
  "private": true,
  "description": "A simple starter to get up and developing quickly with Gatsby",
  "version": "0.1.0",
  "author": "Kyle Mathews <mathews.kyle@gmail.com>",
  "dependencies": {
    "gatsby": "^2.0.105",
    "gatsby-image": "^2.0.20",
    "gatsby-plugin-manifest": "^2.0.15",
    "gatsby-plugin-offline": "^2.0.22",
    "gatsby-plugin-react-helmet": "^3.0.2",
    "gatsby-plugin-sharp": "^2.0.19",
    "gatsby-source-filesystem": "^2.0.8",
    "gatsby-transformer-sharp": "^2.1.13",
    "prop-types": "^15.6.2",
    "react": "^16.6.3",
    "react-dom": "^16.6.3",
    "react-helmet": "^5.2.0"
  },
  "keywords": ["gatsby"],
  "license": "MIT",
  "scripts": {
    "build": "gatsby build",
    "develop": "gatsby develop",
    "start": "npm run develop",
    "format": "prettier --write \"src/**/*.js\"",
    "test": "echo \"Write tests! -> https://gatsby.app/unit-testing\""
  },
  "devDependencies": {
    "prettier": "^1.15.2"
  },
  "repository": {
    "type": "git",
    "url": "https://github.com/gatsbyjs/gatsby-starter-default"
  },
  "bugs": {
    "url": "https://github.com/gatsbyjs/gatsby/issues"
  }
}
  • Run the project with the npm start, npm run develop or gatsby develop
Juan.Pablo.Perez@RIMDUB-0232 MINGW64 /c/Work/Training/Pre/Gatsby/gatsby-garb
$ npm start

> gatsby-starter-default@0.1.0 start C:\Work\Training\Pre\Gatsby\gatsby-garb
> npm run develop


> gatsby-starter-default@0.1.0 develop C:\Work\Training\Pre\Gatsby\gatsby-garb
> gatsby develop

Something is already running at port 8000
Would you like to run the app at another port instead? [Y/n]Y

success load plugins — 0.372 s
success onPreInit — 2.495 s
success delete html and css files from previous builds — 0.033 s
success initialize cache — 0.070 s
success copy gatsby files — 0.602 s
success onPreBootstrap — 0.027 s
success source and transform nodes — 0.130 s
success building schema — 0.736 s
success createPages — 0.001 s
success createPagesStatefully — 0.108 s
success onPreExtractQueries — 0.011 s
success update schema — 0.264 s
success extract queries from components — 0.418 s
success run graphql queries — 0.794 s — 8/8 10.22 queries/second
success write out page data — 0.091 s
success write out redirect data — 0.011 s
Generating image thumbnails [==============================] 6/6 0.4 secs 100%

info bootstrap finished - 91.617 s

> done generating icons for manifest
success onPostBootstrap — 0.329 s
> Warning: React version not specified in eslint-plugin-react settings. See https://github.com/yannickcr/eslint-plugin-react#configuration .
 DONE  Compiled successfully in 12729ms                                                                                                                                                            7:26:31 PM

You can now view gatsby-starter-default in the browser.

  http://localhost:8001/

View GraphiQL, an in-browser IDE, to explore your site's data and schema

  http://localhost:8001/___graphql

Note that the development build is not optimized.
To create a production build, use gatsby build

i 「wdm」:
i 「wdm」: Compiled successfully.

5. Gatsby Starters to Easily Bootstrap New Projects (Optional)

  • There are different starters apart from the Gaspy Default Starter

  • We can chose another starter from Gatsby Starters

# create a new Gatsby site using the blog starter
npx gatsby new my-blog-starter https://github.com/gatsbyjs/gatsby-starter-blog
cd my-blog-starter/
gatsby develop

3. Exploring our Gatsby Project

6. Gatsby Project File Structure

/
|-- /.cache
|-- /plugins
|-- /public
|-- /src
    |-- /pages
    |-- /templates
    |-- html.js
|-- /static
|-- gatsby-config.js
|-- gatsby-node.js
|-- gatsby-ssr.js
|-- gatsby-browser.js
  • /.cache Automatically generated. This folder is an internal cache created automatically by Gatsby. The files inside this folder are not meant for modification. Should be .gitignore-d.

  • /public Automatically generated. The output of gatsby build process will be exposed inside this folder. Should be .gitignore-d. It contains the data that will be used to deploy the solution.
Juan.Pablo.Perez@RIMDUB-0232 MINGW64 /c/Work/Training/Pre/Gatsby/gatsby-garb
$ gatsby build
success open and validate gatsby-configs — 0.011 s
success load plugins — 0.297 s
success onPreInit — 3.037 s
success delete html and css files from previous builds — 0.034 s
success initialize cache — 0.012 s
success copy gatsby files — 0.391 s
success onPreBootstrap — 0.006 s
success source and transform nodes — 0.097 s
success building schema — 0.739 s
success createPages — 0.001 s
success createPagesStatefully — 0.068 s
success onPreExtractQueries — 0.008 s
success update schema — 0.148 s
success extract queries from components — 0.134 s
success run graphql queries — 0.079 s — 7/7 90.40 queries/second
success write out page data — 0.015 s
success write out redirect data — 0.002 s
⠄ onPostBootstrapdone generating icons for manifest
success onPostBootstrap — 0.191 s

info bootstrap finished - 15.895 s

success Building production JavaScript and CSS bundles — 13.383 s
success Building static HTML for pages — 1.530 s — 4/4 4.94 pages/second
info Done building in 30.819 sec

  • With gatsby serve a NodeJs server is started to run the code from the /public folder.
Juan.Pablo.Perez@RIMDUB-0232 MINGW64 /c/Work/Training/Pre/Gatsby/gatsby-garb
$ gatsby serve
info gatsby serve running at: http://localhost:9000/

  • /src This directory will contain all of the code related to what you will see on the frontend of your site (what you see in the browser), like your site header, or a page template. “Src” is a convention for “source code”.

    • /pages are ReactJs Components that under src/pages become pages automatically with paths based on their file name. Check out the pages docs for more detail.
    • /templates Contains templates for programmatically creating pages. Check out the templates docs for more detail.
    • html.js For custom configuration of default .cache/default_html.js. Check out the custom html docs for more detail.
    • /components are ReactJs components.
    • /images contain all the images of our solution.

  • .prettierrc contains the configuration of Prettier used to help the use of the editor.

.prettierrc

{
  "semi": false,
  "singleQuote": true,
  "trailingComma": "es5"
}
  • We can run npm format to format all the JavaScript document from the /src folder.
Juan.Pablo.Perez@RIMDUB-0232 MINGW64 /c/Work/Training/Pre/Gatsby/gatsby-garb
$ npm run format

> gatsby-starter-default@0.1.0 format C:\Work\Training\Pre\Gatsby\gatsby-garb
> prettier --write "src/**/*.js"

src\components\header.js 78ms
src\components\image.js 36ms
src\components\layout.js 27ms
src\components\seo.js 32ms
src\pages\404.js 18ms
src\pages\index.js 16ms
src\pages\page-2.js 10ms

7. Gatsby API Files / Using Git for File Tracking

gatsby-browser.js

/**
 * Implement Gatsby's Browser APIs in this file.
 *
 * See: https://www.gatsbyjs.org/docs/browser-apis/
 */

// You can delete this file if you're not using it

  • For example we could use the onRouteUpdate function.

gatsby-browser.js

exports.onRouteUpdate = ({ location, action }) => {
  console.log("route changed!", { location, action });
};

we need to restart the server calling againg to npm start after any change to the gatsby-*.js documents.

  • Init the git repository
Juan.Pablo.Perez@RIMDUB-0232 MINGW64 /c/Work/Training/Pre/Gatsby/gatsby-garb
$ git init
Initialized empty Git repository in C:/Work/Training/Pre/Gatsby/gatsby-garb/.git/

Juan.Pablo.Perez@RIMDUB-0232 MINGW64 /c/Work/Training/Pre/Gatsby/gatsby-garb (master)
$ git add .

Juan.Pablo.Perez@RIMDUB-0232 MINGW64 /c/Work/Training/Pre/Gatsby/gatsby-garb (master)
$ git status
On branch master

No commits yet

Changes to be committed:
  (use "git rm --cached <file>..." to unstage)

        new file:   .gitignore
        new file:   .prettierrc
        new file:   LICENSE
        new file:   README.md
        new file:   gatsby-browser.js
        new file:   gatsby-config.js
        new file:   gatsby-node.js
        new file:   gatsby-ssr.js
        new file:   package-lock.json
        new file:   package.json
        new file:   src/components/header.js
        new file:   src/components/image.js
        new file:   src/components/layout.css
        new file:   src/components/layout.js
        new file:   src/components/seo.js
        new file:   src/images/gatsby-astronaut.png
        new file:   src/images/gatsby-icon.png
        new file:   src/pages/404.js
        new file:   src/pages/index.js
        new file:   src/pages/page-2.js
        new file:   yarn.lock


Juan.Pablo.Perez@RIMDUB-0232 MINGW64 /c/Work/Training/Pre/Gatsby/gatsby-garb (master)
$ git commit -m "Initial commit"
[master (root-commit) cf449c4] Initial commit
 21 files changed, 28590 insertions(+)
 create mode 100644 .gitignore
 create mode 100644 .prettierrc
 create mode 100644 LICENSE
 create mode 100644 README.md
 create mode 100644 gatsby-browser.js
 create mode 100644 gatsby-config.js
 create mode 100644 gatsby-node.js
 create mode 100644 gatsby-ssr.js
 create mode 100644 package-lock.json
 create mode 100644 package.json
 create mode 100644 src/components/header.js
 create mode 100644 src/components/image.js
 create mode 100644 src/components/layout.css
 create mode 100644 src/components/layout.js
 create mode 100644 src/components/seo.js
 create mode 100644 src/images/gatsby-astronaut.png
 create mode 100644 src/images/gatsby-icon.png
 create mode 100644 src/pages/404.js
 create mode 100644 src/pages/index.js
 create mode 100644 src/pages/page-2.js
 create mode 100644 yarn.lock
  • We can remove the gatsby-browser.js and the gatsby-ssr.js documents. because are not going to be used in this course.

Juan.Pablo.Perez@RIMDUB-0232 MINGW64 /c/Work/Training/Pre/Gatsby/gatsby-garb (master)
$ git add .

Juan.Pablo.Perez@RIMDUB-0232 MINGW64 /c/Work/Training/Pre/Gatsby/gatsby-garb (master)
$ git commit -m "Removed unneeded files gatsby-browser.js and gatsby-ssr.js"
[master 5c9a312] Removed unneeded files gatsby-browser.js and gatsby-ssr.js
 2 files changed, 18 deletions(-)
 delete mode 100644 gatsby-browser.js
 delete mode 100644 gatsby-ssr.js

gatsby-config.js

module.exports = {
  siteMetadata: {
    title: `Gatsby Default Starter`,
    description: `Kick off your next, great Gatsby project with this default starter. This barebones starter ships with the main Gatsby configuration files you might need.`,
    author: `@gatsbyjs`
  },
  plugins: [
    `gatsby-plugin-react-helmet`,
    {
      resolve: `gatsby-source-filesystem`,
      options: {
        name: `images`,
        path: `${__dirname}/src/images`
      }
    },
    `gatsby-transformer-sharp`,
    `gatsby-plugin-sharp`,
    {
      resolve: `gatsby-plugin-manifest`,
      options: {
        name: `gatsby-starter-default`,
        short_name: `starter`,
        start_url: `/`,
        background_color: `#663399`,
        theme_color: `#663399`,
        display: `minimal-ui`,
        icon: `src/images/gatsby-icon.png` // This path is relative to the root of the site.
      }
    }
    // this (optional) plugin enables Progressive Web App + Offline functionality
    // To learn more, visit: https://gatsby.app/offline
    // 'gatsby-plugin-offline',
  ]
};

gatsby-node.js

/**
 * Implement Gatsby's Node APIs in this file.
 *
 * See: https://www.gatsbyjs.org/docs/node-apis/
 */

// You can delete this file if you're not using it

4. Pages and Components in Gatsby

8. Creating Pages, Routes and Nested Routes in Gatsby

  • Create our own page on the pages directory adding the new page-3.js document.

src -> pages -> page-3.js

  • Even though the page doesn't have any content GatsbyJs recompiles

  • But if we try to access the page we get logically an error:

  • We need to create our ReactJs component

src -> pages -> page-3.js

import React from "react";

export default () => (
  <div>
    <h1>Hello from Page 3</h1>
  </div>
);
  • And now, even thought we haven't created the route, the page is shown properly. The route is the name of the page.

  • In order to create nested routes we just have to create a subfolder with the route we want to create.

  • Create the new posts folder and move the page-3 from the main folder to the posts folder.

  • The layout component allows us to have a common layout that can be used in the different pages.

src -> componets -> layout.js

import React from "react";
import PropTypes from "prop-types";
import { StaticQuery, graphql } from "gatsby";

import Header from "./header";
import "./layout.css";

const Layout = ({ children }) => (
  <StaticQuery
    query={graphql`
      query SiteTitleQuery {
        site {
          siteMetadata {
            title
          }
        }
      }
    `}
    render={data => (
      <>
        <Header siteTitle={data.site.siteMetadata.title} />
        <div
          style={{
            margin: `0 auto`,
            maxWidth: 960,
            padding: `0px 1.0875rem 1.45rem`,
            paddingTop: 0
          }}
        >
          {children}
          <footer>
            © {new Date().getFullYear()}, Built with
            {` `}
            <a href="https://www.gatsbyjs.org">Gatsby</a>
          </footer>
        </div>
      </>
    )}
  />
);

Layout.propTypes = {
  children: PropTypes.node.isRequired
};

export default Layout;
  • The page-2 page created by defualt uses this layout component

src -> pages -> page-2.js

import React from "react";
import { Link } from "gatsby";

import Layout from "../components/layout";
import SEO from "../components/seo";

const SecondPage = () => (
  <Layout>
    <SEO title="Page two" />
    <h1>Hi from the second page</h1>
    <p>Welcome to page 2</p>
    <Link to="/">Go back to the homepage</Link>
  </Layout>
);

export default SecondPage;
  • We can modify our page-3 page to use the layout component.

src -> pages -> posts -> page-3.js

import React from "react";
import Layout from "../../components/layout";

export default () => (
  <Layout>
    <h1>Hello from Page 3!</h1>
  </Layout>
);

  • We can move through pages using the GatsbyJs Link component.

src -> pages -> posts -> page-3.js

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

export default () => (
  <Layout>
    <h1>Hello from Page 3!</h1>
    <Link to="/page-2">Go to Page 2</Link>
  </Layout>
);

5. Querying Data with GraphQL / GraphiQL IDE

10. Getting Site Metadata with GraphQL and Static Queries

  • In the layout component there is GrpahQL call that gets information from a metadata repository using the GastbyJs StaticQuery component, that allows very easily use common pieces of the Web Site in any other component or page. Most of the metadata is configured on the gatsby-config.js document.

src -> components -> layout.js

.
.
.
 <StaticQuery
    query={graphql`
      query SiteTitleQuery {
        site {
          siteMetadata {
            title
          }
        }
      }
    `}
.
.
.
  • We can change the value of the title just changing the siteMetadata.title value.

gatsby-config.js

module.exports = {
  siteMetadata: {
    title: `Gatsby Garb`,
    description: `Kick off your next, great Gatsby project with this default starter. This barebones starter ships with the main Gatsby configuration files you might need.`,
    author: `@gatsbyjs`
  },
  plugins: [
    `gatsby-plugin-react-helmet`,
    {
      resolve: `gatsby-source-filesystem`,
      options: {
        name: `images`,
        path: `${__dirname}/src/images`
      }
    },
    `gatsby-transformer-sharp`,
    `gatsby-plugin-sharp`,
    {
      resolve: `gatsby-plugin-manifest`,
      options: {
        name: `gatsby-starter-default`,
        short_name: `starter`,
        start_url: `/`,
        background_color: `#663399`,
        theme_color: `#663399`,
        display: `minimal-ui`,
        icon: `src/images/gatsby-icon.png` // This path is relative to the root of the site.
      }
    }
    // this (optional) plugin enables Progressive Web App + Offline functionality
    // To learn more, visit: https://gatsby.app/offline
    // 'gatsby-plugin-offline',
  ]
};

11. Using GraphiQL IDE to run our GraphQL Queries

  • We can use the GraphiQL IDE to practice what metadat information we can obtain from site.

  • We can access it from http://localhost:8001/___graphql

  • Clicking on the <Docs we can access the Documentation Explorer

request

{
  site {
    siteMetadata {
      title
      description
    }
  }
}

response

{
  "data": {
    "site": {
      "siteMetadata": {
        "title": "Gatsby Garb",
        "description": "Kick off your next, great Gatsby project with this default starter. This barebones starter ships with the main Gatsby configuration files you might need."
      }
    }
  }
}

12. Executing GraphQL Queries in our Gatsby Components

  • Include the author on the footer modifying the layout component.

src -> components -> layout.js

import React from "react";
import PropTypes from "prop-types";
import { StaticQuery, graphql } from "gatsby";

import Header from "./header";
import "./layout.css";

const getSiteMetadata = graphql`
  {
    site {
      siteMetadata {
        title
        author
      }
    }
  }
`;

const Layout = ({ children }) => (
  <StaticQuery
    query={getSiteMetadata}
    render={data => (
      <>
        <Header siteTitle={data.site.siteMetadata.title} />
        <div
          style={{
            margin: `0 auto`,
            maxWidth: 960,
            padding: `0px 1.0875rem 1.45rem`,
            paddingTop: 0
          }}
        >
          {children}
          <footer>
            © {new Date().getFullYear()}, Built by{" "}
            {data.site.siteMetadata.author}
          </footer>
        </div>
      </>
    )}
  />
);

Layout.propTypes = {
  children: PropTypes.node.isRequired
};

export default Layout;

gatsby-config.js

module.exports = {
  siteMetadata: {
    title: `Gatsby Garb`,
    description: `Kick off your next, great Gatsby project with this default starter. This barebones starter ships with the main Gatsby configuration files you might need.`,
    author: `Juan Pablo Perez`
  },
  plugins: [
    `gatsby-plugin-react-helmet`,
    {
      resolve: `gatsby-source-filesystem`,
      options: {
        name: `images`,
        path: `${__dirname}/src/images`
      }
    },
    `gatsby-transformer-sharp`,
    `gatsby-plugin-sharp`,
    {
      resolve: `gatsby-plugin-manifest`,
      options: {
        name: `gatsby-starter-default`,
        short_name: `starter`,
        start_url: `/`,
        background_color: `#663399`,
        theme_color: `#663399`,
        display: `minimal-ui`,
        icon: `src/images/gatsby-icon.png` // This path is relative to the root of the site.
      }
    }
    // this (optional) plugin enables Progressive Web App + Offline functionality
    // To learn more, visit: https://gatsby.app/offline
    // 'gatsby-plugin-offline',
  ]
};

13. Adding / Querying for Custom Fields on Site Metadata

  • We can add personal data adding new fields to the siteMetadata object.

gatsby-config.js

module.exports = {
  siteMetadata: {
    title: `Gatsby Garb`,
    description: `Kick off your next, great Gatsby project with this default starter. This barebones starter ships with the main Gatsby configuration files you might need.`,
    author: `Juan Pablo Perez`,
    createdAt: 2019
  },
  plugins: [
    `gatsby-plugin-react-helmet`,
    {
      resolve: `gatsby-source-filesystem`,
      options: {
        name: `images`,
        path: `${__dirname}/src/images`
      }
    },
    `gatsby-transformer-sharp`,
    `gatsby-plugin-sharp`,
    {
      resolve: `gatsby-plugin-manifest`,
      options: {
        name: `gatsby-starter-default`,
        short_name: `starter`,
        start_url: `/`,
        background_color: `#663399`,
        theme_color: `#663399`,
        display: `minimal-ui`,
        icon: `src/images/gatsby-icon.png` // This path is relative to the root of the site.
      }
    }
    // this (optional) plugin enables Progressive Web App + Offline functionality
    // To learn more, visit: https://gatsby.app/offline
    // 'gatsby-plugin-offline',
  ]
};

  • We need to restart the developtment script calling again npm start

  • We can include the field at the botton of the layout as well.

src -> components -> layout.js

import React from "react";
import PropTypes from "prop-types";
import { StaticQuery, graphql } from "gatsby";

import Header from "./header";
import "./layout.css";

const getSiteMetadata = graphql`
  {
    site {
      siteMetadata {
        title
        author
        createdAt
      }
    }
  }
`;

const Layout = ({ children }) => (
  <StaticQuery
    query={getSiteMetadata}
    render={data => (
      <>
        <Header siteTitle={data.site.siteMetadata.title} />
        <div
          style={{
            margin: `0 auto`,
            maxWidth: 960,
            padding: `0px 1.0875rem 1.45rem`,
            paddingTop: 0
          }}
        >
          {children}
          <footer>
            Built by {data.site.siteMetadata.author}, ©{" "}
            {data.site.siteMetadata.createdAt}
          </footer>
        </div>
      </>
    )}
  />
);

Layout.propTypes = {
  children: PropTypes.node.isRequired
};

export default Layout;

6. Source Plugins and Static Queries

14. Querying for Filesystem Data with the gatsby-source-filesystem

  • We cannot olny query from the metadata but for other sources as well. We can query data from the file system using the gastby-source-filesystem source.

gatsby-config.js

.
.
.
  plugins: [
    `gatsby-plugin-react-helmet`,
    {
      resolve: `gatsby-source-filesystem`,
      options: {
        name: `images`,
        path: `${__dirname}/src/images`,
      },
    },
.
.
.
.

request

{
  allFile {
    edges {
      node {
        relativePath
        size
        extension
        birthTime
      }
    }
  }
}

response

{
  "data": {
    "allFile": {
      "edges": [
        {
          "node": {
            "relativePath": "gatsby-astronaut.png",
            "size": 167273,
            "extension": "png",
            "birthTime": "2019-01-30T19:17:58.000Z"
          }
        },
        {
          "node": {
            "relativePath": "gatsby-icon.png",
            "size": 21212,
            "extension": "png",
            "birthTime": "2019-01-30T19:17:58.002Z"
          }
        }
      ]
    }
  }
}

15. Executing Static Queries from Scratch

  • Modify the page-3.js document to show information about the images

src -> pages -> posts -> page-3.js

import React from "react";
import { graphql, StaticQuery, Link } from "gatsby";
import Layout from "../../components/layout";

const getImageData = graphql`
  {
    allFile {
      edges {
        node {
          relativePath
          size
          extension
          birthTime
        }
      }
    }
  }
`;
export default () => (
  <Layout>
    <h1>Hello from Page 3!</h1>
    <h3>Image file data</h3>
    <StaticQuery
      query={getImageData}
      render={data => (
        <table>
          <thead>
            <tr>
              <th>Relative Path</th>
              <th>Size of Image</th>
              <th>Extension</th>
              <th>Birthtime</th>
            </tr>
          </thead>
          <tbody>
            {data.allFile.edges.map(({ node }, index) => (
              <tr key={index}>
                <td>{node.relativePath}</td>
                <td>{node.size}</td>
                <td>{node.extension}</td>
                <td>{node.birthTime}</td>
              </tr>
            ))}
          </tbody>
        </table>
      )}
    />
    <Link to="/page-2">Go to Page 2</Link>
  </Layout>
);

7. Transformer Plugins / Creating a Blog out of Markdown Files

16. Transformer Plugins / Using gatsby-transformer-remark to Transform Markdown

  • The transformer plugin allows to get file data, like the source plugin, but it also allows us to transform it.

  • We are going to use it to transform Markdown document into Html documents.

  • Install the gatsby-transformer-remark plugin

C:\Work\Training\Pre\Gatsby\gatsby-garb>npm i gatsby-transformer-remark

> sharp@0.21.3 install C:\Work\Training\Pre\Gatsby\gatsby-garb\node_modules\sharp
> (node install/libvips && node install/dll-copy && prebuild-install) || (node-gyp rebuild && node install/dll-copy)

info sharp Using cached C:\Users\juan.pablo.perez\AppData\Roaming\npm-cache\_libvips\libvips-8.7.0-win32-x64.tar.gz
info sharp Creating C:\Work\Training\Pre\Gatsby\gatsby-garb\node_modules\sharp\build\Release
info sharp Copying DLLs from C:\Work\Training\Pre\Gatsby\gatsby-garb\node_modules\sharp\vendor\lib to C:\Work\Training\Pre\Gatsby\gatsby-garb\node_modules\sharp\build\Release

> cwebp-bin@5.0.0 postinstall C:\Work\Training\Pre\Gatsby\gatsby-garb\node_modules\cwebp-bin
> node lib/install.js

  √ cwebp pre-build test passed successfully

> mozjpeg@6.0.1 postinstall C:\Work\Training\Pre\Gatsby\gatsby-garb\node_modules\mozjpeg
> node lib/install.js

  √ mozjpeg pre-build test passed successfully

> pngquant-bin@5.0.1 postinstall C:\Work\Training\Pre\Gatsby\gatsby-garb\node_modules\pngquant-bin
> node lib/install.js

  √ pngquant pre-build test passed successfully
npm WARN optional SKIPPING OPTIONAL DEPENDENCY: fsevents@1.2.4 (node_modules\fsevents):
npm WARN notsup SKIPPING OPTIONAL DEPENDENCY: Unsupported platform for fsevents@1.2.4: wanted {"os":"darwin","arch":"any"} (current: {"os":"win32","arch":"x64"})

+ gatsby-transformer-remark@2.2.4
added 1778 packages from 846 contributors, updated 143 packages and audited 25499 packages in 75.754s
found 0 vulnerabilities
  • Modiy the gatsby-config.js to include the new plugin

gatsby-config.js

module.exports = {
  siteMetadata: {
    title: `Gatsby Garb`,
    description: `Kick off your next, great Gatsby project with this default starter. This barebones starter ships with the main Gatsby configuration files you might need.`,
    author: `Juan Pablo Perez`,
    createdAt: 2019
  },
  plugins: [
    `gatsby-plugin-react-helmet`,
    {
      resolve: `gatsby-source-filesystem`,
      options: {
        name: `images`,
        path: `${__dirname}/src/images`
      }
    },
    `gatsby-transformer-remark`,
    {
      resolve: `gatsby-source-filesystem`,
      options: {
        name: `pages`,
        path: `${__dirname}/src/pages`
      }
    },
    `gatsby-transformer-sharp`,
    `gatsby-plugin-sharp`,
    {
      resolve: `gatsby-plugin-manifest`,
      options: {
        name: `gatsby-starter-default`,
        short_name: `starter`,
        start_url: `/`,
        background_color: `#663399`,
        theme_color: `#663399`,
        display: `minimal-ui`,
        icon: `src/images/gatsby-icon.png` // This path is relative to the root of the site.
      }
    }
    // this (optional) plugin enables Progressive Web App + Offline functionality
    // To learn more, visit: https://gatsby.app/offline
    // 'gatsby-plugin-offline',
  ]
};
  • Add the new blog.js document page (empty, at the moment)

src\pages\blog.js

  • Add the new post-one.md Markdown file

src\pages\post-one.md

---
title: "My First Markdown Post"
date: "2019-02-09"
---

Hello, this is my first post made using Markdown!

Request

{
  allMarkdownRemark {
    totalCount
  }
}

Response

{
  "data": {
    "allMarkdownRemark": {
      "totalCount": 1
    }
  }
}

Request

{
  allMarkdownRemark {
    edges {
      node {
        frontmatter {
          title
          date
        }
        html
      }
    }
  }
}

Response

{
  "data": {
    "allMarkdownRemark": {
      "edges": [
        {
          "node": {
            "frontmatter": {
              "title": "My First Markdown Post",
              "date": "2019-02-09"
            },
            "html": "<p>Hello, this is my first post made using Markdown!</p>"
          }
        }
      ]
    }
  }
}

17. Displaying Preview of Markdown Posts in Blog Page

  • We can use the excerpt field to obatin a preview content.

Request

{
  allMarkdownRemark {
    totalCount
    edges {
      node {
        id
        frontmatter {
          title
          date
        }
        excerpt
      }
    }
  }
}

Response

{
  "data": {
    "allMarkdownRemark": {
      "totalCount": 1,
      "edges": [
        {
          "node": {
            "id": "49bd7379-604f-5d18-a123-b72169825673",
            "frontmatter": {
              "title": "My First Markdown Post",
              "date": "2019-02-09"
            },
            "excerpt": "Hello, this is my first post made using Markdown!"
          }
        }
      ]
    }
  }
}
  • Modify the blog.js document to include a preview of all the post Markdown documents

src\pages\blog.js

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

import Layout from "../components/layout";

const getMarkdownPosts = graphql`
  {
    allMarkdownRemark {
      totalCount
      edges {
        node {
          id
          frontmatter {
            title
            date
          }
          excerpt
        }
      }
    }
  }
`;

export default () => (
  <Layout>
    <div>
      <h1 style={{ display: "inlineBlock", borderBottom: "1px solid" }}>
        Gatsby Garb Blog
      </h1>
      <StaticQuery
        query={getMarkdownPosts}
        render={data => (
          <>
            <h4>{data.allMarkdownRemark.totalCount} Posts</h4>
            {data.allMarkdownRemark.edges.map(({ node }) => (
              <div key={node.id}>
                <h3>
                  {node.frontmatter.title}{" "}
                  <span style={{ color: "#bbb" }}>
                    - {node.frontmatter.date}
                  </span>
                </h3>
                <p>{node.excerpt}</p>
              </div>
            ))}
          </>
        )}
      />
    </div>
  </Layout>
);

  • We can add a second Markdown post document

src\pages\post-two.md

---
title: "The Second Gatsby Garb Post"
date: "2019-02-10"
---

You're currently reading the second post made using Markdown!

DisplayingPreviewOfMarkdownPostsInBlogPage3 DisplayingPreviewOfMarkdownPostsInBlogPage4

8. Programmatically Creating Pages with Gatsby

18. Programmatically Creating Slugs for Blog Posts with gatsby-node

  • The gastby-node.js document allows us to create pages programatically.

  • We are going to use OnCreateNode

Called when a new node is created. Plugins wishing to extend or transform nodes created by other plugins should implement this API.

Example

exports.onCreateNode = ({ node, actions }) => {
  const { createNode, createNodeField } = actions;
  // Transform the new node here and create a new node or
  // create a new node field.
};
  • We need to modify gastby-node.js to allow us to manage the use of onCreateNode.

gastby-node.js

const { createFilePath } = require("gatsby-source-filesystem");

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

  if (node.internal.type === "MarkdownRemark") {
    const slug = createFilePath({ node, getNode, basepath: "posts" });
    createNodeField({
      node,
      name: "slug",
      value: slug
    });
  }
};
  • We can now query the new slug field created.

Request

{
  allMarkdownRemark {
    totalCount
    edges {
      node {
        fields {
          slug
        }
        id
        frontmatter {
          title
          date
        }
        excerpt
      }
    }
  }
}

Response

{
  "data": {
    "allMarkdownRemark": {
      "totalCount": 2,
      "edges": [
        {
          "node": {
            "fields": {
              "slug": "/post-one/"
            },
            "id": "49bd7379-604f-5d18-a123-b72169825673",
            "frontmatter": {
              "title": "My First Markdown Post",
              "date": "2019-02-09"
            },
            "excerpt": "Hello, this is my first post made using Markdown!"
          }
        },
        {
          "node": {
            "fields": {
              "slug": "/post-two/"
            },
            "id": "fe04fc66-45a4-54f6-8502-3150bb52bde2",
            "frontmatter": {
              "title": "The Second Gatsby Garb Post",
              "date": "2019-02-10"
            },
            "excerpt": "You're currently reading the second post made using Markdown!"
          }
        }
      ]
    }
  }
}

19. Programmatically Creating Pages with gatsby-node

  • We are going to query our page data with GraphQL and generate individual html pages from the Markdown documents. In order to do this we are going to use another function called createPages.

Tell plugins to add pages. This extension point is called only after the initial sourcing and transformation of nodes plus creation of the GraphQL schema are complete so you can query your data in order to create pages.

Example

const path = require(`path`);

exports.createPages = ({ graphql, actions }) => {
  const { createPage } = actions;
  const blogPostTemplate = path.resolve(`src/templates/blog-post.js`);
  // Query for markdown nodes to use in creating pages.
  // You can query for whatever data you want to create pages for e.g.
  // products, portfolio items, landing pages, etc.
  return graphql(`
    {
      allMarkdownRemark(limit: 1000) {
        edges {
          node {
            fields {
              slug
            }
          }
        }
      }
    }
  `).then(result => {
    if (result.errors) {
      throw result.errors;
    }

    // Create blog post pages.
    result.data.allMarkdownRemark.edges.forEach(edge => {
      createPage({
        // Path for this page — required
        path: `${edge.node.fields.slug}`,
        component: blogPostTemplate,
        context: {
          // Add optional context data to be inserted
          // as props into the page component..
          //
          // The context data can also be used as
          // arguments to the page GraphQL query.
          //
          // The page "path" is always available as a GraphQL
          // argument.
        }
      });
    });
  });
};
  • We need to modify gastby-node.js to allow us to manage the use of createPages using the following query:

Request

{
  allMarkdownRemark {
    edges {
      node {
        fields {
          slug
        }
      }
    }
  }
}

Response

{
  "data": {
    "allMarkdownRemark": {
      "edges": [
        {
          "node": {
            "fields": {
              "slug": "/post-one/"
            }
          }
        },
        {
          "node": {
            "fields": {
              "slug": "/post-two/"
            }
          }
        }
      ]
    }
  }
}
  • Before modifying the gastby-node.js document we neeed to create a template (ReactJs component) used to create the pages. A new templates folder must be created and inside it we need to create the post-template.js document.

templates/post-template.js

import React from "react";

import Layout from "../components/layout";

export default () => (
  <Layout>
    <h2>Programmatically created blog post!</h2>
  </Layout>
);

gastby-node.js

const path = require("path");
const { createFilePath } = require("gatsby-source-filesystem");

const PostTemplate = path.resolve("./src/templates/post-template.js");

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

  if (node.internal.type === "MarkdownRemark") {
    const slug = createFilePath({ node, getNode, basepath: "posts" });
    createNodeField({
      node,
      name: "slug",
      value: slug
    });
  }
};

exports.createPages = async ({ graphql, actions }) => {
  const { createPage } = actions;
  const result = await graphql(`
    {
      allMarkdownRemark {
        edges {
          node {
            fields {
              slug
            }
          }
        }
      }
    }
  `);
  const posts = result.data.allMarkdownRemark.edges;

  posts.forEach(({ node: post }) => {
    createPage({
      path: `posts${post.fields.slug}`,
      component: PostTemplate,
      context: {
        slug: post.fields.slug
      }
    });
  });
};
  • Move page-3.js out of the posts folder to the pages folder and then remove the posts folder.

  • After restarting the node server we can see that even though there is no post folder created it is created dinamically by Gatsby.

9. GraphQL Arguments, Variables, and Page Queries

20. Using GraphQL Arguments and Variables to Get Individual Posts

  • We can use the markdownRemark GraphQL funtion to get information about an individual post.

Request

{
  markdownRemark(fields: { slug: { eq: "/post-two/" } }) {
    html
    frontmatter {
      title
    }
  }
}

Response

{
  "data": {
    "markdownRemark": {
      "html": "<p>You're currently reading the second post made using Markdown!</p>",
      "frontmatter": {
        "title": "The Second Gatsby Garb Post"
      }
    }
  }
}
  • We can use the query graphQL command to query using variables

Request

query($slug: String) {
  markdownRemark(fields: { slug: { eq: $slug } }) {
    html
    frontmatter {
      title
    }
  }
}

Query variables

{
  "slug": "/post-one/"
}

Response

{
  "data": {
    "markdownRemark": {
      "html": "<p>Hello, this is my first post made using Markdown!</p>",
      "frontmatter": {
        "title": "My First Markdown Post"
      }
    }
  }
}

21. Fetching Individual Post Data by Slug with Page Queries

  • Modify the post-template.js template document to properly generate the html pages.

src/templates/post-template.js

import React from "react";
import { graphql } from "gatsby";

import Layout from "../components/layout";

const PostTemplate = ({ data: post }) => (
  <Layout>
    <h1>{post.markdownRemark.frontmatter.title}</h1>
    <div dangerouslySetInnerHTML={{ __html: post.markdownRemark.html }} />
  </Layout>
);

// This will be rendered and sent to the `data` parameter
export const query = graphql`
  query($slug: String) {
    markdownRemark(fields: { slug: { eq: $slug } }) {
      html
      frontmatter {
        title
      }
    }
  }
`;

export default PostTemplate;

10. Pagination in Gatsby

  • Modify the blog.js document to include the link using the fields/slug field.

pages\blog.js

import React from "react";
import { StaticQuery, graphql, Link } from "gatsby";

import Layout from "../components/layout";

const getMarkdownPosts = graphql`
  {
    allMarkdownRemark {
      totalCount
      edges {
        node {
          fields {
            slug
          }
          id
          frontmatter {
            title
            date
          }
          excerpt
        }
      }
    }
  }
`;

export default () => (
  <Layout>
    <div>
      <h1 style={{ display: "inlineBlock", borderBottom: "1px solid" }}>
        Gatsby Garb Blog
      </h1>
      <StaticQuery
        query={getMarkdownPosts}
        render={data => (
          <>
            <h4>{data.allMarkdownRemark.totalCount} Posts</h4>
            {data.allMarkdownRemark.edges.map(({ node }) => (
              <div key={node.id}>
                <h3>
                  <Link to={`/post${node.fields.slug}`}>
                    {node.frontmatter.title}
                  </Link>{" "}
                  <span style={{ color: "#bbb" }}>
                    - {node.frontmatter.date}
                  </span>
                </h3>
                <p>{node.excerpt}</p>
              </div>
            ))}
          </>
        )}
      />
    </div>
  </Layout>
);

  • Move pages\blog.js to templates\blog_template.js

  • Modify the gatsby-node.js document to include pagination of the Blogs.

gatsby-node.js

const path = require("path");
const { createFilePath } = require("gatsby-source-filesystem");

const PostTemplate = path.resolve("./src/templates/post-template.js");
const BlogTemplate = path.resolve("./src/templates/blog-template.js");

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

  if (node.internal.type === "MarkdownRemark") {
    const slug = createFilePath({ node, getNode, basepath: "posts" });
    createNodeField({
      node,
      name: "slug",
      value: slug
    });
  }
};

exports.createPages = async ({ graphql, actions }) => {
  const { createPage } = actions;
  const result = await graphql(`
    {
      allMarkdownRemark {
        edges {
          node {
            fields {
              slug
            }
          }
        }
      }
    }
  `);
  const posts = result.data.allMarkdownRemark.edges;

  posts.forEach(({ node: post }) => {
    createPage({
      path: `posts${post.fields.slug}`,
      component: PostTemplate,
      context: {
        slug: post.fields.slug
      }
    });
  });

  posts.forEach((_, index, postArr) => {
    const totalPages = postArr.length;
    const postsPerPage = 1;
    const currentPage = index + 1;
    const isFirstPage = index === 0;
    const isLastPage = currentPage === totalPages;
    createPage({
      path: isFirstPage ? "/blog/" : `blog/${currentPage}`,
      component: BlogTemplate,
      context: {
        limit: postsPerPage,
        skip: index * postsPerPage,
        isFirstPage,
        isLastPage,
        currentPage,
        totalPages
      }
    });
  });
};
  • Update the blog_template.js to include and use the new fields added.

templates\blog_template.js

import React from "react";
import { graphql, Link } from "gatsby";

import Layout from "../components/layout";

export default ({ data, pageContext }) => {
  const { currentPage, isFirstPage, isLastPage } = pageContext;
  const nextPage = `/blog/${String(currentPage + 1)}`;
  const prevPage =
    currentPage - 1 === 1 ? "/blog" : `/blog/${String(currentPage - 1)}`;
  return (
    <Layout>
      <div>
        <h1 style={{ display: "inlineBlock", borderBottom: "1px solid" }}>
          Gatsby Garb Blog
        </h1>
        <h4>{data.allMarkdownRemark.totalCount} Posts</h4>
        {data.allMarkdownRemark.edges.map(({ node }) => (
          <div key={node.id}>
            <h3>
              <Link to={`/posts${node.fields.slug}`}>
                {node.frontmatter.title}
              </Link>{" "}
              <span style={{ color: "#bbb" }}>- {node.frontmatter.date}</span>
            </h3>
            <p>{node.excerpt}</p>
          </div>
        ))}
        {/* Pagination Links */}
        <div>
          {!isFirstPage && (
            <Link to={prevPage} rel="prev">
              Prev Page
            </Link>
          )}
          {!isLastPage && (
            <Link to={nextPage} rel="next">
              Next Page
            </Link>
          )}
        </div>
      </div>
    </Layout>
  );
};
export const query = graphql`
  query($skip: Int!, $limit: Int!) {
    allMarkdownRemark(skip: $skip, limit: $limit) {
      totalCount
      edges {
        node {
          fields {
            slug
          }
          id
          frontmatter {
            title
            date
          }
          excerpt
        }
      }
    }
  }
`;

24. Adding Numbered Pagination

  • Move all the posts to the new posts folder

  • We need to modify the gatsby-transformer-remark plugin to point at the posts folder in the gastby-config.js document.

    gastby-config.js

module.exports = {
  siteMetadata: {
    title: `Gatsby Garb`,
    description: `Kick off your next, great Gatsby project with this default starter. This barebones starter ships with the main Gatsby configuration files you might need.`,
    author: `Juan Pablo Perez`,
    createdAt: 2019
  },
  plugins: [
    `gatsby-plugin-react-helmet`,
    {
      resolve: `gatsby-source-filesystem`,
      options: {
        name: `images`,
        path: `${__dirname}/src/images`
      }
    },
    `gatsby-transformer-remark`,
    {
      resolve: `gatsby-source-filesystem`,
      options: {
        name: `posts`,
        path: `${__dirname}/src/posts`
      }
    },
    `gatsby-transformer-sharp`,
    `gatsby-plugin-sharp`,
    {
      resolve: `gatsby-plugin-manifest`,
      options: {
        name: `gatsby-starter-default`,
        short_name: `starter`,
        start_url: `/`,
        background_color: `#663399`,
        theme_color: `#663399`,
        display: `minimal-ui`,
        icon: `src/images/gatsby-icon.png` // This path is relative to the root of the site.
      }
    }
    // this (optional) plugin enables Progressive Web App + Offline functionality
    // To learn more, visit: https://gatsby.app/offline
    // 'gatsby-plugin-offline',
  ]
};
  • Create two new posts

post-three.md

---
title: "Third Gatsby Garb Post"
date: "2019-02-13"
---

Hello, this is my third post so far!

post-four.md

---
title: "My Fourth Markdown Post"
date: "2019-02-14"
---

Hello, this is my fourth post made using Markdown and I'm getting practice!

  • Modify the blog-template.js document to include the number of each page.

templates\blog_template.js

import React from "react";
import { graphql, Link } from "gatsby";

import Layout from "../components/layout";

export default ({ data, pageContext }) => {
  const { currentPage, isFirstPage, isLastPage, totalPages } = pageContext;
  const nextPage = `/blog/${String(currentPage + 1)}`;
  const prevPage =
    currentPage - 1 === 1 ? "/blog" : `/blog/${String(currentPage - 1)}`;
  return (
    <Layout>
      <div>
        <h1 style={{ display: "inlineBlock", borderBottom: "1px solid" }}>
          Gatsby Garb Blog
        </h1>
        <h4>{data.allMarkdownRemark.totalCount} Posts</h4>
        {data.allMarkdownRemark.edges.map(({ node }) => (
          <div key={node.id}>
            <h3>
              <Link to={`/posts${node.fields.slug}`}>
                {node.frontmatter.title}
              </Link>{" "}
              <span style={{ color: "#bbb" }}>- {node.frontmatter.date}</span>
            </h3>
            <p>{node.excerpt}</p>
          </div>
        ))}
        {/* Pagination Links */}
        <div>
          {!isFirstPage && (
            <Link to={prevPage} rel="prev">
              Prev Page
            </Link>
          )}
          {Array.from({ length: totalPages }, (_, index) => (
            <Link key={index} to={`/blog/${index === 0 ? "" : index + 1}`}>
              {index + 1}
            </Link>
          ))}
          {!isLastPage && (
            <Link to={nextPage} rel="next">
              Next Page
            </Link>
          )}
        </div>
      </div>
    </Layout>
  );
};
export const query = graphql`
  query($skip: Int!, $limit: Int!) {
    allMarkdownRemark(skip: $skip, limit: $limit) {
      totalCount
      edges {
        node {
          fields {
            slug
          }
          id
          frontmatter {
            title
            date
          }
          excerpt
        }
      }
    }
  }
`;

  • We are going to format the pagination

templates\blog_template.js

import React from "react";
import { graphql, Link } from "gatsby";

import Layout from "../components/layout";

export default ({ data, pageContext }) => {
  const { currentPage, isFirstPage, isLastPage, totalPages } = pageContext;
  const nextPage = `/blog/${String(currentPage + 1)}`;
  const prevPage =
    currentPage - 1 === 1 ? "/blog" : `/blog/${String(currentPage - 1)}`;
  return (
    <Layout>
      <div>
        <h1 style={{ display: "inlineBlock", borderBottom: "1px solid" }}>
          Gatsby Garb Blog
        </h1>
        <h4>{data.allMarkdownRemark.totalCount} Posts</h4>
        {data.allMarkdownRemark.edges.map(({ node }) => (
          <div key={node.id}>
            <h3>
              <Link to={`/posts${node.fields.slug}`}>
                {node.frontmatter.title}
              </Link>{" "}
              <span style={{ color: "#bbb" }}>- {node.frontmatter.date}</span>
            </h3>
            <p>{node.excerpt}</p>
          </div>
        ))}
        {/* Pagination Links */}
        <div
          style={{
            display: "flex",
            alignItems: "center",
            justifyContent: "space-around",
            maxWidth: 300,
            margin: "0 auto"
          }}
        >
          {!isFirstPage && (
            <Link to={prevPage} rel="prev">
              Prev Page
            </Link>
          )}
          {Array.from({ length: totalPages }, (_, index) => (
            <Link key={index} to={`/blog/${index === 0 ? "" : index + 1}`}>
              {index + 1}
            </Link>
          ))}
          {!isLastPage && (
            <Link to={nextPage} rel="next">
              Next Page
            </Link>
          )}
        </div>
      </div>
    </Layout>
  );
};
export const query = graphql`
  query($skip: Int!, $limit: Int!) {
    allMarkdownRemark(skip: $skip, limit: $limit) {
      totalCount
      edges {
        node {
          fields {
            slug
          }
          id
          frontmatter {
            title
            date
          }
          excerpt
        }
      }
    }
  }
`;

  • We need to modify the gatsby-node.js to make the number of posts per page configurable.

gatsby-node.js

const path = require("path");
const { createFilePath } = require("gatsby-source-filesystem");

const PostTemplate = path.resolve("./src/templates/post-template.js");
const BlogTemplate = path.resolve("./src/templates/blog-template.js");

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

  if (node.internal.type === "MarkdownRemark") {
    const slug = createFilePath({ node, getNode, basepath: "posts" });
    createNodeField({
      node,
      name: "slug",
      value: slug
    });
  }
};

exports.createPages = async ({ graphql, actions }) => {
  const { createPage } = actions;
  const result = await graphql(`
    {
      allMarkdownRemark {
        edges {
          node {
            fields {
              slug
            }
          }
        }
      }
    }
  `);
  const posts = result.data.allMarkdownRemark.edges;

  posts.forEach(({ node: post }) => {
    createPage({
      path: `posts${post.fields.slug}`,
      component: PostTemplate,
      context: {
        slug: post.fields.slug
      }
    });
  });

  const postsPerPage = 2;
  const totalPages = Math.ceil(posts.length / postsPerPage);

  Array.from({ length: totalPages }).forEach((_, index) => {
    const currentPage = index + 1;
    const isFirstPage = index === 0;
    const isLastPage = currentPage === totalPages;
    createPage({
      path: isFirstPage ? "/blog/" : `blog/${currentPage}`,
      component: BlogTemplate,
      context: {
        limit: postsPerPage,
        skip: index * postsPerPage,
        isFirstPage,
        isLastPage,
        currentPage,
        totalPages
      }
    });
  });
};

11. Sorting and Filtering with GraphQL Queries / Formatting Dates and Text

25. Sorting, Filtering, Limiting, Skipping w/ GraphQL Arguments / Listing Blog Posts

Query using skip and limit.

{
  allMarkdownRemark(limit: 2, skip: 1) {
    totalCount
    edges {
      node {
        id
        frontmatter {
          title
          date
        }
        excerpt
      }
    }
  }
}

Response

{
  "data": {
    "allMarkdownRemark": {
      "totalCount": 4,
      "edges": [
        {
          "node": {
            "id": "683d1527-5d01-5c36-8f4a-900b528c9473",
            "frontmatter": {
              "title": "My Fourth Markdown Post",
              "date": "2019-02-14"
            },
            "excerpt": "Hello, this is my fourth post made using Markdown and I'm getting practice!"
          }
        },
        {
          "node": {
            "id": "d4004257-d250-5042-8af4-618d9114a084",
            "frontmatter": {
              "title": "Third Gatsby Garb Post",
              "date": "2019-02-13"
            },
            "excerpt": "Hello, this is my third post so far!"
          }
        }
      ]
    }
  }
}

Query using filter.

{
  allMarkdownRemark(
    filter: { frontmatter: { title: { eq: "My Fourth Markdown Post" } } }
  ) {
    totalCount
    edges {
      node {
        id
        frontmatter {
          title
          date
        }
        excerpt
      }
    }
  }
}

Response

{
  "data": {
    "allMarkdownRemark": {
      "totalCount": 1,
      "edges": [
        {
          "node": {
            "id": "683d1527-5d01-5c36-8f4a-900b528c9473",
            "frontmatter": {
              "title": "My Fourth Markdown Post",
              "date": "2019-02-14"
            },
            "excerpt": "Hello, this is my fourth post made using Markdown and I'm getting practice!"
          }
        }
      ]
    }
  }
}

Query using filter with nin.

{
  allMarkdownRemark(
    filter: {
      frontmatter: {
        title: { nin: ["My Fourth Markdown Post", "Third Gatsby Garb Post"] }
      }
    }
  ) {
    totalCount
    edges {
      node {
        id
        frontmatter {
          title
          date
        }
        excerpt
      }
    }
  }
}

Response

{
  "data": {
    "allMarkdownRemark": {
      "totalCount": 2,
      "edges": [
        {
          "node": {
            "id": "ac80ba83-9ee3-51dc-9ceb-c602bc97302a",
            "frontmatter": {
              "title": "My First Markdown Post",
              "date": "2019-02-09"
            },
            "excerpt": "Hello, this is my first post made using Markdown!"
          }
        },
        {
          "node": {
            "id": "2cbc3788-9a75-5571-9ee1-25664101c806",
            "frontmatter": {
              "title": "The Second Gatsby Garb Post",
              "date": "2019-02-10"
            },
            "excerpt": "You're currently reading the second post made using Markdown!"
          }
        }
      ]
    }
  }
}

Query using sort.

{
  allMarkdownRemark(sort: { fields: [frontmatter___date], order: DESC }) {
    totalCount
    edges {
      node {
        id
        frontmatter {
          title
          date
        }
        excerpt
      }
    }
  }
}

Response

{
  "data": {
    "allMarkdownRemark": {
      "totalCount": 4,
      "edges": [
        {
          "node": {
            "id": "683d1527-5d01-5c36-8f4a-900b528c9473",
            "frontmatter": {
              "title": "My Fourth Markdown Post",
              "date": "2019-02-14"
            },
            "excerpt": "Hello, this is my fourth post made using Markdown and I'm getting practice!"
          }
        },
        {
          "node": {
            "id": "d4004257-d250-5042-8af4-618d9114a084",
            "frontmatter": {
              "title": "Third Gatsby Garb Post",
              "date": "2019-02-13"
            },
            "excerpt": "Hello, this is my third post so far!"
          }
        },
        {
          "node": {
            "id": "2cbc3788-9a75-5571-9ee1-25664101c806",
            "frontmatter": {
              "title": "The Second Gatsby Garb Post",
              "date": "2019-02-10"
            },
            "excerpt": "You're currently reading the second post made using Markdown!"
          }
        },
        {
          "node": {
            "id": "ac80ba83-9ee3-51dc-9ceb-c602bc97302a",
            "frontmatter": {
              "title": "My First Markdown Post",
              "date": "2019-02-09"
            },
            "excerpt": "Hello, this is my first post made using Markdown!"
          }
        }
      ]
    }
  }
}
  • Modify the gatsby-node.js document to limit the number of documents returned to 1,000.

gatsby-node.js

const path = require("path");
const { createFilePath } = require("gatsby-source-filesystem");

const PostTemplate = path.resolve("./src/templates/post-template.js");
const BlogTemplate = path.resolve("./src/templates/blog-template.js");

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

  if (node.internal.type === "MarkdownRemark") {
    const slug = createFilePath({ node, getNode, basepath: "posts" });
    createNodeField({
      node,
      name: "slug",
      value: slug
    });
  }
};

exports.createPages = async ({ graphql, actions }) => {
  const { createPage } = actions;
  const result = await graphql(`
    {
      allMarkdownRemark(limit: 1000) {
        edges {
          node {
            fields {
              slug
            }
          }
        }
      }
    }
  `);
  const posts = result.data.allMarkdownRemark.edges;

  posts.forEach(({ node: post }) => {
    createPage({
      path: `posts${post.fields.slug}`,
      component: PostTemplate,
      context: {
        slug: post.fields.slug
      }
    });
  });

  const postsPerPage = 2;
  const totalPages = Math.ceil(posts.length / postsPerPage);

  Array.from({ length: totalPages }).forEach((_, index) => {
    const currentPage = index + 1;
    const isFirstPage = index === 0;
    const isLastPage = currentPage === totalPages;
    createPage({
      path: isFirstPage ? "/blog/" : `blog/${currentPage}`,
      component: BlogTemplate,
      context: {
        limit: postsPerPage,
        skip: index * postsPerPage,
        isFirstPage,
        isLastPage,
        currentPage,
        totalPages
      }
    });
  });
};
  • Modify the templates\blog-template.js document to use the order clause.

templates\blog-template.js

import React from "react";
import { graphql, Link } from "gatsby";

import Layout from "../components/layout";

export default ({ data, pageContext }) => {
  const { currentPage, isFirstPage, isLastPage, totalPages } = pageContext;
  const nextPage = `/blog/${String(currentPage + 1)}`;
  const prevPage =
    currentPage - 1 === 1 ? "/blog" : `/blog/${String(currentPage - 1)}`;
  return (
    <Layout>
      <div>
        <h1 style={{ display: "inlineBlock", borderBottom: "1px solid" }}>
          Gatsby Garb Blog
        </h1>
        <h4>{data.allMarkdownRemark.totalCount} Posts</h4>
        {data.allMarkdownRemark.edges.map(({ node }) => (
          <div key={node.id}>
            <h3>
              <Link to={`/posts${node.fields.slug}`}>
                {node.frontmatter.title}
              </Link>{" "}
              <span style={{ color: "#bbb" }}>- {node.frontmatter.date}</span>
            </h3>
            <p>{node.excerpt}</p>
          </div>
        ))}
        {/* Pagination Links */}
        <div
          style={{
            display: "flex",
            alignItems: "center",
            justifyContent: "space-around",
            maxWidth: 300,
            margin: "0 auto"
          }}
        >
          {!isFirstPage && (
            <Link to={prevPage} rel="prev">
              Prev Page
            </Link>
          )}
          {Array.from({ length: totalPages }, (_, index) => (
            <Link key={index} to={`/blog/${index === 0 ? "" : index + 1}`}>
              {index + 1}
            </Link>
          ))}
          {!isLastPage && (
            <Link to={nextPage} rel="next">
              Next Page
            </Link>
          )}
        </div>
      </div>
    </Layout>
  );
};
export const query = graphql`
  query($skip: Int!, $limit: Int!) {
    allMarkdownRemark(
      skip: $skip
      limit: $limit
      sort: { fields: [frontmatter___date], order: DESC }
    ) {
      totalCount
      edges {
        node {
          fields {
            slug
          }
          id
          frontmatter {
            title
            date
          }
          excerpt
        }
      }
    }
  }
`;

26. Formatting Dates in Gatsby / Format Function in Moment

Query using moment to format dates.

{
  allMarkdownRemark(
    limit: 2
    sort: { fields: [frontmatter___date], order: DESC }
  ) {
    totalCount
    edges {
      node {
        id
        frontmatter {
          title
          date(formatString: "dddd, MMMM Do YYYY")
        }
        excerpt
      }
    }
  }
}

Response

{
  "data": {
    "allMarkdownRemark": {
      "totalCount": 4,
      "edges": [
        {
          "node": {
            "id": "683d1527-5d01-5c36-8f4a-900b528c9473",
            "frontmatter": {
              "title": "My Fourth Markdown Post",
              "date": "Thursday, February 14th 2019"
            },
            "excerpt": "Hello, this is my fourth post made using Markdown and I'm getting practice!"
          }
        },
        {
          "node": {
            "id": "d4004257-d250-5042-8af4-618d9114a084",
            "frontmatter": {
              "title": "Third Gatsby Garb Post",
              "date": "Wednesday, February 13th 2019"
            },
            "excerpt": "Hello, this is my third post so far!"
          }
        }
      ]
    }
  }
}

Query using locale.

{
  allMarkdownRemark(
    limit: 2
    sort: { fields: [frontmatter___date], order: DESC }
  ) {
    totalCount
    edges {
      node {
        id
        frontmatter {
          title
          date(formatString: "dddd, DD MMMM YYYY", locale: "es")
        }
        excerpt
      }
    }
  }
}

Response

{
  "data": {
    "allMarkdownRemark": {
      "totalCount": 4,
      "edges": [
        {
          "node": {
            "id": "683d1527-5d01-5c36-8f4a-900b528c9473",
            "frontmatter": {
              "title": "My Fourth Markdown Post",
              "date": "jueves, 14 febrero 2019"
            },
            "excerpt": "Hello, this is my fourth post made using Markdown and I'm getting practice!"
          }
        },
        {
          "node": {
            "id": "d4004257-d250-5042-8af4-618d9114a084",
            "frontmatter": {
              "title": "Third Gatsby Garb Post",
              "date": "miércoles, 13 febrero 2019"
            },
            "excerpt": "Hello, this is my third post so far!"
          }
        }
      ]
    }
  }
}

Query using fromNow.

{
  allMarkdownRemark(
    limit: 2
    sort: { fields: [frontmatter___date], order: DESC }
  ) {
    totalCount
    edges {
      node {
        id
        frontmatter {
          title
          date(fromNow: true)
        }
        excerpt
      }
    }
  }
}

Response

{
  "data": {
    "allMarkdownRemark": {
      "totalCount": 4,
      "edges": [
        {
          "node": {
            "id": "683d1527-5d01-5c36-8f4a-900b528c9473",
            "frontmatter": {
              "title": "My Fourth Markdown Post",
              "date": "3 days ago"
            },
            "excerpt": "Hello, this is my fourth post made using Markdown and I'm getting practice!"
          }
        },
        {
          "node": {
            "id": "d4004257-d250-5042-8af4-618d9114a084",
            "frontmatter": {
              "title": "Third Gatsby Garb Post",
              "date": "4 days ago"
            },
            "excerpt": "Hello, this is my third post so far!"
          }
        }
      ]
    }
  }
}

Query using difference.

{
  allMarkdownRemark(
    limit: 2
    sort: { fields: [frontmatter___date], order: DESC }
  ) {
    totalCount
    edges {
      node {
        id
        frontmatter {
          title
          date(difference: "hours")
        }
        excerpt
      }
    }
  }
}

Response

{
  "data": {
    "allMarkdownRemark": {
      "totalCount": 4,
      "edges": [
        {
          "node": {
            "id": "683d1527-5d01-5c36-8f4a-900b528c9473",
            "frontmatter": {
              "title": "My Fourth Markdown Post",
              "date": "78"
            },
            "excerpt": "Hello, this is my fourth post made using Markdown and I'm getting practice!"
          }
        },
        {
          "node": {
            "id": "d4004257-d250-5042-8af4-618d9114a084",
            "frontmatter": {
              "title": "Third Gatsby Garb Post",
              "date": "102"
            },
            "excerpt": "Hello, this is my third post so far!"
          }
        }
      ]
    }
  }
}

27. Formatting Excerpts

Query shorting the excerpts values.

{
  allMarkdownRemark(
    limit: 2
    sort: { fields: [frontmatter___date], order: DESC }
  ) {
    totalCount
    edges {
      node {
        id
        frontmatter {
          title
          date(difference: "hours")
        }
        excerpt(pruneLength: 15)
      }
    }
  }
}

Response

{
  "data": {
    "allMarkdownRemark": {
      "totalCount": 4,
      "edges": [
        {
          "node": {
            "id": "683d1527-5d01-5c36-8f4a-900b528c9473",
            "frontmatter": {
              "title": "My Fourth Markdown Post",
              "date": "78"
            },
            "excerpt": "Hello, this is…"
          }
        },
        {
          "node": {
            "id": "d4004257-d250-5042-8af4-618d9114a084",
            "frontmatter": {
              "title": "Third Gatsby Garb Post",
              "date": "102"
            },
            "excerpt": "Hello, this is…"
          }
        }
      ]
    }
  }
}

Query using excerpts truncate.

{
  allMarkdownRemark(
    limit: 2
    sort: { fields: [frontmatter___date], order: DESC }
  ) {
    totalCount
    edges {
      node {
        id
        frontmatter {
          title
          date(difference: "hours")
        }
        excerpt(pruneLength: 15, truncate: true)
      }
    }
  }
}

Response

{
  "data": {
    "allMarkdownRemark": {
      "totalCount": 4,
      "edges": [
        {
          "node": {
            "id": "683d1527-5d01-5c36-8f4a-900b528c9473",
            "frontmatter": {
              "title": "My Fourth Markdown Post",
              "date": "78"
            },
            "excerpt": "Hello, this is…"
          }
        },
        {
          "node": {
            "id": "d4004257-d250-5042-8af4-618d9114a084",
            "frontmatter": {
              "title": "Third Gatsby Garb Post",
              "date": "102"
            },
            "excerpt": "Hello, this is…"
          }
        }
      ]
    }
  }
}

Query using excerpts format.

{
  allMarkdownRemark(
    limit: 2
    sort: { fields: [frontmatter___date], order: DESC }
  ) {
    totalCount
    edges {
      node {
        id
        frontmatter {
          title
          date(difference: "hours")
        }
        html
        excerpt(pruneLength: 15, format: HTML)
      }
    }
  }
}

Response

{
  "data": {
    "allMarkdownRemark": {
      "totalCount": 4,
      "edges": [
        {
          "node": {
            "id": "683d1527-5d01-5c36-8f4a-900b528c9473",
            "frontmatter": {
              "title": "My Fourth Markdown Post",
              "date": "78"
            },
            "html": "<p>Hello, this is my fourth post made using Markdown and I'm getting practice!</p>",
            "excerpt": "<p>Hello, this is…</p>"
          }
        },
        {
          "node": {
            "id": "d4004257-d250-5042-8af4-618d9114a084",
            "frontmatter": {
              "title": "Third Gatsby Garb Post",
              "date": "102"
            },
            "html": "<p>Hello, this is my third post so far!</p>",
            "excerpt": "<p>Hello, this is…</p>"
          }
        }
      ]
    }
  }
}

28. Time To Read for each Post / Formatting Dates on our Blog Pages

Query using timeToRead.

{
  allMarkdownRemark(
    limit: 2
    sort: { fields: [frontmatter___date], order: DESC }
  ) {
    totalCount
    edges {
      node {
        id
        frontmatter {
          title
          date(formatString: "MMM Do, YYYY")
        }
        timeToRead
        html
        excerpt(pruneLength: 15, format: HTML)
      }
    }
  }
}

Response

{
  "data": {
    "allMarkdownRemark": {
      "totalCount": 4,
      "edges": [
        {
          "node": {
            "id": "683d1527-5d01-5c36-8f4a-900b528c9473",
            "frontmatter": {
              "title": "My Fourth Markdown Post",
              "date": "78"
            },
            "timeToRead": 1,
            "html": "<p>Hello, this is my fourth post made using Markdown and I'm getting practice!</p>",
            "excerpt": "<p>Hello, this is…</p>"
          }
        },
        {
          "node": {
            "id": "d4004257-d250-5042-8af4-618d9114a084",
            "frontmatter": {
              "title": "Third Gatsby Garb Post",
              "date": "102"
            },
            "timeToRead": 1,
            "html": "<p>Hello, this is my third post so far!</p>",
            "excerpt": "<p>Hello, this is…</p>"
          }
        }
      ]
    }
  }
}
  • Modify the templates\blog-template.js document to format the date.

templates\blog-template.js

import React from "react";
import { graphql, Link } from "gatsby";

import Layout from "../components/layout";

export default ({ data, pageContext }) => {
  const { currentPage, isFirstPage, isLastPage, totalPages } = pageContext;
  const nextPage = `/blog/${String(currentPage + 1)}`;
  const prevPage =
    currentPage - 1 === 1 ? "/blog" : `/blog/${String(currentPage - 1)}`;
  return (
    <Layout>
      <div>
        <h1 style={{ display: "inlineBlock", borderBottom: "1px solid" }}>
          Gatsby Garb Blog
        </h1>
        <h4>{data.allMarkdownRemark.totalCount} Posts</h4>
        {data.allMarkdownRemark.edges.map(({ node }) => (
          <div key={node.id}>
            <h3>
              <Link to={`/posts${node.fields.slug}`}>
                {node.frontmatter.title}
              </Link>{" "}
              <span style={{ color: "#bbb" }}>- {node.frontmatter.date}</span>
            </h3>
            <p>{node.excerpt}</p>
          </div>
        ))}
        {/* Pagination Links */}
        <div
          style={{
            display: "flex",
            alignItems: "center",
            justifyContent: "space-around",
            maxWidth: 300,
            margin: "0 auto"
          }}
        >
          {!isFirstPage && (
            <Link to={prevPage} rel="prev">
              Prev Page
            </Link>
          )}
          {Array.from({ length: totalPages }, (_, index) => (
            <Link key={index} to={`/blog/${index === 0 ? "" : index + 1}`}>
              {index + 1}
            </Link>
          ))}
          {!isLastPage && (
            <Link to={nextPage} rel="next">
              Next Page
            </Link>
          )}
        </div>
      </div>
    </Layout>
  );
};
export const query = graphql`
  query($skip: Int!, $limit: Int!) {
    allMarkdownRemark(
      skip: $skip
      limit: $limit
      sort: { fields: [frontmatter___date], order: DESC }
    ) {
      totalCount
      edges {
        node {
          fields {
            slug
          }
          id
          frontmatter {
            title
            date(formatString: "MMM Do, YYYY")
          }
          excerpt
        }
      }
    }
  }
`;
  • Modify the templates\post-template.js document to include the time to read (timeToRead) field.

templates\post-template.js

import React from "react";
import { graphql } from "gatsby";

import Layout from "../components/layout";

const PostTemplate = ({ data: post }) => (
  <Layout>
    <h1>{post.markdownRemark.frontmatter.title}</h1>
    <h4>
      {post.markdownRemark.timeToRead}
      {post.markdownRemark.timeToRead > 1 ? "minutes" : "minute"}
    </h4>
    <div dangerouslySetInnerHTML={{ __html: post.markdownRemark.html }} />
  </Layout>
);

// This will be rendered and sent to the `data` parameter
export const query = graphql`
  query($slug: String) {
    markdownRemark(fields: { slug: { eq: $slug } }) {
      html
      timeToRead
      frontmatter {
        title
      }
    }
  }
`;

export default PostTemplate;

12. Images in Gatsby / Amazing Responsive Images with Gatsby Image

  • Modify the components\header.js document to use an image for the header.

components\header.js

import { Link } from "gatsby";
import PropTypes from "prop-types";
import React from "react";

import gatsbyLogo from "../images/gatsby-icon.png";

const Header = ({ siteTitle }) => (
  <div
    style={{
      background: `rebeccapurple`,
      marginBottom: `1.45rem`
    }}
  >
    <div
      style={{
        margin: `0 auto`,
        maxWidth: 960,
        padding: `1.45rem 1.0875rem`
      }}
    >
      {/* Title / Logo */}
      <span
        style={{
          display: "flex",
          alignItems: "center"
        }}
      >
        <img
          src={gatsbyLogo}
          alt="Gastby Garb Logo"
          style={{
            borderRadius: "50%",
            border: "3px solid orange",
            margin: "0 5px",
            width: `3rem`
          }}
        />
        <h1 style={{ margin: 0 }}>
          <Link
            to="/"
            style={{
              color: `white`,
              textDecoration: `none`
            }}
          >
            {siteTitle}
          </Link>
        </h1>
      </span>
    </div>
  </div>
);

Header.propTypes = {
  siteTitle: PropTypes.string
};

Header.defaultProps = {
  siteTitle: ``
};

export default Header;

  • If we don't want webpack to manage content, we should create an additional folder called static and put there the images, css files, or other document that we want.

30. What is Gatsby Image / Why Use Gatsby Image

  • gatsby-image is used to manage the images properly to be rendered for each device the best way it can. It is used along with the StaticQuery component.

  • It can be seen how it works having a look at the \src\components\image.js document.#

\src\components\image.js

import React from "react";
import { StaticQuery, graphql } from "gatsby";
import Img from "gatsby-image";

/*
 * This component is built using `gatsby-image` to automatically serve optimized
 * images with lazy loading and reduced file sizes. The image is loaded using a
 * `StaticQuery`, which allows us to load the image from directly within this
 * component, rather than having to pass the image data down from pages.
 *
 * For more information, see the docs:
 * - `gatsby-image`: https://gatsby.app/gatsby-image
 * - `StaticQuery`: https://gatsby.app/staticquery
 */

const Image = () => (
  <StaticQuery
    query={graphql`
      query {
        placeholderImage: file(relativePath: { eq: "gatsby-astronaut.png" }) {
          childImageSharp {
            fluid(maxWidth: 300) {
              ...GatsbyImageSharpFluid
            }
          }
        }
      }
    `}
    render={data => <Img fluid={data.placeholderImage.childImageSharp.fluid} />}
  />
);
export default Image;

31. Querying for Images in GraphiQL / gatsby-transformer-sharp Plugin in Action

  • In order to use gatsby-image we need to query for the image useing the placeholderImage.

Query using file and childImageSharp.

{
  file(relativePath: { eq: "gatsby-icon.png" }) {
    childImageSharp {
      fluid {
        src
        srcSet
        srcSetWebp
        aspectRatio
        base64
        tracedSVG
        sizes
        originalImg
        originalName
      }
    }
  }
}

Response

{
  "data": {
    "file": {
      "childImageSharp": {
        "fluid": {
          "src": "/static/4a9773549091c227cd2eb82ccd9c5e3a/4d7ce/gatsby-icon.png",
          "srcSet": "/static/4a9773549091c227cd2eb82ccd9c5e3a/e22ac/gatsby-icon.png 200w,\n/static/4a9773549091c227cd2eb82ccd9c5e3a/12519/gatsby-icon.png 400w,\n/static/4a9773549091c227cd2eb82ccd9c5e3a/4d7ce/gatsby-icon.png 512w",
          "srcSetWebp": "/static/4a9773549091c227cd2eb82ccd9c5e3a/1da1b/gatsby-icon.webp 200w,\n/static/4a9773549091c227cd2eb82ccd9c5e3a/8f67f/gatsby-icon.webp 400w,\n/static/4a9773549091c227cd2eb82ccd9c5e3a/500e5/gatsby-icon.webp 512w",
          "aspectRatio": 1,
          "base64": "",
          "tracedSVG": "data:image/svg+xml,%3csvg xmlns='http://www.w3.org/2000/svg' width='400' height='400' viewBox='0 0 400 400' version='1'%3e%3cpath d='M173 1A201 201 0 0 0 23 294a200 200 0 0 0 377-78c1 1 1-7 1-17l-1-15-2-9A201 201 0 0 0 173 1m5 43A158 158 0 0 0 47 164c0 1 42 44 94 95l94 94 8-2a158 158 0 0 0 114-148v-3H257v28h34l34 1a127 127 0 0 1-72 87l-10 5-82-82-82-83a128 128 0 0 1 221-36l5 5 11-8 10-9c2-2-25-29-37-38-18-11-37-20-57-24-11-2-44-4-54-2M43 212a156 156 0 0 0 148 144h6l-76-76-77-77-1 9' fill='lightgray' fill-rule='evenodd'/%3e%3c/svg%3e",
          "sizes": "(max-width: 512px) 100vw, 512px",
          "originalImg": "/static/4a9773549091c227cd2eb82ccd9c5e3a/4d7ce/gatsby-icon.png",
          "originalName": "gatsby-icon.png"
        }
      }
    }
  }
}

13. Advanced Image Concepts in Gatsby / GraphQL Fragments

32. GraphQL Fragments for Easier Image Queries / Displaying Images with Gatsby Image

  • Fragments are not going to be identified by GraphiQL although we can use them inside the app.

Query using file and GatsbyImageSharpFluid.

{
  file(relativePath: { eq: "gatsby-icon.png" }) {
    childImageSharp {
      fluid(maxWidth: 590) {
        ...GatsbyImageSharpFluid
      }
    }
  }
}

Response

{
  "errors": [
    {
      "message": "Unknown fragment \"GatsbyImageSharpFluid\".",
      "locations": [
        {
          "line": 7,
          "column": 11
        }
      ]
    }
  ]
}
  • We can modify the components/image.js document to use it, we can even execute two queries at the same time.

components/image.js

import React from "react";
import { StaticQuery, graphql } from "gatsby";
import Img from "gatsby-image";

/*
 * This component is built using `gatsby-image` to automatically serve optimized
 * images with lazy loading and reduced file sizes. The image is loaded using a
 * `StaticQuery`, which allows us to load the image from directly within this
 * component, rather than having to pass the image data down from pages.
 *
 * For more information, see the docs:
 * - `gatsby-image`: https://gatsby.app/gatsby-image
 * - `StaticQuery`: https://gatsby.app/staticquery
 */

const Image = () => (
  <StaticQuery
    query={graphql`
      query {
        placeholderImage: file(relativePath: { eq: "gatsby-astronaut.png" }) {
          childImageSharp {
            fluid(maxWidth: 300) {
              ...GatsbyImageSharpFluid
            }
          }
        }
        iconImage: file(relativePath: { eq: "gatsby-icon.png" }) {
          childImageSharp {
            fluid(maxWidth: 590) {
              ...GatsbyImageSharpFluid
            }
          }
        }
      }
    `}
    render={data => (
      <>
        <Img fluid={data.placeholderImage.childImageSharp.fluid} />
        <Img fluid={data.iconImage.childImageSharp.fluid} />
      </>
    )}
  />
);
export default Image;

33. Key Details about Images Served with Gatsby Image

  • The images served by Gatsby as Fluid images, meaning they are responsive.

  • The size of the logo is 300px x 300px.

  • The reason is becuase it is set in pages\index.js

    pages\index.js

import React from "react";
import { Link } from "gatsby";

import Layout from "../components/layout";
import Image from "../components/image";
import SEO from "../components/seo";

const IndexPage = () => (
  <Layout>
    <SEO title="Home" keywords={[`gatsby`, `application`, `react`]} />
    <h1>Hi people</h1>
    <p>Welcome to your new Gatsby site.</p>
    <p>Now go build something great.</p>
    <div style={{ maxWidth: `300px`, marginBottom: `1.45rem` }}>
      <Image />
    </div>
    <Link to="/page-2/">Go to page 2</Link>
  </Layout>
);

export default IndexPage;
  • We can change it

pages\index.js

  • We revert to the original size.

pages\index.js

import React from "react";
import { Link } from "gatsby";

import Layout from "../components/layout";
import Image from "../components/image";
import SEO from "../components/seo";

const IndexPage = () => (
  <Layout>
    <SEO title="Home" keywords={[`gatsby`, `application`, `react`]} />
    <h1>Hi people</h1>
    <p>Welcome to your new Gatsby site.</p>
    <p>Now go build something great.</p>
    <div style={{ maxWidth: `300px`, marginBottom: `1.45rem` }}>
      <Image />
    </div>
    <Link to="/page-2/">Go to page 2</Link>
  </Layout>
);

export default IndexPage;
  • We can see there are some other attributes in the html related to the Fluid images
<picture>
  <source
    srcset="
      /static/4a9773549091c227cd2eb82ccd9c5e3a/57ddb/gatsby-icon.png 148w,
      /static/4a9773549091c227cd2eb82ccd9c5e3a/131ad/gatsby-icon.png 295w,
      /static/4a9773549091c227cd2eb82ccd9c5e3a/a19f4/gatsby-icon.png 512w
    "
    sizes="(max-width: 512px) 100vw, 512px"
  />
  <img
    alt=""
    src="/static/4a9773549091c227cd2eb82ccd9c5e3a/a19f4/gatsby-icon.png"
    style="position: absolute; top: 0px; left: 0px; width: 100%; height: 100%; object-fit: cover; object-position: center center; opacity: 1; transition: opacity 0.5s ease 0s;"
  />
</picture>

34. Embedding Images in Markdown Files with gatsby-remark-images

  • We can include a video inside a MarkDown file using the iframe tag,

src\pages\post-two.md

---
title: "The Second Gatsby Garb Post"
date: "2019-02-10"
---

You're currently reading the second post made using Markdown!

<iframe src="https://youtube.com/embed/tgbNymZ7vqY"></iframe>

  • But in order to include images we should install the new gatsby-remark-images
Juan.Pablo.Perez@RIMDUB-0232 MINGW64 /c/Work/Training/Pre/Gatsby/gatsby-garb (master)
$ npm i gatsby-remark-images
npm WARN optional SKIPPING OPTIONAL DEPENDENCY: fsevents@1.2.4 (node_modules\fsevents):
npm WARN notsup SKIPPING OPTIONAL DEPENDENCY: Unsupported platform for fsevents@1.2.4: wanted {"os":"darwin","arch":"any"} (current: {"os":"win32","arch":"x64"})

+ gatsby-remark-images@3.0.4
added 3 packages from 3 contributors and audited 25555 packages in 52.557s
found 3 low severity vulnerabilities
  run `npm audit fix` to fix them, or `npm audit` for details
  • It have to be used in conjuction with the gatsby-transformer-remark plugin.

  • We need to change the gatsby-config.js docuemnt to change how gatsby-transformer-remark is et up.

gatsby-config.js

module.exports = {
  siteMetadata: {
    title: `Gatsby Garb`,
    description: `Kick off your next, great Gatsby project with this default starter. This barebones starter ships with the main Gatsby configuration files you might need.`,
    author: `Juan Pablo Perez`,
    createdAt: 2019
  },
  plugins: [
    `gatsby-plugin-react-helmet`,
    {
      resolve: `gatsby-source-filesystem`,
      options: {
        name: `images`,
        path: `${__dirname}/src/images`
      }
    },
    {
      resolve: "gatsby-transformer-remark",
      options: {
        plugins: [
          {
            resolve: "gatsby-remark-images"
          }
        ]
      }
    },
    {
      resolve: `gatsby-source-filesystem`,
      options: {
        name: `posts`,
        path: `${__dirname}/src/posts`
      }
    },
    `gatsby-transformer-sharp`,
    `gatsby-plugin-sharp`,
    {
      resolve: `gatsby-plugin-manifest`,
      options: {
        name: `gatsby-starter-default`,
        short_name: `starter`,
        start_url: `/`,
        background_color: `#663399`,
        theme_color: `#663399`,
        display: `minimal-ui`,
        icon: `src/images/gatsby-icon.png` // This path is relative to the root of the site.
      }
    }
    // this (optional) plugin enables Progressive Web App + Offline functionality
    // To learn more, visit: https://gatsby.app/offline
    // 'gatsby-plugin-offline',
  ]
};
  • We have to put the images we want to use in the same directory where the Markdown file is.

  • We can include the image in the post:

src\pages\post-two.md

---
title: "The Second Gatsby Garb Post"
date: "2019-02-10"
---

You're currently reading the second post made using Markdown!

![Tranquil Beach](tranquil-beach.jpg)

  • The image is converted to a Fluid image.

<a
  class="gatsby-resp-image-link"
  href="/static/f23bf22db2e727effaacecfeaf9be5de/db7d6/tranquil-beach.jpg"
  style="display: block"
  target="_blank"
  rel="noopener"
>
  <span
    class="gatsby-resp-image-wrapper"
    style="position: relative; display: block;  max-width: 650px; margin-left: auto; margin-right: auto;"
  >
    <span
      class="gatsby-resp-image-background-image"
      style='padding-bottom: 75%; position: relative; bottom: 0px; left: 0px; background-image: url(""); background-size: cover; display: block; transition: opacity 0.5s ease 0.5s; opacity: 0;'
    ></span>
    <img
      class="gatsby-resp-image-image"
      style="width: 100%; height: 100%; margin: 0px; vertical-align: middle; position: absolute; top: 0px; left: 0px; box-shadow: white 0px 0px 0px 400px inset; opacity: 1; transition: opacity 0.5s ease 0s;"
      alt="Tranquil Beach"
      title=""
      src="/static/f23bf22db2e727effaacecfeaf9be5de/b80fa/tranquil-beach.jpg"
      srcset="
        /static/f23bf22db2e727effaacecfeaf9be5de/cf410/tranquil-beach.jpg  163w,
        /static/f23bf22db2e727effaacecfeaf9be5de/62f2a/tranquil-beach.jpg  325w,
        /static/f23bf22db2e727effaacecfeaf9be5de/b80fa/tranquil-beach.jpg  650w,
        /static/f23bf22db2e727effaacecfeaf9be5de/08cb4/tranquil-beach.jpg  975w,
        /static/f23bf22db2e727effaacecfeaf9be5de/db7d6/tranquil-beach.jpg 1024w
      "
      sizes="(max-width: 650px) 100vw, 650px"
    />
  </span>
</a>

14. Contentful as our Headless CMS / Creating and Managing Products with Contentful

35. Intro to Headless CMS and Contentful / Creating Content Model for our Products

  • Create a new project.

  • Put the Gatsby Garb name.

  • Put Product for Name

  • Put name for Name and Field ID and select (*)Short text, exact search

  • Put price for Name and Field ID and select (*)Decimal

  • Put image for Name and Field ID and select (*)One file

  • Put description for Name and Field ID and select (*)Short text, exact search

  • Put slut for Name and Field ID and select (*)Short text, exact search

  • Put private for Name and Field ID

IntroToHeadlessCmsAndContentful47

36. Managing Environment Variables within Gatsby / Fetching Products from Contentful

  • We need to install the gatsby-source-contentful plugin and the dotenv package.
Juan.Pablo.Perez@RIMDUB-0232 MINGW64 /c/Work/Training/Pre/Gatsby/gatsby-garb (master)
$ npm i gatsby-source-contentful dotenv
npm WARN optional SKIPPING OPTIONAL DEPENDENCY: fsevents@1.2.4 (node_modules\fsevents):
npm WARN notsup SKIPPING OPTIONAL DEPENDENCY: Unsupported platform for fsevents@1.2.4: wanted {"os":"darwin","arch":"any"} (current: {"os":"win32","arch":"x64"})

+ dotenv@6.2.0
+ gatsby-source-contentful@2.0.30
added 43 packages from 28 contributors, updated 1 package and audited 29295 packages in 82.356s
found 3 low severity vulnerabilities
  run `npm audit fix` to fix them, or `npm audit` for details
  • We need to create a new .env document where we have to put the Space ID and Content Delivery API - access token values copied from the ContentFul API Page.

.env

CONTENTFUL_SPACE_ID=..
CONTENTFUL_ACCESS_TOKEN=..
  • We need to modify the gatsby-config.js document to enable the new plugin using the new environment variables.

gatsby-config.js

const dotenv = require("dotenv");

if (process.env.NODE_ENV !== "production") {
  dotenv.config();
}

module.exports = {
  siteMetadata: {
    title: `Gatsby Garb`,
    description: `Kick off your next, great Gatsby project with this default starter. This barebones starter ships with the main Gatsby configuration files you might need.`,
    author: `Juan Pablo Perez`,
    createdAt: 2019
  },
  plugins: [
    `gatsby-plugin-react-helmet`,
    {
      resolve: `gatsby-source-filesystem`,
      options: {
        name: `images`,
        path: `${__dirname}/src/images`
      }
    },
    {
      resolve: "gatsby-transformer-remark",
      options: {
        plugins: [
          {
            resolve: "gatsby-remark-images"
          }
        ]
      }
    },
    {
      resolve: `gatsby-source-filesystem`,
      options: {
        name: `posts`,
        path: `${__dirname}/src/posts`
      }
    },
    `gatsby-transformer-sharp`,
    `gatsby-plugin-sharp`,
    {
      resolve: `gatsby-plugin-manifest`,
      options: {
        name: `gatsby-starter-default`,
        short_name: `starter`,
        start_url: `/`,
        background_color: `#663399`,
        theme_color: `#663399`,
        display: `minimal-ui`,
        icon: `src/images/gatsby-icon.png` // This path is relative to the root of the site.
      }
    },
    {
      resolve: "gatsby-source-contentful",
      options: {
        spaceId: process.env.CONTENTFUL_SPACE_ID,
        accessToken: process.env.CONTENTFUL_ACCESS_TOKEN
      }
    }
    // this (optional) plugin enables Progressive Web App + Offline functionality
    // To learn more, visit: https://gatsby.app/offline
    // 'gatsby-plugin-offline',
  ]
};

request

{
  allContentfulProduct {
    totalCount
    edges {
      node {
        name
        price
        description
        slug
        image {
          fluid {
            src
          }
        }
      }
    }
  }
}

response

{
  "data": {
    "allContentfulProduct": {
      "totalCount": 1,
      "edges": [
        {
          "node": {
            "name": "Rayban Sunglasses",
            "price": 89.95,
            "description": "Really cool Roy Orbison's style Sun Glasses",
            "slug": "rayban-sunglasses",
            "image": {
              "fluid": {
                "src": "//images.ctfassets.net/q7qvrab0p640/4UguCYRX2jXlUjnfT8QIXi/332a3bd43f422ec3870241721bd86087/Rayban_sunglasses.jpg?w=800&q=50"
              }
            }
          }
        }
      ]
    }
  }
}

37. Creating our Individual Product Pages / Displaying Contentful Images

  • We need to create a new query that is going to be used for the creation of the Product pages on the fly.

  • For one product:

request

{
  contentfulProduct(slug: { eq: "rayban-sunglasses" }) {
    name
    price
    description
    slug
    image {
      fluid {
        src
      }
    }
  }
}

response

{
  "data": {
    "contentfulProduct": {
      "name": "Rayban Sunglasses",
      "price": 89.95,
      "description": "Really cool Roy Orbison's style Sun Glasses",
      "slug": "rayban-sunglasses",
      "image": {
        "fluid": {
          "src": "//images.ctfassets.net/q7qvrab0p640/4UguCYRX2jXlUjnfT8QIXi/332a3bd43f422ec3870241721bd86087/Rayban_sunglasses.jpg?w=800&q=50"
        }
      }
    }
  }
}
  • We need to make it variable:

Request

query($slug: String) {
  contentfulProduct(slug: { eq: $slug }) {
    name
    price
    description
    createdAt(formatString: "MMM Do, YYYY, h:mm:ss a")
    image {
      fluid {
        src
      }
    }
  }
}

Quey variables

{
  "slug": "rayban-sunglasses"
}

Response

{
  "data": {
    "contentfulProduct": {
      "name": "Rayban Sunglasses",
      "price": 89.95,
      "description": "Really cool Roy Orbison's style Sun Glasses",
      "createdAt": "Feb 20th, 2019, 5:44:24 am",
      "image": {
        "fluid": {
          "src": "//images.ctfassets.net/q7qvrab0p640/4UguCYRX2jXlUjnfT8QIXi/332a3bd43f422ec3870241721bd86087/Rayban_sunglasses.jpg?w=800&q=50"
        }
      }
    }
  }
}
  • We need to create the new templates\product-template.js document used to generate the product pages on the fly.

templates\product-template.js

import React from "react";
import { graphql } from "gatsby";
import Img from "gatsby-image";

import Layout from "../components/layout";

const ProductTemplate = ({ data: { contentfulProduct } }) => (
  <Layout>
    <div
      style={{
        marginLeft: "0 auto",
        width: "100%",
        textAlign: "center"
      }}
    >
      {/* Product Info */}
      <h2>
        {contentfulProduct.name} -{" "}
        <span style={{ color: "#ccc" }}>
          Added on {contentfulProduct.createdAt}
        </span>
      </h2>
      <p>{contentfulProduct.description}</p>
    </div>
    <Img
      style={{ margin: "0 auto", maxWidth: 600 }}
      fluid={contentfulProduct.image.fluid}
    />
  </Layout>
);

export const query = graphql`
  query($slug: String!) {
    contentfulProduct(slug: { eq: $slug }) {
      slug
      name
      price
      description
      createdAt(formatString: "MMMM Do, YYYY, h:mm:ss a")
      image {
        fluid(maxWidth: 800) {
          ...GatsbyContentfulFluid
        }
      }
    }
  }
`;

export default ProductTemplate;
  • Modify the gatsby-node.js document to include the content product query and the use of the new ProductTemplate to generate the page for each product..

gatsby-node.js

const path = require("path");
const { createFilePath } = require("gatsby-source-filesystem");

const PostTemplate = path.resolve("./src/templates/post-template.js");
const BlogTemplate = path.resolve("./src/templates/blog-template.js");
const ProductTemplate = path.resolve("./src/templates/product-template.js");

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

  if (node.internal.type === "MarkdownRemark") {
    const slug = createFilePath({ node, getNode, basepath: "posts" });
    createNodeField({
      node,
      name: "slug",
      value: slug
    });
  }
};

exports.createPages = async ({ graphql, actions }) => {
  const { createPage } = actions;
  const result = await graphql(`
    {
      allMarkdownRemark(limit: 1000) {
        edges {
          node {
            fields {
              slug
            }
          }
        }
      }

      allContentfulProduct {
        edges {
          node {
            slug
          }
        }
      }
    }
  `);
  const posts = result.data.allMarkdownRemark.edges;

  posts.forEach(({ node: post }) => {
    createPage({
      path: `posts${post.fields.slug}`,
      component: PostTemplate,
      context: {
        slug: post.fields.slug
      }
    });
  });

  const postsPerPage = 2;
  const totalPages = Math.ceil(posts.length / postsPerPage);

  Array.from({ length: totalPages }).forEach((_, index) => {
    const currentPage = index + 1;
    const isFirstPage = index === 0;
    const isLastPage = currentPage === totalPages;
    createPage({
      path: isFirstPage ? "/blog/" : `blog/${currentPage}`,
      component: BlogTemplate,
      context: {
        limit: postsPerPage,
        skip: index * postsPerPage,
        isFirstPage,
        isLastPage,
        currentPage,
        totalPages
      }
    });
  });

  const products = result.data.allContentfulProduct.edges;
  products.forEach(({ node: product }) => {
    createPage({
      path: `/products/${product.slug}`,
      component: ProductTemplate,
      context: {
        slug: product.slug
      }
    });
  });
};
  • Go to an invalid path to check if the link to the products is created on the fly.

38. Querying / Previewing all Products on Products Page

  • Create the new pages/products.js page where we can see all the products.

pages/products.js

import React from "react";
import { graphql, Link } from "gatsby";
import Img from "gatsby-image";

import Layout from "../components/layout";

const Products = ({ data: { allContentfulProduct } }) => (
  <Layout>
    <div>
      {/* Products List */}
      {allContentfulProduct.edges.map(({ node: product }) => (
        <div key={product.id}>
          <h2>Garb Products</h2>
          <Link to={`/products/${product.slug}`}>
            <h3>{product.name}</h3>
          </Link>
          <Img style={{ maxWidth: 400 }} fluid={product.image.fluid} />
        </div>
      ))}
    </div>
  </Layout>
);

export const query = graphql`
  {
    allContentfulProduct {
      edges {
        node {
          id
          slug
          name
          image {
            fluid(maxWidth: 400) {
              ...GatsbyContentfulFluid
            }
          }
        }
      }
    }
  }
`;

export default Products;

If we use GatsbyContentfulFluid_tracedSVG there is an error when compiling.


i 「wdm」: Compiling...
error UNHANDLED EXCEPTION


  TypeError: Cannot read property 'bitmap' of undefined

  - Potrace.js:1000 Potrace._processLoadedImage
    [gatsby-garb]/[potrace]/lib/Potrace.js:1000:35

  - Potrace.js:1046 Jimp.<anonymous>
    [gatsby-garb]/[potrace]/lib/Potrace.js:1046:14

  - index.js:85 Jimp.throwError
    [gatsby-garb]/[jimp]/index.js:85:44

  - index.js:201 ReadFileContext.callback
    [gatsby-garb]/[jimp]/index.js:201:44


npm ERR! code ELIFECYCLE
npm ERR! errno 1
npm ERR! gatsby-starter-default@0.1.0 develop: `gatsby develop -p 8001`
npm ERR! Exit status 1
npm ERR!
npm ERR! Failed at the gatsby-starter-default@0.1.0 develop script.
npm ERR! This is probably not a problem with npm. There is likely additional logging output above.

npm ERR! A complete log of this run can be found in:
npm ERR!     C:\Users\juan.pablo.perez\AppData\Roaming\npm-cache\_logs\2019-02-21T07_30_56_720Z-debug.log
npm ERR! code ELIFECYCLE
npm ERR! errno 1
npm ERR! gatsby-starter-default@0.1.0 start: `npm run develop`
npm ERR! Exit status 1
npm ERR!
npm ERR! Failed at the gatsby-starter-default@0.1.0 start script.
npm ERR! This is probably not a problem with npm. There is likely additional logging output above.

npm ERR! A complete log of this run can be found in:
npm ERR!     C:\Users\juan.pablo.perez\AppData\Roaming\npm-cache\_logs\2019-02-21T07_30_56_823Z-debug.log

15. Creating a Shopping Cart / Checkout Functionality with Snipcart

39. Add Ability to Purchase Products with Snipcart

  • We need to modify the pages/product.js and template/product-template.js documents to include the price of the product.

pages/product.js

import React from "react";
import { graphql, Link } from "gatsby";
import Img from "gatsby-image";

import Layout from "../components/layout";

const Products = ({ data: { allContentfulProduct } }) => (
  <Layout>
    <div>
      {/* Products List */}
      {allContentfulProduct.edges.map(({ node: product }) => (
        <div key={product.id}>
          <h2>Garb Products</h2>
          <Link
            to={`/products/${product.slug}`}
            style={{ textDecoration: "none", color: "#551a8b" }}
          >
            <h3>
              {product.name} ·{" "}
              <span
                style={{
                  fontSize: "1.2rem",
                  fontWeight: 300,
                  color: "#f60"
                }}
              >
                ${product.price}
              </span>
            </h3>
          </Link>
          <Img style={{ maxWidth: 400 }} fluid={product.image.fluid} />
        </div>
      ))}
    </div>
  </Layout>
);

export const query = graphql`
  {
    allContentfulProduct {
      edges {
        node {
          id
          slug
          name
          price
          image {
            fluid(maxWidth: 400) {
              ...GatsbyContentfulFluid
            }
          }
        }
      }
    }
  }
`;

export default Products;

template/product-template.js

import React from "react";
import { graphql } from "gatsby";
import Img from "gatsby-image";

import Layout from "../components/layout";

const ProductTemplate = ({ data: { contentfulProduct } }) => (
  <Layout>
    <div
      style={{
        marginLeft: "0 auto",
        width: "100%",
        textAlign: "center"
      }}
    >
      {/* Product Info */}
      <h2>
        {contentfulProduct.name} -{" "}
        <span style={{ color: "#ccc" }}>
          Added on {contentfulProduct.createdAt}
        </span>
      </h2>
      <h4>${contentfulProduct.price}</h4>
      <p>{contentfulProduct.description}</p>
    </div>
    <Img
      style={{ margin: "0 auto", maxWidth: 600 }}
      fluid={contentfulProduct.image.fluid}
    />
  </Layout>
);

export const query = graphql`
  query($slug: String!) {
    contentfulProduct(slug: { eq: $slug }) {
      slug
      name
      price
      description
      createdAt(formatString: "MMMM Do, YYYY, h:mm:ss a")
      image {
        fluid(maxWidth: 800) {
          ...GatsbyContentfulFluid
        }
      }
    }
  }
`;

export default ProductTemplate;

  • Copy the API key to the .env document.

.env

CONTENTFUL_SPACE_ID=q7XXXXXXp640
CONTENTFUL_ACCESS_TOKEN=caXXXXXXd2c940334XXXXXXf6ce75XXXXXX72a019XXXXXXa3fcb9XXXXXX05b73
SNIPCART_API_KEY=NzBlYjVXXXXXXjg2Yy0XXXXXXTg2YjQXXXXXXDhjZjcXXXXXXjM2ODYXXXXXXzYzMTA2ODU2
  • We need to install the gatsby-pluig-snipcart
Juan.Pablo.Perez@RIMDUB-0232 MINGW64 /c/Work/Training/Pre/Gatsby/gatsby-garb (master)
$ npm i gatsby-plugin-snipcart
npm WARN optional SKIPPING OPTIONAL DEPENDENCY: fsevents@1.2.4 (node_modules\fsevents):
npm WARN notsup SKIPPING OPTIONAL DEPENDENCY: Unsupported platform for fsevents@1.2.4: wanted {"os":"darwin","arch":"any"} (current: {"os":"win32","arch":"x64"})

+ gatsby-plugin-snipcart@1.0.6
added 1 package from 1 contributor and audited 29296 packages in 47.555s
found 3 low severity vulnerabilities
  run `npm audit fix` to fix them, or `npm audit` for details
  • Modify the gatsby-config.js document to include the new plugin

gatsby-config.js

const dotenv = require("dotenv");

if (process.env.NODE_ENV !== "production") {
  dotenv.config();
}

module.exports = {
  siteMetadata: {
    title: `Gatsby Garb`,
    description: `Kick off your next, great Gatsby project with this default starter. This barebones starter ships with the main Gatsby configuration files you might need.`,
    author: `Juan Pablo Perez`,
    createdAt: 2019
  },
  plugins: [
    `gatsby-plugin-react-helmet`,
    {
      resolve: `gatsby-source-filesystem`,
      options: {
        name: `images`,
        path: `${__dirname}/src/images`
      }
    },
    {
      resolve: "gatsby-transformer-remark",
      options: {
        plugins: [
          {
            resolve: "gatsby-remark-images"
          }
        ]
      }
    },
    {
      resolve: `gatsby-source-filesystem`,
      options: {
        name: `posts`,
        path: `${__dirname}/src/posts`
      }
    },
    `gatsby-transformer-sharp`,
    `gatsby-plugin-sharp`,
    {
      resolve: `gatsby-plugin-manifest`,
      options: {
        name: `gatsby-starter-default`,
        short_name: `starter`,
        start_url: `/`,
        background_color: `#663399`,
        theme_color: `#663399`,
        display: `minimal-ui`,
        icon: `src/images/gatsby-icon.png` // This path is relative to the root of the site.
      }
    },
    {
      resolve: "gatsby-source-contentful",
      options: {
        spaceId: process.env.CONTENTFUL_SPACE_ID,
        accessToken: process.env.CONTENTFUL_ACCESS_TOKEN
      }
    },
    {
      resolve: "gatsby-plugin-snipcart",
      options: {
        apiKey: process.env.SNIPCART_API_KEY,
        autopop: true
      }
    }
    // this (optional) plugin enables Progressive Web App + Offline functionality
    // To learn more, visit: https://gatsby.app/offline
    // 'gatsby-plugin-offline',
  ]
};
  • We need to modify the templates\product-template.js document to include a button to add the product to the snipcart cart.

templates\product-template.js

import React from "react";
import { graphql } from "gatsby";
import Img from "gatsby-image";

import Layout from "../components/layout";

const ProductTemplate = ({ data: { contentfulProduct }, location }) => (
  <Layout>
    <div
      style={{
        marginLeft: "0 auto",
        width: "100%",
        textAlign: "center"
      }}
    >
      {/* Product Info */}
      <h2>
        {contentfulProduct.name} -{" "}
        <span style={{ color: "#ccc" }}>
          Added on {contentfulProduct.createdAt}
        </span>
      </h2>
      <h4>${contentfulProduct.price}</h4>
      <p>{contentfulProduct.description}</p>
      <button
        className="snipcart-add-item"
        data-item-id={contentfulProduct.slug}
        data-item-price={contentfulProduct.price}
        data-item-image={contentfulProduct.image.file.url}
        data-item-name={contentfulProduct.name}
        data-item-url={location.pathname}
      >
        Add to Cart
      </button>
    </div>
    <Img
      style={{ margin: "0 auto", maxWidth: 600 }}
      fluid={contentfulProduct.image.fluid}
    />
  </Layout>
);

export const query = graphql`
  query($slug: String!) {
    contentfulProduct(slug: { eq: $slug }) {
      slug
      name
      price
      description
      createdAt(formatString: "MMMM Do, YYYY, h:mm:ss a")
      image {
        fluid(maxWidth: 800) {
          ...GatsbyContentfulFluid
        }
        file {
          url
        }
      }
    }
  }
`;

export default ProductTemplate;

40. Display Shopping Cart Summary in Header

  • Modify the templates\product-template.js document to better style the Add to cart button..

templates\product-template.js

import React from "react";
import { graphql } from "gatsby";
import Img from "gatsby-image";

import Layout from "../components/layout";

const ProductTemplate = ({ data: { contentfulProduct }, location }) => (
  <Layout>
    <div
      style={{
        marginLeft: "0 auto",
        width: "100%",
        textAlign: "center"
      }}
    >
      {/* Product Info */}
      <h2>
        {contentfulProduct.name} -{" "}
        <span style={{ color: "#ccc" }}>
          Added on {contentfulProduct.createdAt}
        </span>
      </h2>
      <h4>${contentfulProduct.price}</h4>
      <p>{contentfulProduct.description}</p>
      <button
        style={{
          background: "darkorange",
          color: "white",
          padding: "0.3em",
          borderRadius: "5px",
          cursor: "pointer"
        }}
        className="snipcart-add-item"
        data-item-id={contentfulProduct.slug}
        data-item-price={contentfulProduct.price}
        data-item-image={contentfulProduct.image.file.url}
        data-item-name={contentfulProduct.name}
        data-item-url={location.pathname}
      >
        Add to Cart
      </button>
    </div>
    <Img
      style={{ margin: "0 auto", maxWidth: 600 }}
      fluid={contentfulProduct.image.fluid}
    />
  </Layout>
);

export const query = graphql`
  query($slug: String!) {
    contentfulProduct(slug: { eq: $slug }) {
      slug
      name
      price
      description
      createdAt(formatString: "MMMM Do, YYYY, h:mm:ss a")
      image {
        fluid(maxWidth: 800) {
          ...GatsbyContentfulFluid
        }
        file {
          url
        }
      }
    }
  }
`;

export default ProductTemplate;

  • We need to modify the components/header.js document to include the Cart summary.

components/header.js

import { Link } from "gatsby";
import PropTypes from "prop-types";
import React from "react";

import gatsbyLogo from "../images/gatsby-icon.png";

const Header = ({ siteTitle }) => (
  <div
    style={{
      background: `rebeccapurple`,
      marginBottom: `1.45rem`
    }}
  >
    <div
      style={{
        display: "flex",
        alignItems: "center",
        justifyContent: "space-between",
        margin: `0 auto`,
        maxWidth: 960,
        padding: `1.45rem 1.0875rem`
      }}
    >
      {/* Title / Logo */}
      <span
        style={{
          display: "flex",
          alignItems: "center"
        }}
      >
        <img
          src={gatsbyLogo}
          alt="Gastby Garb Logo"
          style={{
            borderRadius: "50%",
            border: "3px solid orange",
            margin: "0 5px",
            width: `3rem`
          }}
        />
        <h1 style={{ margin: 0 }}>
          <Link
            to="/"
            style={{
              color: `white`,
              textDecoration: `none`
            }}
          >
            {siteTitle}
          </Link>
        </h1>
      </span>
      {/* Shopping Cart Summary */}
      <div
        style={{ color: "white", cursor: "pointer" }}
        className="snipcart-summary snipcart-checkout"
      >
        <div>
          <strong>My Cart</strong>
        </div>
        <div>
          <span
            style={{ fontWeight: "bold" }}
            className="snipcart-total-items"
          />{" "}
          Items in Cart
        </div>
        <div>
          Total price{" "}
          <span
            style={{ fontWeight: "bold" }}
            className="snipcart-total-price"
          />
        </div>
      </div>
    </div>
  </div>
);

Header.propTypes = {
  siteTitle: PropTypes.string
};

Header.defaultProps = {
  siteTitle: ``
};

export default Header;

  • We need to modify the components/layout.css document to add the styles for the header links.

    components/layout.css

.
.
.
/* Navlink */
.active {
  color: orange;
  text-decoration: none;
}

.active:hover,
.navlink:hover {
  text-decoration: underline;
}

.navlink {
  color: white;
  text-decoration: none;
}
  • We need to modify the components/header.js document to include the links to the blog and the products.

components/header.js

import { Link } from "gatsby";
import PropTypes from "prop-types";
import React from "react";

import gatsbyLogo from "../images/gatsby-icon.png";

const isActive = ({ isCurrent }) => {
  return { className: isCurrent ? "active" : "navlink" };
};

const NavLink = props => <Link getProps={isActive} {...props} />;

const Header = ({ siteTitle }) => (
  <div
    style={{
      background: `rebeccapurple`,
      marginBottom: `1.45rem`
    }}
  >
    <div
      style={{
        display: "flex",
        alignItems: "center",
        justifyContent: "space-between",
        margin: `0 auto`,
        maxWidth: 960,
        padding: `1.45rem 1.0875rem`
      }}
    >
      {/* Title / Logo */}
      <span
        style={{
          display: "flex",
          alignItems: "center"
        }}
      >
        <img
          src={gatsbyLogo}
          alt="Gastby Garb Logo"
          style={{
            borderRadius: "50%",
            border: "3px solid orange",
            margin: "0 5px",
            width: `3rem`
          }}
        />
        <h1 style={{ margin: 0 }}>
          <NavLink to="/">{siteTitle}</NavLink>
        </h1>
      </span>

      <NavLink to="/blog">Blog</NavLink>

      <NavLink to="/products">Store</NavLink>

      {/* Shopping Cart Summary */}
      <div
        style={{ color: "white", cursor: "pointer" }}
        className="snipcart-summary snipcart-checkout"
      >
        <div>
          <strong>My Cart</strong>
        </div>
        <div>
          <span
            style={{ fontWeight: "bold" }}
            className="snipcart-total-items"
          />{" "}
          Items in Cart
        </div>
        <div>
          Total price{" "}
          <span
            style={{ fontWeight: "bold" }}
            className="snipcart-total-price"
          />
        </div>
      </div>
    </div>
  </div>
);

Header.propTypes = {
  siteTitle: PropTypes.string
};

Header.defaultProps = {
  siteTitle: ``
};

export default Header;

16. Deploying our Gatsby Sites with Netlify / Setting up Continuous Integration

42. Using Netlify for Deployment / Continuous Deployment with Github

  • We need to create a new Github repository.

  • We need to push this repository from the command line:
Juan.Pablo.Perez@RIMDUB-0232 MINGW64 /c/Work/Training/Pre/Gatsby/gatsby-garb (master)
$ git remote add origin https://github.com/peelmicro/the-gatsby-masterclass.git

Juan.Pablo.Perez@RIMDUB-0232 MINGW64 /c/Work/Training/Pre/Gatsby/gatsby-garb (master)
$ git push -u origin master
Enumerating objects: 98, done.
Counting objects: 100% (98/98), done.
Delta compression using up to 4 threads
Compressing objects: 100% (94/94), done.
Writing objects: 100% (98/98), 656.45 KiB | 4.94 MiB/s, done.
Total 98 (delta 38), reused 0 (delta 0)
remote: Resolving deltas: 100% (38/38), done.
To https://github.com/peelmicro/the-gatsby-masterclass.git
 * [new branch]      master -> master
Branch 'master' set up to track remote branch 'master' from 'origin'.

  • We need to add our Environment variables:

6:49:03 AM: Build ready to start
6:49:29 AM: build-image version: 84aca9ba39e0ee86ba194760fbfc51a808f62543
6:49:29 AM: buildbot version: 1ac64ca11e029436ed45ac81a38b9839778ec314
6:49:29 AM: Fetching cached dependencies
6:49:30 AM: Failed to fetch cache, continuing with build
6:49:30 AM: Starting to prepare the repo for build
6:49:30 AM: No cached dependencies found. Cloning fresh repo
6:49:30 AM: git clone https://github.com/peelmicro/the-gatsby-masterclass
6:49:31 AM: Preparing Git Reference refs/heads/master
6:49:31 AM: Starting build script
.
.
.
6:50:42 AM: Starting to deploy site from 'public/'
6:50:44 AM: Starting post processing
6:50:46 AM: Post processing done
6:50:46 AM: Site is live
6:50:59 AM: Finished processing build request in 1m29.470138198s
6:50:59 AM: Shutting down logging, 0 messages pending

17. User Authentication in Gatsby / Netlify Identity

43. Using Netlify Identity for Multi-Factor Authentication in our Site

  • Go to Netlify Identity

  • We need to install the Widget.
Juan.Pablo.Perez@RIMDUB-0232 MINGW64 /c/Work/Training/Pre/Gatsby/gatsby-garb (master)
$ npm i netlify-identity-widget
npm WARN optional SKIPPING OPTIONAL DEPENDENCY: fsevents@1.2.4 (node_modules\fsevents):
npm WARN notsup SKIPPING OPTIONAL DEPENDENCY: Unsupported platform for fsevents@1.2.4: wanted {"os":"darwin","arch":"any"} (current: {"os":"win32","arch":"x64"})

+ netlify-identity-widget@1.5.2
added 1 package from 1 contributor and audited 29297 packages in 81.074s
found 3 low severity vulnerabilities
  run `npm audit fix` to fix them, or `npm audit` for details
  • We need to modify the components/header.js documents to add the links to authenticate.

components/header.js

  • We can improve the style changing the the components/layout.css document.

    components/layout.css

.
.
.
/* Navlink */
.active {
  color: orange;
  text-decoration: none;
}

.active:hover,
.navlink:hover {
  text-decoration: underline;
}

.navlink {
  color: white;
  text-decoration: none;
}

/* Netlify Identity */
.netlify-identity-menu {
  list-style: none;
  color: white;
  margin: 0;
}

.netlify-identity-signup {
  color: white;
}

.netlify-identity-login {
  color: white;
}

.netlify-identity-logout {
  color: white;
}

  • Netlify Identity only works on production, so we need to commit and push the changes.
Juan.Pablo.Perez@RIMDUB-0232 MINGW64 /c/Work/Training/Pre/Gatsby/gatsby-garb (master)
$ git add .

Juan.Pablo.Perez@RIMDUB-0232 MINGW64 /c/Work/Training/Pre/Gatsby/gatsby-garb (master)
$ git commit "Added Netlify Identity"
error: pathspec 'Added Netlify Identity' did not match any file(s) known to git

Juan.Pablo.Perez@RIMDUB-0232 MINGW64 /c/Work/Training/Pre/Gatsby/gatsby-garb (master)
$ git commit -m "Added Netlify Identity Widget to header"
[master 3a1eefc] Added Netlify Identity Widget to header
 4 files changed, 124 insertions(+), 91 deletions(-)
 rewrite src/components/header.js (77%)

Juan.Pablo.Perez@RIMDUB-0232 MINGW64 /c/Work/Training/Pre/Gatsby/gatsby-garb (master)
$ git push origin HEAD
Enumerating objects: 15, done.
Counting objects: 100% (15/15), done.
Delta compression using up to 4 threads
Compressing objects: 100% (8/8), done.
Writing objects: 100% (8/8), 1.67 KiB | 285.00 KiB/s, done.
Total 8 (delta 5), reused 0 (delta 0)
remote: Resolving deltas: 100% (5/5), completed with 5 local objects.
To https://github.com/peelmicro/the-gatsby-masterclass.git
   1d6760e..3a1eefc  HEAD -> master

44. Displaying Private Products for Auth Users

  • Modify the pages/product.js document to show all the products only if the user is authenticated.

pages/product.js

import React from "react";
import { graphql, Link } from "gatsby";
import Img from "gatsby-image";
import netlifyIdentity from "netlify-identity-widget";

import Layout from "../components/layout";

class Products extends React.Component {
  state = {
    products: []
  };

  componentDidMount() {
    this.getProducts();
    netlifyIdentity.on("login", user => this.getProducts(user));
    netlifyIdentity.on("logout", () => this.getProducts());
  }

  getProducts = user => {
    console.log("Current User", user);
    const allProducts = this.props.data.allContentfulProduct.edges;
    const products =
      netlifyIdentity.currentUser() !== null
        ? allProducts
        : allProducts.filter(({ node: product }) => !product.private);
    this.setState({ products });
  };

  render() {
    const { products } = this.state;

    return (
      <Layout>
        <div>
          {/* Products List */}
          {products.map(({ node: product }) => (
            <div key={product.id}>
              <h2>Garb Products</h2>
              <Link
                to={`/products/${product.slug}`}
                style={{ textDecoration: "none", color: "#551a8b" }}
              >
                <h3>
                  {product.name} ·{" "}
                  <span
                    style={{
                      fontSize: "1.2rem",
                      fontWeight: 300,
                      color: "#f60"
                    }}
                  >
                    ${product.price}
                  </span>
                </h3>
              </Link>
              <Img style={{ maxWidth: 400 }} fluid={product.image.fluid} />
            </div>
          ))}
        </div>
      </Layout>
    );
  }
}

export const query = graphql`
  {
    allContentfulProduct {
      edges {
        node {
          id
          slug
          name
          price
          private
          image {
            fluid(maxWidth: 400) {
              ...GatsbyContentfulFluid
            }
          }
        }
      }
    }
  }
`;

export default Products;

  • Login doesn't work locally

  • We need to commit the changes so that the web site is deployed by Netlify
Juan.Pablo.Perez@RIMDUB-0232 MINGW64 /c/Work/Training/Pre/Gatsby/gatsby-garb (master)
$ git add .
warning: LF will be replaced by CRLF in src/pages/products.js.
The file will have its original line endings in your working directory

Juan.Pablo.Perez@RIMDUB-0232 MINGW64 /c/Work/Training/Pre/Gatsby/gatsby-garb (master)
$ git commit -m "Added private product feature to auth users"
[master ed05371] Added private product feature to auth users
 2 files changed, 92 insertions(+), 64 deletions(-)
 rewrite src/pages/products.js (61%)

Juan.Pablo.Perez@RIMDUB-0232 MINGW64 /c/Work/Training/Pre/Gatsby/gatsby-garb (master)
$ git push origin HEAD
Enumerating objects: 13, done.
Counting objects: 100% (13/13), done.
Delta compression using up to 4 threads
Compressing objects: 100% (7/7), done.
Writing objects: 100% (7/7), 1.19 KiB | 203.00 KiB/s, done.
Total 7 (delta 5), reused 0 (delta 0)
remote: Resolving deltas: 100% (5/5), completed with 5 local objects.
To https://github.com/peelmicro/the-gatsby-masterclass.git
   3a1eefc..ed05371  HEAD -> master