title | tags | public | type | date | ||
---|---|---|---|---|---|---|
Hugo (Go) |
|
true |
note |
2021-06-22 |
A static-site generator written in Golang.
brew install hugo
hugo version
Hugo Static Site Generator v0.80.0/extended darwin/amd64 BuildDate: unknown
hugo new site <folder-name>
ls -1
archetypes/
config.toml
content/
data/
layouts/
static/
themes/
- archetypes/ - entities that are smth like markdown templates of content
- config.toml - main configuration file
- content/ - folder that contains markdown content
- data/ - folder that contains configuration files and data that used in generating your site
- layout/ - folder that contains html templates for specific parts of website
- static/ - folder that contains static files like images, fonts, etc.
- themes/ - folder that can contain existing Hugo themes
hugo server
To running and see draft content:
hugo server -D
The empty white page in just created app is okay
On root route (/) Hugo try to render layout with name:
- list.html
- or
- index.html (index.html takes priority)
Types of pages:
- single (page of the specific post)
- list (page that contains list of posts)
By default it is config.toml
It is possible to use YAML file. Just rename it to config.yml
Default settings:
config.toml
baseURL = "http://example.org/"
languageCode = "en-us"
title = "My New Hugo Site"
config.yml
baseURL: "http://example.org/"
languageCode: "en-us"
title: "My New Hugo Site"
Create template of the main page in path /layouts/list.html
<!DOCTYPE html>
<html lang="en">
<head>
<meta charset="UTF-8" />
<meta http-equiv="X-UA-Compatible" content="IE=edge" />
<meta name="viewport" content="width=device-width, initial-scale=1.0" />
<title>{{ .Site.Title }}</title>
</head>
<body></body>
</html>
.Site.Title:
- Site - context
- it is context of config.yml file
- Title - variable
Create file /content/_index.md
---
title: "Home"
---
- _index.md - file that contains content of home page (root route - '/')
Add title of home page in /layouts/list.html:
<!DOCTYPE html>
<html lang="en">
<head>
<meta charset="UTF-8" />
<meta http-equiv="X-UA-Compatible" content="IE=edge" />
<meta name="viewport" content="width=device-width, initial-scale=1.0" />
<title>{{ .Site.Title }}</title>
<!-- Title of website -->
</head>
<body>
<h1>{{ .Page.Title }}</h1>
<!-- Title of the page -->
</body>
</html>
Because the layout is already in context of the page you can just change it:
<h1>{{ .Page.Title }}</h1>
<!-- Title of the page -->
to
<h1>{{ .Title }}</h1>
<!-- Title of the page -->
Create new pages /content/about.md and /content/contact.md
---
title: "About"
---
To render list of these pages with their links just add this code in layouts/list.html:
<ul>
{{ range .Pages }}
<li><a href="{{ .Permalink }}">{{ .Title }}</a></li>
{{ end }}
</ul>
- range is a loop here. it iterates in .Pages which is a list of data about each page
- .Permalink - link to the specific page
- .Title - the title of the page which is declared in markdown metadata
Create folder /layouts/_default and move layouts/list.html there
Layouts from this folder /layouts/_default will be used for all content by default
Create file /layouts/_default/single.html:
<!DOCTYPE html>
<html lang="en">
<head>
<meta charset="UTF-8" />
<meta http-equiv="X-UA-Compatible" content="IE=edge" />
<meta name="viewport" content="width=device-width, initial-scale=1.0" />
<title>{{ .Title }} | {{ .Site.Title }}</title>
</head>
<body>
<h1>{{ .Title }}</h1>
<div>{{ .Content }}</div>
</body>
</html>
If you open page /about in the browser you'll see this html structure.
Change the loop in the layout to display list of resources that have type "page":
<ul>
{{ range where .Pages "Type" "page" }}
<li><a href="{{ .Permalink }}">{{ .Title }}</a></li>
{{ end }}
</ul>
By default markdown resources in /content has type "page".
If you don't want to show some specific resource as page you can specify type in metadata:
---
title: "About"
type: "post"
---
More information you can find in official doc of Hugo
I understand sections like sub folders in a folder content
Basically, you have a folder content that contains two folders - blog and notes.
In that case, blog and notes are Sections in Hugo.
{{ range where .Site.RegularPages "Section" "blog" }}
<li>
<a href="{{ .Permalink }}">{{ .Title }}</a>
</li>
{{ end }}
<div>
{{ $paginator := .Paginate (where .Pages "Section" "blog") }} {{ range
$paginator.Pages }}
<article>
<div>
<a href="{{ .Permalink }}">{{ .Title }}</a>
</div>
</article>
{{ end }} {{ template "_internal/pagination.html" . }}
</div>
Let's say we have two layouts: for home page layouts/_default/list.html and for single page layouts/_default/single.html
They have pretty similar html structure:
list.html
<!DOCTYPE html>
<html lang="en">
<head>
<meta charset="UTF-8" />
<meta http-equiv="X-UA-Compatible" content="IE=edge" />
<meta name="viewport" content="width=device-width, initial-scale=1.0" />
<title>{{ .Site.Title }}</title>
<!-- Title of website -->
</head>
<body>
<h1>{{ .Title }}</h1>
<div>
<ul>
{{ range where .Pages "Type" "page" }}
<li><a href="{{ .Permalink }}">{{ .Title }}</a></li>
{{ end }}
</ul>
</div>
</body>
</html>
single.html
<!DOCTYPE html>
<html lang="en">
<head>
<meta charset="UTF-8" />
<meta http-equiv="X-UA-Compatible" content="IE=edge" />
<meta name="viewport" content="width=device-width, initial-scale=1.0" />
<title>{{ .Title }} | {{ .Site.Title }}</title>
</head>
<body>
<h1>{{ .Title }}</h1>
<div>{{ .Content }}</div>
</body>
</html>
To avoid repeating code we can make base layout that will contain common html structure.
Create the file /layouts/_default/baseof.html
<!DOCTYPE html>
<html lang="en">
<head>
<meta charset="UTF-8" />
<meta http-equiv="X-UA-Compatible" content="IE=edge" />
<meta name="viewport" content="width=device-width, initial-scale=1.0" />
<title>{{ .Site.Title }}</title>
<!-- Title of website -->
</head>
<body>
<h1>{{ .Site.Title }}</h1>
<div>
<ul>
{{ range where .Site.RegularPages "Type" "page" }}
<li><a href="{{ .Permalink }}">{{ .Title }}</a></li>
{{ end }}
</ul>
</div>
{{ block "content" . }}{{ end }}
</body>
</html>
{{ block "content" . }}{{ end }}
renders layouts that have the same "name"
For home page let's change /layouts/_default/list.html:
{{ define "content" }}
<h2>Index</h2>
{{ end }}
And now the baseof layout renders the sub layout list.html for home page
For single page let's change /layouts/_default/single.html:
{{ define "content"}}
<h2>{{ .Title }}</h2>
<div>{{ .Content }}</div>
{{ end }}
Create folder /content/blog and put there new file /content/blog/_index.md
---
title: "Blog"
---
And then create a few blank posts:
---
title: "First Post"
---
Deserunt duis fugiat reprehenderit officia cillum qui cillum Lorem. Aliquip sint esse fugiat elit. In incididunt quis ullamco cupidatat tempor. Lorem ad non incididunt ad magna nostrud.
---
title: "Second Post"
---
...
To display list of posts let's create new layout. But first of all, we need to create new folder /layouts/blog.
Why?
- /layouts/_default could used for general resources like pages
- /layouts/blog here are layouts for resources like blog posts
In /layouts/blog create file list.html:
{{ define "content" }}
<div>
<h2>{{ .Title }}</h2>
<ul>
{{ range where .Site.RegularPages "Type" "blog" }}
<li>
<a href="{{ .Permalink }}">{{ .Title }}</a>
</li>
{{ end }}
</ul>
</div>
{{ end }}
It will display list of posts links
Let's create a layout for page of post
In folder /layouts/blog create file single.html:
{{ define "content" }}
<div>
<h2>Post: {{ .Title }}</h2>
<div>{{ .Content }}</div>
</div>
{{ end }}
This html structure will be used only for posts. It won't be used for all pages (for example about or contact).
Partials templates are specific layouts or files that can be used as parts of layout.
Let's say we have file /layouts/_default/baseof.html:
<!DOCTYPE html>
<html lang="en">
<head>
<meta charset="UTF-8" />
<meta http-equiv="X-UA-Compatible" content="IE=edge" />
<meta name="viewport" content="width=device-width, initial-scale=1.0" />
<title>{{ .Site.Title }}</title>
<!-- Title of website -->
</head>
<body>
<!-- something here -->
</body>
</html>
Here we have section head that we could make it as partial template.
Create folder /layouts/partials. It is a specific folder for all partials.
In that new folder let's create file head.html with head section of html structure:
<head>
<meta charset="UTF-8" />
<meta http-equiv="X-UA-Compatible" content="IE=edge" />
<meta name="viewport" content="width=device-width, initial-scale=1.0" />
<title>{{ .Site.Title }}</title>
<!-- Title of website -->
</head>
To use this partial change the file baseof.html:
<!DOCTYPE html>
<html lang="en">
<!-- using partial with section "head" -->
{{ partial "head" . }}
<body>
<!-- something here -->
</body>
</html>
More information here (gohugo.io)
Basically, it's something similar to components in MDX but without React/JSX.
You can put a specific code in your markdown file and it will be rendered like a widget.
There are several built-in shortcodes in Hugo that can be used out of box.
For example, shortcode youtube that renders a video on the page:
{{< youtube w7Ft2ymGmfc >}}
Basically, taxonomies are tags and categories or some new thing that (you can make it) has similar behavior to tags or categories.
I take it that way:
- Sections: One-to-many
- one section (blog)
- many resources
- Taxonomies: Many-to-many
- many taxonomies (tags)
- many resources
Let's call it technologies
First of all, we need to specify new taxonomy in config file:
baseURL: "http://example.org/"
languageCode: "en-us"
title: "My New Hugo Site"
taxonomies:
category: categories
tag: tags
technology: technologies
notice that we need to specify built-in taxonomies too if you want to use them
Add this in some of your markdown resources:
---
title: "Second Post"
technologies: ["typescript", "react", "node"]
---
...
To render the list of technologies in each post:
<div>
{{ $paginator := .Paginate (where .Pages "Section" "blog") }} {{ range
$paginator.Pages }}
<article>
<!-- ... -->
<div>
{{ range (.GetTerms "technologies") }}
<li><a href="{{ .Permalink }}">#{{ .LinkTitle }}</a></li>
{{ end }}
</div>
<!-- ... -->
</article>
{{ end }} {{ template "_internal/pagination.html" . }}
</div>
To render the list of posts that has specific technology:
- file: /layouts/taxonomy/technology.html
- in the browser: /technologies/typescript
{{ define "content" }}
<div>
<h2>Technology: {{ .Title }}</h2>
<ul>
{{ range .Data.Pages }}
<li>
<a href="{{.RelPermalink}}">{{ .Title }}</a>
</li>
{{ end }}
</ul>
</div>
{{ end }}
Tags in Hugo are built-in taxonomies.
<!-- /content/blog/First Post.md -->
---
title: "First Post"
draft: false
tags: ["tag1", "tag2"]
---
....
- file: /layouts/tag/list.html
- in the browser: /tags
{{ define "content" }}
<div>
<h2>{{ .Title }}</h2>
<ul>
{{ range .Site.Taxonomies.tags }}
<li><a href="{{ .Page.Permalink }}">{{ .Page.Title }}</a> {{ .Count }}</li>
{{ end }}
</ul>
</div>
{{ end }}
- file: /layouts/taxonomy/tag.html
- in the browser: /tags/<tag_name>
{{ define "content" }}
<div>
<h2>#{{ .Title }}</h2>
<ul>
{{ range .Data.Pages }}
<li>
<a href="{{.RelPermalink}}">{{ .Title }}</a>
</li>
{{ end }}
</ul>
</div>
{{ end }}