This is a personal or professional journal/posts starter build on Next.js, a React framework, with Tailwind CSS and content provided as markdown or MDX files by one or more authors. It is adapted from Tailwind Nextjs Starter Blog template by Timothy Lin, extended to include a resume and support non-blog pages (also authored in MD or MDX), among other features. It is the source code for jeffruss.com but, like the Tailwind Nextjs Starter Blog, can be used as a template as well. Here are some other features:
-
Light and dark theme with easy styling customization with Tailwind 3.0.
-
Mobile-friendly and responsive
-
SEO friendly with RSS feed, sitemaps, etc.
-
Preconfigured security headers
-
Newsletter support using mailchimp, buttondown, convertkit, klaviyo and revue
-
Analytics support with plausible, simple analytics and google analytics
-
For all MD/MDX pages:
- Code blocks with syntax highlighting, copy button, line numbers and line highlighting via rehype-prism-plus
- Math blocks and inline math display supported via KaTeX
- Citation / bibliography support with rehype-citation
- Support for nested routing of blog posts and pages
-
Blog/Journal Posts with script for generating boilerplate with metadata (Frontmatter):
- Support for tags - each unique tag will be its own page
- Support for multiple authors
- Table of Contents component
- Comment system using giscus, utterances or disqus
This project uses node and npm, which must be locally installed, along with git.
To install with the intent to Contribute changes:
git clone https://github.com/Jeff-Russ/next-markdown-journal.git
cd next-markdown-journal
npm install
To install with the intent to customize for personal use:
npx degit https://github.com/Jeff-Russ/next-markdown-journal.git
cd next-markdown-journal
npm install
To run the app, making it available at available http://localhost:3000 you can run npm run dev
or you can run:
npm start
This way will reload application markdown and other contents of data/
changes.
data/siteMetadata.js
is where most information related to a customized deployment of the app would be set, with some values are taken as process.env.NEXT_PUBLIC_*
environment variables set in .env*
files. The value set for defaultAuthorSlug
in data/siteMetadata.js
must match a filename without extension which is located within data/authors
. This file will be used to define the metadata for the default author of a post when none is specified, as well as the author details displayed on /about
page.
Overall, you'll at least need to create or modify the following files:
- Modify
data/siteMetadata.js
, which contains most of the site related information which should be modified for a user's needs. - Create
`authors/${siteMetadata.defaultAuthorSlug}.md|mdx`
which is the default author information. Additional authors can be added as files indata/authors/
. - Modify
data/projectsData.js
to generate styled cards on the/projects
page. - Replace images in
public/static/favicons/
with your own logo. - Replace/add images in
public/static/images/
referenced indata/siteMetadata.js
as well in author files indata/authors/
and any posts you may create indata/posts
anddata/pages
. - Replace files in
data/posts
with your own posts.
Additionally you may need or want to do the following:
- Modify
next.config.js
to adapt the Content Security Policy if you want to load scripts, images etc. from other domains. You'll also modifyContentSecurityPolicy
to your choice of analytics and commenting system. - Modify
tailwind.config.js
andcss/tailwind.css
to change styling of the site. - Modify or add files to
layouts/
to change the layout and styling pages. The component filenames must match what it specified in the markdown frontmatterlayout
field. - Modify
css/prism.css
to change the styles associated with the code blocks, such as by using your preferred prism theme. - Add other icons to
components/social-links
such as from Simple Icons or heroicons.and map them inindex.js
. - Modify
data/headerNavLinks.js
if you'd like to change the links in the navbar. - Modify
./components/Pre.js
to customize how all code blocks (rendered as<pre>
elements) are rendered. - Install and use a self-hosted font from Fontsource.
You can run node ./scripts/compose.js
and follow the interactive prompt to generate a post with pre-filled frontmatter.
Frontmatter follows Hugo's standards. Currently 7 fields are supported.
title (required)
date (required)
tags (required, can be empty array)
lastmod (optional)
draft (optional)
summary (optional)
images (optional, if none provided defaults to socialBanner in siteMetadata config)
authors (optional list which should correspond to the file names in `data/authors`. Uses `default` if none is specified)
layout (optional list which should correspond to the file names in `data/layouts`)
canonicalUrl (optional, canonical url for the post for SEO)
Both .md
and .mdx
, beside the frontmatter, are authored in markdown syntax but .mdx
files allow for React components as JSX elements to be inserted such as <TOCInline >
The props.toc
variable containing all the top headings of the document is passed to the MDX file and can be styled into a table of contents by providing it as the value of the toc
attribute of <TOCInline >
. For example, the following can be added to a .mdx
post:
<TOCInline toc={props.toc} exclude="Overview" toHeading={2} />
- You can customize the range of heading depths that are displayed by configuring the
fromHeading
andtoHeading
props. - Exclude particular headings by passing a string or a string array to the
exclude
prop. - You can change the indent size of each deeper heading relative to the previous by setting the
indentRem
property. Typical values would be0.5
to5
.s - The
asDisclosure
prop can be used to render the TOC within an expandable disclosure element. - Adding a
closed
attribute in addition toasDisclosure
makes the TOC be collapsed by default.
<TOCInline toc={props.toc} asDisclosure closed />
- The following code block will have lines 1, 3 and 4 highlighted due to the addition of
{1, 3-4}
. - Line numbers are displayed due to the addition of
showLineNumbers
var num1, num2, sum
num1 = prompt('Enter first number')
num2 = prompt('Enter second number')
sum = parseInt(num1) + parseInt(num2) // "+" means "add"
alert('Sum = ' + sum) // "+" means combine into a string
To modify the styles applied to this, change the .code-highlight
, .code-line
, .code-line.inserted
, .code-line.deleted
, .highlight-line
, and .line-number::before
class selectors in the prism.css
file.
rehype-citation
plugin is added to the xdm processing pipeline in v1.2.1. This allows you to easily format citations and insert bibliography from an existing bibtex or CSL-json file.
For example, the following markdown code sample:
Standard citation [@Nash1950]
In-text citations e.g. @Nash1951
Multiple citations [see @Nash1950; @Nash1951, page 50]
**References:**
[^ref]
is rendered to the following:
Standard citation [@Nash1950]
In-text citations e.g. @Nash1951
Multiple citations [see @Nash1950; @Nash1951, page 50]
References:
[^ref]
A bibliography will be inserted at the end of the document, but this can be overwritten by specifying a [^Ref]
tag at the intended location.
The plugin uses APA citation formation, but also supports the following CSLs, 'apa', 'vancouver', 'harvard1', 'chicago', 'mla', or a path to a user-specified CSL file.
See rehype-citation readme for more information on the configuration options.
According to next.js/docs/basic-features/environment-variables.md on GitHub sensitive data should only be specified in .env.local
as it is where secrets should be stored. As a result, it, along with all other .env*.local
files are .gitignore
d. Default values should be defined in .env
, .env.development
, and .env.production
which all should be files should be included in your repository.
In all likelihood, You will need to create the file .env.local
. as a duplicate of .env.example
and then fill in the values as needed as per the instructions below.
For more information on the required variables, check out .env.example
and this guide
Fonts can be added from Fontsource. These are self-hosted fonts which have advantages:
Self-hosting brings significant performance gains as loading fonts from hosted services, such as Google Fonts, lead to an extra (render blocking) network request. To provide perspective, for simple websites it has been seen to double visual load times.
Fonts remain version locked. Google often pushes updates to their fonts without notice, which may interfere with your live production projects. Manage your fonts like any other NPM dependency.
Commit to privacy. Google does track the usage of their fonts and for those who are extremely privacy concerned, self-hosting is an alternative.
This leads to a smaller font bundle and a 0.1s faster load time (webpagetest comparison).
The default font is Fontsource's 'inter' font. You can see the dependency "@fontsource/inter"
is added to package.json
.
To change the default font (here is another good guide if you need further details):
- Pick a font at Fontsource. Here are some notable ones:
abril-fatface
is good for attention-grabbing headings.- For: mono fonts:
fira-mono
,source-code-pro
are decent. - Regular looking for paragraphs:
inter
,poppins
andabel
(both sans) - Interesting looking for paragraphs:
alice
(serif) - Very interesting looking for paragraphs:
advent-pro
,aldrich
(both sans) - A versatile collection: The
ibm-plex-*
fonts.
- Install the preferred font -
npm install -save @fontsource/<font-name>
You might need to do some sleuthing inside node_modules/@fontsource/<font-name>/
to do the next two steps.
- Update the import at
pages/_app.js
-import '@fontsource/<font-name*>'
. Do you sleuthing: If you want to importnode_modules/@fontsource/alice/index.css
, the*
in<font-name*>
would be nothing; you would just:import '@fontsource/alice'
. But you may want to import different files such asnode_modules/@fontsource/inter/variable-full.css
in which case the import statement would beimport '@fontsource/inter/variable-full.css'
- Update the
fontfamily
property in thetailwind.config.js
file. Read this and this for help. Again, you'll need to do some sleuthing. Taking the example of theinter
font where we did'@fontsource/inter/variable-full.css'
we see in that file the linefont-family: 'InterVariable'
which means we addsans: ['InterVariable', ...defaultTheme.fontFamily.sans],
totailwind.config.js
because we found'InterVariable'
is the font family and it is a sans serif font.
Of course, if you are not using any particular font such as @fontsource/<unused-font-name>
, you can then delete it with npm uninstall -save @fontsource/<unused-font-name>
.
We have also added support for giscus, utterances or disqus for comments to posts. To enable a blog comments system, edit the siteMetadata.analytics
properties in siteMetadata.js
. Some correspond to the following environment variables found in .env*
such as .env.local
:
NEXT_PUBLIC_GISCUS_REPO=
NEXT_PUBLIC_GISCUS_REPOSITORY_ID=
NEXT_PUBLIC_GISCUS_CATEGORY=
NEXT_PUBLIC_GISCUS_CATEGORY_ID=
NEXT_PUBLIC_UTTERANCES_REPO=
NEXT_PUBLIC_DISQUS_SHORTNAME=
In addition to the these four environmental variables for Giscus:
GISCUS_REPO
which sets siteMetadata.comment.giscusConfig.repo
GISCUS_REPOSITORY_ID
which sets siteMetadata.comment.giscusConfig.repositoryId
GISCUS_CATEGORY
which sets siteMetadata.comment.giscusConfig.category
GISCUS_CATEGORY_ID
which sets siteMetadata.comment.giscusConfig.categoryId
...we also need these values:
siteMetadata.comment.giscusConfig.mapping
siteMetadata.comment.giscusConfig.reactions
siteMetadata.comment.giscusConfig.metadata
siteMetadata.comment.giscusConfig.inputPosition
siteMetadata.comment.giscusConfig.lang = en
siteMetadata.comment.giscusConfig.theme
siteMetadata.comment.giscusConfig.darkTheme
siteMetadata.comment.giscusConfig.themeURL
Visiting https://giscus.app/ will provide us with these values, but here is a more in-depth is a summary of the instructions (highighted content shows the variable value obtained with each steps):
Repository section at https://giscus.app/:
- We see we should pick a public repository on GitHub to hold the discussion. I believe this can be the repository that has the source for your site but doesn't have to be; it can simply be a repository who's only purpose is to hold the discussions.
- Next we install the giscus app: Visit https://github.com/apps/giscus, click "Install," selecting "Only select repositories," selecting the one or more respositories and then clicking "install." Doing so will probably take you here, where you can configure Giscus.
- Next, enable the Discussions feature for your repository by visiting Settings > General at your repository on Github and checking off "Discussions." Keep this page open because later on we'll hit "Set up Discussions."
Now back in the bottom of the 'Configuration' section at https://giscus.app/, you should be to type your github-username/project-name
and have it say "Success! This repository meets all of the above criteria." After choosing our repository, we can now use it to assign our environment variable:
GISCUS_REPO=github-username/project-name which sets siteMetadata.comment.giscusConfig.repo
If you scroll down the page at after seeing "Success! This repository meets all of the above criteria" to the "Enable giscus" section, you'll see a value for data-category-id="SOME_VALUE"
which can use to set the environment variable:
GISCUS_REPOSITORY_ID="SOME_VALUE" which sets siteMetadata.comment.giscusConfig.repositoryId
The Page ↔ Discussions Mapping section (after 'Repository') at https://giscus.app/ is where we choose the mapping between the embedding page and the embedded discussion.
☑ Discussion title contains page
pathname
— Giscus will search for a discussion whose title contains the page'spathname
URL component. ☐ Discussion title contains pageURL
— Giscus will search for a discussion whose title contains the page's URL. ☐ Discussion title contains page<title>
— Giscus will search for a discussion whose title contains the page's<title>
HTML tag. ☐ Discussion title contains pageog:title
— Giscus will search for a discussion whose title contains the page's<meta property="og:title">
HTML tag. ☐ Discussion title contains a specific term — Giscus will search for a discussion whose title contains a specific term. ☐ Specific discussion number — Giscus will load a specific discussion by number. This option does not support automatic discussion creation.
Choose the pathname
option since others might not be distinct. But beware of modifying the site to change paths! I'm not sure that this would be a problem but it certainly may be. In any case, we don't need to fill this out here as we will set this in:
siteMetadata.comment.giscusConfig.mapping = 'pathname'
At the Discussion Category section at https://giscus.app/ you won't find a category that suits what we are doing so...
Back in Settings > General at your repository where you checked off "Discussions," click "Set up Discussions." This will take you to a pre-written message from you to "Start a new discussion." Click "Start discussion" to post it. Now click on the Discussions tab (already selected) to take you to the home of Discussions. Click on the pencil logo next to "Categories" to create a new category. Click "New category." Make the title be "Blog Comments," make the icon be the speech_balloon, type whatever you want for "Description," such as the url for the blog and select "Open ended Discussion" and hit "Create."
- Now at the Discussion Category section at https://giscus.app/ you can select "Blog Comments."
- We can select "only search for discussions in this category" — when searching for a matching discussion, giscus will only search in this category.
After creating the category "Blog Comments" we can now use it to assign our environment variables:
GISCUS_CATEGORY="Blog Comments" which sets siteMetadata.comment.giscusConfig.category
If you scroll down the page at after seeing "Blog Comments" to the "Enable giscus" section, you'll see a value for data-category-id="SOME_VALUE"
which can use to set the environment variable:
GISCUS_CATEGORY_ID="SOME_VALUE" which sets siteMetadata.comment.giscusConfig.categoryId
At the Features section at https://giscus.app/ we choose whether specific features should be enabled. We can toggle on or off each of the following, but we'll just select the first one for now.
☑ Enable reactions for the main post — The reactions for the discussion's main post will be shown before the comments. We'll set this in:
siteMetadata.comment.giscusConfig.reactions = 1☐ Emit discussion metadata — Discussion metadata will be sent periodically to the parent window (the embedding page). For demonstration, enable this option and open your browser's console on this page. See the documentation for more details. We'll set this in:
siteMetadata.comment.giscusConfig.metadata = 0☐ Place the comment box above the comments — The comment input box will be placed above the comments, so that users can leave a comment without scrolling to the bottom of the discussion. We'll set this in:
siteMetadata.comment.giscusConfig.inputPosition = 'bottom'☐ Load the comments lazily — Loading of the comments will be deferred until the user scrolls near the comments container. This is done by adding
loading="lazy"
to the<iframe>
element. I'm not sure where this is set by our website.
At the Theme section at https://giscus.app/, we choose a theme that matches your website. I think we'll choose preferred_color_scheme but it doesn't matter since this will be set in
siteMetadata.comment.giscusConfig.theme = 'light'
siteMetadata.comment.giscusConfig.darkTheme = 'transparent_dark'
and possibly.
siteMetadata.comment.giscusConfig.themeURL = ''
At the Enable giscus section at https://giscus.app/, we add the following <script>
tag (with the square bracket values below with their appropriate values filled in for us) to our website's template where you want the comments to appear. If an element with the class giscus
exists, the comments will be placed there instead.
<script src="https://giscus.app/client.js"
data-repo="[ENTER REPO HERE]"
data-repo-id="[ENTER REPO ID HERE]"
data-category="[ENTER CATEGORY NAME HERE]"
data-category-id="[ENTER CATEGORY ID HERE]"
data-mapping="pathname"
data-reactions-enabled="1"
data-emit-metadata="0"
data-input-position="bottom"
data-theme="light"
data-lang="en"
crossorigin="anonymous"
async>
</script>
You can customize the container layout using the .giscus
and .giscus-frame
selectors from the embedding page.
Next add giscus to the content security policy in the next.config.js
file:
const ContentSecurityPolicy = `
default-src 'self';
script-src 'self' 'unsafe-eval' 'unsafe-inline' giscus.app;
style-src 'self' 'unsafe-inline';
img-src * blob: data:;
media-src 'none';
connect-src *;
font-src 'self';
frame-src giscus.app
`
The template now supports plausible, simple analytics and google analytics, and umami.
After configuring your site, <Analytics />
is loaded into pages/_app.js
, if you are not on the development server.
Custom events are also supported. You can import the logEvent
function from @components/analytics/[ANALYTICS-PROVIDER]
file and call it when triggering certain events of interest.
Note: Additional configuration might be required depending on the analytics provider, please check their official documentation for more information.
First, configure siteMetadata.js
with the settings that correspond with the desired analytics provider.
analytics: {
// supports plausible, simpleAnalytics or googleAnalytics
plausibleDataDomain: '', // e.g. next-markdown-journal.vercel.app
simpleAnalytics: false, // true or false
umamiWebsiteId: process.env.NEXT_PUBLIC_UMAMI_WEBSITE_ID, // e.g. 123e4567-e89b-12d3-a456-426614174000
googleAnalyticsId: process.env.NEXT_PUBLIC_GOOGLE_ANALYTICS_ID, // e.g. UA-000000-2 or G-XXXXXXX
},
The last two are set in .env*
such as .env.local
.
-
To get your googleAnalyticsId, go to Sign into Google Analytics On the top, to the left of the search field you should see:
All accounts > YOURSITE.COM
All Web Site DataIf 'YOURSITE.COM' is not the domain you're working on, click on 'All Web Site Data.' In this modal, you'll see your Analytics account o n the left pane. Select that and you'll see all of your domains in the middle pane. Here you should see the google Analytics Id for that domain.
You can get more details by select the domain then on the right pane you'll 'All Web Site Data'... click that.
You'll see a gear icon on the lower left. If you see 'Admin' to the right of it, click it to view the Admin details for that domain. -
If you want to use an analytics provider you have to add it to the content security policy in the
next.config.js
file. You would probably have to add www.googletagmanager.com and www.google-analytics.com if you are using google analytics. After that along withgiscus.app
added, it may look as shown below:const ContentSecurityPolicy = ` default-src 'self'; script-src 'self' 'unsafe-eval' 'unsafe-inline' giscus.app *.googletagmanager.com *.google-analytics.com; style-src 'self' 'unsafe-inline'; img-src * blob: data:; media-src 'none'; connect-src *; font-src 'self'; frame-src giscus.app `
The newsletter component gives you an easy way to build an audience. It integrates with the following providers:
To use it, specify the provider which you are using in the next.config.js
file and add the necessary environment variables to the .env*
file.
Two components are exported, a default NewsletterForm
and a BlogNewsletterForm
component, which is also passed in as an MDX component
and can be used in a blog post:
<BlogNewsletterForm title="Like what you are reading?" />
The component relies on nextjs's API routes which requires a server-side instance of nextjs to be setup and is not compatible with a 100% static site export. Users should either self-host or use a compatible platform like Vercel or Netlify which supports this functionality.
A static site compatible alternative is to substitute the route in the newsletter component with a form API endpoint provider.
Go to Buttondown and sign up. To set up our we only need these three environment variables:
BUTTONDOWN_API_URL
, which should behttps://api.buttondown.email/v1/
and theBUTTONDOWN_API_KEY
. After being signed in, click the three horizontal lines button on the upper right.
Here is a quick overview to important items in the menu:
- Settings
- Programming is where your API Key (
BUTTONDOWN_API_KEY
) is found, which can be regenerated. You can also set Webhooks, and view Events and Requests. - Your newsletter is where you Newsletter name, Newsletter description, and other things are set.
- Account you can set the Your name ('From' field in emails from you, which defaults to be the Newsletter name) and Your address, which is blank by default but t's a legal requirement for newsletters to contain a physical address.
- Settings is where you can change your Username, Newsletter name and Newsletter description, among other things.
- Subscribing is where you can modify how people sign up (confirmation and welcome emails, etc).
- Programming is where your API Key (
Sometimes email confirmations from Buttondown are identified as spam by Gmail and possibly others. Here is advice from Buttondown on this.
The easiest way to deploy the template is to use the Vercel Platform from the creators of Next.js. Check out the Next.js deployment documentation for more details.
When deploying a Next.js app to Vercel, all types of Environment Variables should be configured in the Project Settings, even Environment Variables used in Development – which can then be downloaded onto your local device. If you've configured Development Environment Variables you can pull them into .env.local
for use on your local machine with the following command:
vercel env pull .env.local
When using the Vercel CLI to deploy make sure you add a .vercelignore
that includes files that should not be uploaded, generally these are the same files included in .gitignore
.
You may want to refer to this guide: Deploying Next.js to Vercel with Environment Variables - YouTube.
If you don't see the changes to the deployed app, it's probably because you are not looking at the particular deployment that has the environmental variables loaded, even if that is the most recent one because the most recent one is not necessarily the "current" one.
As the template uses next/image
for image optimization, additional configurations have to be made to deploy on other popular static hosting websites like Netlify or GitHub Pages. An alternative image optimization provider such as Imgix, Cloudinary or Akamai has to be used. Alternatively, replace the next/image
component with a standard <img>
tag. See next/image
documentation for more details.
The API routes used in the newsletter component cannot be used in a static site export. You will need to use a form API endpoint provider and substitute the route in the newsletter component accordingly. Other hosting platforms such as Netlify also offer alternative solutions - please refer to their docs for more information.
PRs accepted.
Small note: If editing the README, please conform to the standard-readme specification.
MIT © 2022 Jeff Russ