A Gatsby theme for creating a blog. Check out the live demo.
The default styling for the list of blogposts looks like this:
And the default styling for a single blogpost:
The lighthouse score for most pages looks like this:
- A plug and play feature rich blog platform
To use this theme in your Gatsby sites:
-
Install the theme
npm install --save @nickymeuleman/gatsby-theme-blog
-
Add the theme to your
gatsby-config.js
:module.exports = { plugins: ["@nickymeuleman/gatsby-theme-blog"], };
-
Start your site
gatsby develop
-
Add an
authors
file and create a post! Instructions/details in the Usage section
Key | Default value | Description |
---|---|---|
assetPath |
"data/assets" |
Folder location to house extra assets (like the author file.) |
instances |
See instance options |
Array of instance options objects |
gatsbyRemarkPlugins |
[] |
Additional plugins array to be used by gatsby-plugin-mdx |
remarkPlugins |
[] |
Additional plugins array to be used by gatsby-plugin-mdx |
rehypePlugins |
[] |
Additional plugins array to be used by gatsby-plugin-mdx |
This theme uses gatsby-plugin-mdx
and allows you to customize some configuration that it uses via the gatsbyRemarkPlugins
, remarkPlugins
, and rehypePlugins
options.
Those options take an array of the relevant plugins to be used.
For more information on how to use those options,
see the documentation in gatsby-plugin-mdx
.
Key | Default value | Description |
---|---|---|
basePath |
"" |
Root url for this instance. eg: blog |
contentPath |
"data/posts" |
Folder location to house individual post-folders for this instance |
pagination |
undefined |
Optional object, enables pagination if provided |
Key | Default value | Description |
---|---|---|
postsPerPage |
10 |
Amount of posts per paginated page |
prefixPath |
"" |
Optional string. Prefixes numbers in the URL paths from paginated pages. eg: page |
// gatsby-config.js
module.exports = {
plugins: ["@nickymeuleman/gatsby-theme-blog"],
};
// gatsby-config.js
module.exports = {
plugins: [
{
resolve: "@nickymeuleman/gatsby-theme-blog",
options: {
assetPath: "assets",
instances: [
{
contentPath: "posts",
basePath: "blog",
pagination: {
postsPerPage: 10,
prefixPath: "page",
},
},
{
contentPath: "notes",
basePath: "notes",
},
],
gatsbyRemarkPlugins: [
{
resolve: `gatsby-remark-katex`,
options: {
strict: `ignore`,
},
},
],
},
},
],
};
This will result in a seperate /blog
and /notes
path.
The posts in /blog
will be sourced from the posts
folder.
The posts in /notes
will be sourced from the notes
folder.
Only posts in /blog
will be paginated. With each paginated page holding a maximum of 10 posts.
Paginated pages after the one that lists the first 10 posts, will be prefixed be /page
. eg. /blog
, /blog/page/2
, /blog/page/3
, etc.
A plugin to add support for math equations via KaTeX is added.
Sidenote: For KaTeX to work correctly, The plugin on its own is not enough.
The CSS also has to be included on the pages equations are used, in the demo this is done in gatsby-browser.js
.
Authors are shared between instances (be it /blog
or /notes
). A single author can write posts in both instances.
Refer to the Adding authors to see how to add authors.
In addition to the theme options, there are a handful of items you can customize via the siteMetadata
object in your site's gatsby-config.js
// gatsby-config.js
module.exports = {
siteMetadata: {
// Used for the site title, SEO, and header component title.
title: `My Blog Title`,
// Used for SEO
description: `My site description...`,
// Used for SEO and as default if an author has no twitter defined
social: {
twitter: `@NMeuleman`,
},
},
};
In the folder that was created for the contentPath
(data/posts
by default). Create a folder to hold a blog post. Unless a slug is provided, the title of this folder will serve as the slug for the blogpost.
Inside that folder, an index.mdx
or index.md
file will be the blog post itself. Along this file can be several different files specific to that blogpost (e.g. images)
If no date is specified, the date the .md(x)
file was created will serve as the date for the blogpost.
NOTE: If you dislike having a folder per blogpost, loose
.md(x)
files are also supported. Place them inside the folder created forcontentPath
. The title of the file will then serve as the slug of the blogpost if a slug not specified in the post's slug field.
This theme adds a couple of things you can do inside an .mdx
file.
Codeblocks, obtained by surrounding a piece of code with three backticks ```
are powered by prism-react-renderer.
As a result, these codeblocks support syntax highlighting for a range of languages.
Highlighting specific lines is possible by passing a string to the hl
(short for highlight-lines) attribute when writing a codeblock.
A title may also be added to the codeblock by passing a string to the title
attribute when writing a codeblock.
Numbering lines is possible by passing a numberLines
attribute.
Don't like the copy button being there? Disable it by passing the noCopy
attribute.
The value you pass to hl
should be understood by node-parse-numeric-range.
If passing a value to numberLines
, it should be either true
(to start the numbering at 1), or a decimal integer (to start the numbering at that number).
The code below will highlight the code inside as the jsx
language, highlight lines 1,2,3, and 5, and put a small block above it displaying src/components/CodeBlock.jsx
The lines will be numbered starting at 1.
(Omit the #
in front of these lines, I only used them here to make the three backticks show up).
# ```jsx numberLines hl=1-3,5 title=src/components/CodeBlock.jsx
# import React from 'react'
# import Highlight, { defaultProps } from 'prism-react-renderer'
#
# export default ({ children, className }) => {
# const language = className.replace(/language-/, '') || ""
# return (
# <Highlight {...defaultProps}
# code={children}
# language={language}
# >
# {({ className, style, tokens, getLineProps, getTokenProps }) => (
# <pre className={className} style={{ ...style }}>
# {tokens.map((line, index) => {
# const lineProps = getLineProps({ line, key: index })
# return (
# <div key={index} {...lineProps}>
# {line.map((token, key) => (
# <span key={key}{...getTokenProps({ token, key })} />
# ))}
# </div>
# )
# }
# )}
# </pre>
# )}
# </Highlight>
# )
# }
# ```
An example folder tree for this theme with the default options:
.
└── data
├── assets
│ ├── authors.json
│ └── image-used-often.jpg
└── posts
├── my-first-post
│ ├── index.mdx
│ ├── coverPhoto.jpg
│ ├── boop.png
│ └── infinite-boop.gif
└── my-second-post
├── index.md
├── f1-car.jpg
└── speed-data.svg
In the folder that was created for the assetPath
(data/assets
by default). Create a file called authors.json
or authors.yaml
. This file (or files, both formats can work together) holds an array of author objects.
An authors file contains a top level array filled with object describing individual authors. An author can have several different field with information specific to them.
Key | Value | Required | Description |
---|---|---|---|
shortName |
string | yes | Unique, url-safe identifier for the author, used in author field for blog posts |
name |
string | yes | Full name eg. "Nicky Meuleman" |
twitter |
string | no | Twitter handle without @ |
image |
string | no | Relative path from the authors file. |
The blogpost itself (.md
or .mdx
file for now, others coming soon) can have several different fields with extra information.
In .md
or .mdx
files these fields are set via the frontmatter.
Key | Value | Required | Description |
---|---|---|---|
title |
string | no | Title of the post. |
date |
date string | no | Date the post was written. |
updatedAt |
date string | no | Date the post was last updated. |
canonicalUrl |
full url string | no | Canonical url. |
authors |
array of shortName strings |
no | Authors of the post. Should not be used in combination with the author key. |
author |
shortName string |
no | Author of the post. Should not be used in combination with the authors key. |
tags |
array of tag strings | no | Tags for the post. |
keywords |
array of keyword strings | no | Keywords for SEO. |
cover |
relative path to cover image | no | Displayed as cover image, in social cards. |
published |
boolean, defaults to true |
no | Include the post in production. |
slug |
string | no | The last part of the URL for the post. |
series |
string | no | Name of the group/series this post is a part of. |
Overwriting these with your own is highly encouraged. This can be done via component shadowing.
These components house the Gatsby template-query.
They lightly transform that data and pass it on to the corresponsing Page component.
Changes to these might require changes to the corresponding Page components because of that.
Location to shadow: @nickymeuleman/gatsby-theme-blog/templates/<component-name>
- BlogPostQuery
- BlogPostListQuery
- TagQuery
- TagListQuery
- AuthorQuery
- AuthorListQuery
These components render an entire page.
Each component is wrapped in the Layout
component that centers the content and adds the Header
.
Location to shadow: @nickymeuleman/gatsby-theme-blog/components/<component-name>
- BlogPostPage
- BlogPostListPage
- TagPage
- TagListPage
- AuthorPage
- AuthorListPage
These components are used by the Page components
Location to shadow: @nickymeuleman/gatsby-theme-blog/components/<component-name>
- PostCard
- PostExtra
- Pagination
- CodeBlock
- SeriesSelect
These components are usable in .mdx
files without importing them first.
- Every component from Gatby-mdx-embed
- Aside
To add to this list, shadow @nickymeuleman/gatsby-theme-blog/components/mdx-components
.
Every component that is exported as a named export will be available for use in .mdx
under that name, without importing it first.
example declaration of a <Shia />
component
// in src/@nickymeuleman/gatsby-theme-blog/components/mdx-components/index.js
import React from "react";
export * from "@nickymeuleman/gatsby-theme-blog/src/components/mdx-components/index";
const Shia = ({ children }) => (
<p>Don't let your dreams be dreams! {children}</p>
);
export { Shia };
example usage in an .mdx
file
# Shia LaBeouf says:
<Shia>
Just do it!
</Shia>
- Header
- layout
- Main
If you want to use component shadowing, create a file at the following path in your site:
src/@nickymeuleman/gatsby-theme-blog/components/<component-name>.js
In any MDX blogpost:
import { ComponentName } from "@nickymeuleman/gatsby-theme-blog"
---
<frontmatter-fields>
---
# Lorem Ipsum
<ComponentName />
Components from Gatby-mdx-embed can be used without first importing them
In any React component:
import React from "react";
import { ComponentName } from "@nickymeuleman/gatsby-theme-blog";
export default () => (
<div>
<ComponentName />
</div>
);
This theme uses theme-ui extensively for styling.
That means a lot of the "look" (CSS-styling) can be overridden by the user, through their own theme-file.
To use one, export an object that adheres to the theme-ui spec from src/gatsby-plugin-theme-ui/index.js
.
It is recommended that you extend this theme's theme-ui file and add your own overrides on top.
The code below does this by merging the config used by this theme and overwriting a few color
values that are used by this theme to make a beautiful, beautiful, website.
import { themeConfig } from "@nickymeuleman/gatsby-theme-blog";
import merge from "deepmerge";
const theme = merge(themeConfig, {
colors: {
text: `tomato`, // the named CSS color
mutedText: `coral`, // the named CSS color
background: `#000`,
mutedBackground: `rgb(50,50,50)`,
primary: `red`, // the last entry from the array in the theme's config for "red", not raw CSS red
mutedPrimary: `red[2]`, // the red at index 2 from the array in the theme's config for "red"
},
});
export default theme;
To apply a different syntax highlighting theme to code blocks. Overwrite the styles under styles.CodeBlock
in the theme-ui file.
- To adjust the styles used for line highlighting, add new rules under
styles.CodeBlock.lineHighlight
. - To adjust the styles used for the title section of a codeblock, add new rules under
styles.CodeBlock.title
. - To adjust the styles used for the copy button, add new rules under
styles.CodeBlock.copyButton
.
Various pregenerated themes are available in the /presets
folder of @theme-ui/prism.
To use them, import your chosen theme's .json
file, and apply it to the styles.CodeBlock
of your theme-ui file.
The code below imports a syntax highlighting theme based on the popular night-owl by Sarah Drasner. It also adds some styles so highlighting a line, or adding a title to the codeblock, will match the installed syntax theme.
import nightOwl from "@theme-ui/prism/presets/night-owl.json";
import { themeConfig } from "@nickymeuleman/gatsby-theme-blog";
import merge from "deepmerge";
const theme = merge(themeConfig, {
styles: {
CodeBlock: {
...nightOwl,
highlightLine: {
backgroundColor: `#01121f`,
borderLeftColor: `#9ccc65`,
},
title: {
backgroundColor: nightOwl.backgroundColor,
borderBottomColor: `#262a39`,
color: nightOwl.color,
},
},
},
});
export default theme;
- Multi sourcing
- 🚧 👷 revamp way slugs are handled
- Links with a basepath broke around june 2020. Reworked to use
path.join
and force starting/
- Links with a basepath broke around june 2020. Reworked to use
- code blocks
- Look into code blocks that pass contrast checks system-ui/theme-ui#564
- some light styling? (future Nicky notes: That ended up being more than light.)
- redo styles with theme-ui, like
<li>
margins?
- redo styles with theme-ui, like
- css reset?
- rss feed?
- offline, manifest, ...?
- icons (default icon for SEO if no fitting image found)
- icons for tags?
- look into https://zeit.co/blog/social-og-image-cards-as-a-service
- canonical url support:
- functionality works, now add docs
- add a "originally published at" line to blogpost component
- multiple authors support
- multiple authors per post
- avatar for each author, because, pretty pictures are tight (watch @theryangeorge)
- rename "author" in graphql to "authors", it's an array
- revamp how SEO component works
- migrate from Helmet props to nested html tags?
- pass less props into SEO component
- Document what tasks individual components perform, how ones include others
- make tags array optional
- cannot return null for non-nullable field MdxTag.id
- published frontmatter field.
[ ] option to hide unpublished articles when running "gatsby develop"?hidden in blog list, individual pages exist.
- different content folder for authors? (maybe together with images etc that are not directly tied to a single blogpost)
- Ability to specify path in frontmatter. See: gatsbyjs/gatsby#16611
- Double images when linked like
![](image.png)
in mdx. Blurry and fullsize.- Eventual fix gatsbyjs/gatsby#16242
- temporary fix: add
plugins
instead ofgatsbyRemarkPlugins
- Add link icons next to headings via: https://theme-ui.com/recipes/linked-headings/
- note: Why was this better than
gatsby-remark-autolink-headers
again? Saw it in a GitHub issue somewhere - ChristopherBiscardi/gatsby-mdx#204 and gatsbyjs/gatsby#14520
- note: Why was this better than
- Refactor theme options to use defaults if not specified to avoid repeating yourself. See: https://github.com/gatsbyjs/gatsby/blob/master/themes/gatsby-theme-blog-core/utils/default-options.js
- Make the date of blogposts default to the time the file was created in case the date frontmatter field was not specified.
- Make title of blogposts default to the unkebabcased title of the folder.
- Allow unkebabcased slugs in frontmatter
- Allow blog posts as plain markdown/mdx files, not in a folder
- document loose files as blogposts
- Allow author field in frontmatter to be optional
- allow for authorless posts
- Make demo website function like a big readme.
- gatsbyjs/gatsby#16149 got merged, use it.
- gatsbyjs/gatsby#17284 got merged, use it. Temp hack: add empty proxy directive: gatsbyjs/gatsby#21476 Temp hack upon temp hack time! (I hate this, but it works) Add a meaningless { from: "foo" } to that proxy directive datocms/gatsby-source-datocms#188
- Migrate theme to TypeScript, leave demo as JavaScript
- Add testing with react-testing-library and cypress
- Actually add tests once the scaffolding is done
- Use something like codecov to ensure test coverage
- Social share images
Research: - https://github.com/zeit/og-image - https://lannonbr.com/blog/2019-11-10-og-images/ - https://github.com/ChristopherBiscardi/gatsby-plugins/tree/master/packages/gatsby-plugin-printer - https://www.learnwithjason.dev/blog/auto-generate-social-image/ - https://github.com/jlengstorf/get-share-image - https://www.swyx.io/writing/jamstack-og-images/ - https://aless.co/gatsby-wasm-plugin/ - https://github.com/alessbell/gatsby-remark-twitter-cards - https://andrewingram.net/posts/automatic-social-cards-with-gatsby/ - https://github.com/syntra/gatsby-remark-social-cards > twitter card-size image (630 x 1200 px) > from: Andres Ingram link >
// Set the viewport to the desired dimensions of the image
>await page.setViewport({ width: 2048, height: 1170 });
personal note: analysis paralysis is real - Components in .mdx 🚧
Like this
<Aside>
https://github.com/jlengstorf/learnwithjason.dev/blob/master/site/src/components/aside.js also<YouTube>
,<Twitter>
,<CodeSandbox>
etc. maybe use https://github.com/MichaelDeBoey/gatsby-remark-embedder for part of those embeds. https://github.com/PaulieScanlon/gatsby-mdx-embed is in line with original idea of creating the components myself. Implemented. Caused some issues: detailed PaulieScanlon/gatsby-mdx-embed#11 - Migrate to theme-ui v0.3
- Rename Underpost to PostExtra
- Rework default
contentPath
andassetPath
- Remove duplicate remark images plugin
- Page per author
- Avatar support for authors
- Use avatars in demo.
- CLI to scaffold out now blogposts (See how Kyle Shevlin does this.)
- theme border values. Only take the sharp edge off, don't round too much.
- add support for updating blogposts (updatedAt frontmatter field?)
- Rework how cards are done
BlogList
https://inclusive-components.design/cards/ - Use inversion of control and component composition more.
- eg. Using
props.children
inside thePostExtra
component.
- eg. Using
- Refactor all page creations to use same pattern. Query in templates/* that only renders a component named *Page. That component has the
<SEO />
- Rename all top level page components to have
Page
in their name. - Relocate
Layout
component from templates toPage
components. - Rename all gatsby templates to have
Query
in their name. Since they are made up of a query, passing that data to aPage
component. - Handle data in every page component the same way
- Handle passing of
basePath
the same everywhere. - Sort named exports from this theme
- Put theme options in a Gatsby node
https://www.christopherbiscardi.com/post/applying-theme-options-using-custom-configuration-nodes/
- Make React hook that queries for those options and use the hook instead of passing options around in pageContext etc
- Ability to add custom OG images for pages like BlogListPage (also use the
<SEO />
image
prop for that?)-
type
prop for<SEO />
? Different behavior based on type?- The
basePath
prop now fulfills this role.
- The
- refactor
Page
components so<SEO />
can easily be swapped?
-
- Give components a
variant
so the user can theme them via theme-ui - Fix links in headers being hidden (because of the link icon)
- Table of contents
- Style active link in table of contents
- Only run that logic on desktop and if table of contents is open
- Style active link in table of contents
- Refactor exports to be named exports as often as possible (reason: node import syntax getting closer to ESM, but only named. While not needed here because of compilation, good to be consistent everywhere, regardless of technology used.)
- Refactor @theme-ui/prism to self-rolled setup of prism-react-renderer for increased flexibility.
- Add line highlighting
- Add code title support
- document everything
- Support multiple basePaths (eg
/blog
,/notes
, and/tips
)- Redo theme options, users can define multiple
instances
. Some options are specific to the instance. - Create folders based on the
contentPath
- Support sharing authors across those multiple basePaths.
- Moved authors page paths from
/<basepath>/author/<name>
to the top level/author/<name>
as a result.
- Moved authors page paths from
- Support different
Page
components for different instances.
- Redo theme options, users can define multiple
- PaginationContext and PageContext in BlogPostListPage are duplicated right now
- Redo how the
<SEO />
gathers information. It's becoming an a-prop-calypse in there. - Support line numbering in codeblocks
- Support adding own plugins to the internal
gatsby-plugin-mdx
- Migrate theme-ui to Typescript https://theme-ui.com/guides/typescript/
- Add copy button to codeblocks
- Use Gatsby v4
- Use mdx v2
- Use theme-ui v0.15
- Use Gatsby head instead of React-Helmet: https://www.gatsbyjs.com/docs/reference/built-in-components/gatsby-head/
notes: can't see the change mentioned at https://www.gatsbyjs.com/docs/reference/release-notes/migrating-from-v3-to-v4/#gatsby-transformer-json no jsonId or yamlId to be found. maybe this change happened in v5 of that plugin and this theme is on v4?