Let's start with a clean slate e.g. in our home directory e.g. in a folder named ~/CAHTFS/
.
Its name dosen't matter and to keep things consistent and easy to replay,
all paths are relative too this folder and all comands are executes sitting in that folder, too.
westenfalke@MiniDEV:~$ mkdir CAHTFS
westenfalke@MiniDEV:~$ cd CAHTFS/
westenfalke@MiniDEV:~/CAHTFS$
and westenfalke@MiniDEV:~/CAHTFS$
is replaced with just #
just like this:
westenfalke@MiniDEV:~/CAHTFS$ ls -la
total 8
drwxr-xr-x 2 westenfalke westenfalke 4096 Jan 2 15:52 .
drwxr-xr-x 11 westenfalke westenfalke 4096 Jan 2 15:52 ..
# ls -la
total 8
drwxr-xr-x 2 westenfalke westenfalke 4096 Jan 2 15:52 .
drwxr-xr-x 11 westenfalke westenfalke 4096 Jan 2 15:52 ..
hugo
the start of this show
# sudo apt install hugo
To display ascii style information regarding the files and the folder structure,
I introduce the tree
command to for sake of this tutorial.
# sudo apt install tree
I like git
to do the version controll thingi here,
if version control is not in you scope you can start
here Creating A HUGO Theme From Scratch
# sudo apt install git
Hence I like to see you participating in this project I'll publish it on GitHUB via CLI using the gh
command.
# sudo apt-key adv --keyserver keyserver.ubuntu.com --recv-key C99B11DEB97541F0
# sudo apt-add-repository https://cli.github.com/packages
# sudo apt update
# sudo apt install gh
This prepatations are optional, but for convinience I like to have of site version a control in place
# gh auth login
# gh auth login
? What account do you want to log into? [Use arrows to move, type to filter]
> GitHub.com
GitHub Enterprise Server
? What account do you want to log into? GitHub.com
- Logging into github.com
? You're already logged into github.com as westenfalke. Do you want to re-authenticate? (y/N) Y
? How would you like to authenticate? [Use arrows to move, type to filter]
> Login with a web browser
Paste an authentication token
! First copy your one-time code: nXlm-kijo
- Press Enter to open github.com in your browser...
✓ Authentication complete. Press Enter to continue...
? Choose default git protocol [Use arrows to move, type to filter]
> HTTPS
SSH
- gh config set -h github.com git_protocol https
✓ Configured git protocol
✓ Logged in as westenfalke
# gh repo create onomy-hugo-theme
? Visibility Public
? This will create 'onomy-hugo-theme' in your current directory. Continue? Yes
✓ Created repository westenfalke/onomy-hugo-theme on GitHub
? Create a local project directory for westenfalke/onomy-hugo-theme? (Y/n) n
? Create a local project directory for westenfalke/onomy-hugo-theme? No
# git clone https://github.com/westenfalke/onomy-hugo-theme.git themes/onomy
Cloning into 'themes/onomy'...
warning: You appear to have cloned an empty repository.
Here is the the real thing and from now on all commands are relative to the ~/CAHTFS
.
# hugo new theme onomy
Creating theme at /home/westenfalke/CAHTFS/themes/onomy
# tree
.
├── resources
│ └── _gen
│ ├── assets
│ └── images
└── themes
└── onomy
├── LICENSE
├── archetypes
│ └── default.md
├── layouts
│ ├── 404.html
│ ├── _default
│ │ ├── baseof.html
│ │ ├── list.html
│ │ └── single.html
│ ├── index.html
│ └── partials
│ ├── footer.html
│ ├── head.html
│ └── header.html
├── static
│ ├── css
│ └── js
└── theme.toml
13 directories, 11 files
While creating the theme hugo
creates some file in the resources/
folder we no longer need.
# rm -rf resources/
The example website in the theme will be used to sowcase off its features and will contains this tutorial as a reference implementaion.
# hugo new site exampleSite
Congratulations! Your new Hugo site is created in /home/westenfalke/CAHTFS/themes/onomy/exampleSite.
Just a few more steps and you're ready to go:
1. Download a theme into the same-named folder.
Choose a theme from https://themes.gohugo.io/ or
create your own with the "hugo new theme <THEMENAME>" command.
2. Perhaps you want to add some content. You can add single files
with "hugo new <SECTIONNAME>/<FILENAME>.<FORMAT>".
3. Start the built-in live server via "hugo server".
Visit https://gohugo.io/ for quickstart guide and full documentation.
# tree themes/onomy/exampleSite/
themes/onomy/exampleSite/
├── archetypes
│ └── default.md
├── config.toml
├── content
├── data
├── layouts
├── static
└── themes
Currently HUGO creates a LICENSE file using "The MIT License (MIT)" as default.
You can change this e.g. by using the CC License Chooser, or create a LICENCE file right now on GitHUB, hence here is a standard suggestion (on empty repositories) which offers several different licenses during file creation.
link to a page about LICENSE selection on gitgub link to a page about LICENSE selection with CC License Chooser
If you are cool with "The MIT License (MIT)" just keep it, I highly recommand to do this vi GiuHub.
# cat themes/onomy/LICENSE
MIT License
Copyright (c) 2021 Stefan Welke
Permission is hereby granted, free of charge, to any person obtaining a copy
of this software and associated documentation files (the "Software"), to deal
in the Software without restriction, including without limitation the rights
to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
copies of the Software, and to permit persons to whom the Software is
furnished to do so, subject to the following conditions:
The above copyright notice and this permission notice shall be included in all
copies or substantial portions of the Software.
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
SOFTWARE.
# touch themes/onomy/README.md
# echo "# Creating A HUGO Theme westenfalke/onomy-hugo-theme From Scratch" >> themes/onomy/README.md
Hence the content of this theme is straigt from the hugo documentation I'll skip the original
section and you can find a list auf websites or authors who enabled/inpired me to write this tutorial in the credits section of this README.
# cat themes/onomy/theme.toml
### theme.toml template for a Hugo theme
### See https://github.com/gohugoio/hugoThemes#themetoml for an example
name = "onomy"
license = "MIT"
licenselink = "https://github.com/westenfalke/onomy-hugo-theme/blob/master/LICENSE"
description = "onomy (hugo taxonomy template)"
homepage = "https://github.com/westenfalke/onomy-hugo-theme"
tags = [gdpr selfcontaint learning tutorial]
features = [taxonomy pagination cookie-free]
min_version = "0.41"
[author]
name = "westenfalke"
homepage = "https://github.com/westenfalke"
### If porting an existing theme
###[original]
### name = ""
### homepage = ""
### repo = ""
One could think, this is it, and yes firing up hugo right now will give us a working sitemap-xml
and three empty pages and hence content creation is our task, this is pretty much what to expect.
# hugo server --source ./themes/onomy/exampleSite/ --themesDir ../../../themes/ --theme onomy
| EN
-------------------+-----
Pages | 3
Paginator pages | 0
Non-page files | 0
Static files | 0
Processed images | 0
Aliases | 0
Sitemaps | 1
Cleaned | 0
Built in 3 ms
Watching for changes in /home/westenfalke/CAHTFS/themes/onomy/{archetypes,exampleSite,layouts,static}
Watching for config changes in themes/onomy/exampleSite/config.toml
Environment: "development"
Serving pages from memory
Running in Fast Render Mode. For full rebuilds on change: hugo server --disableFastRender
Web Server is available at http://localhost:1313/ (bind address 127.0.0.1)
Press Ctrl+C to stop
http://localhost:1313/sitemap.xml
<?xml version="1.0" encoding="utf-8" standalone="yes"?>
<urlset xmlns="http://www.sitemaps.org/schemas/sitemap/0.9"
xmlns:xhtml="http://www.w3.org/1999/xhtml">
<url>
<loc>http://localhost:1313/categories/</loc>
</url>
<url>
<loc>http://localhost:1313/</loc>
</url>
<url>
<loc>http://localhost:1313/tags/</loc>
</url>
</urlset>
hugo
creates all templates for us and besites the baseof template, all are empty.
Before we can see some thing on our new created theme, by running the exampleSite with hugo
, we'll simply copy and paste the stuff fom the hugo template documentation into the templates.
# tree -s themes/onomy/layouts/
themes/onomy/layouts/
├── [ 0] 404.html
├── [ ] _default
│ ├── [250] baseof.html
│ ├── [ 0] list.html
│ └── [ 0] single.html
├── [ 0] index.html
└── [ ] partials
├── [ 0] footer.html
├── [ 0] head.html
└── [ 0] header.html
2 directories, 8 files
Put Hugo Templates Homepage into themes/onomy/layouts/index.html
# cat themes/onomy/layouts/index.html
{{ define "main" }}
<main aria-role="main">
<header class="homepage-header">
<h1>{{.Title}}</h1>
{{ with .Params.subtitle }}
<span class="subtitle">{{.}}</span>
{{ end }}
</header>
<div class="homepage-content">
<!-- Note that the content for index.html, as a sort of list page, will pull from content/_index.md -->
{{.Content}}
</div>
<div>
{{ range first 10 .Site.RegularPages }}
{{ .Render "summary"}}
{{ end }}
</div>
</main>
{{ end }}
Put Hugo Templates List into themes/onomy/layouts/_default/list.html
# cat themes/onomy/layouts/_default/list.html
{{ define "main" }}
<main>
<article>
<header>
<h1>{{.Title}}</h1>
</header>
<!-- "{{.Content}}" pulls from the markdown content of the corresponding _index.md -->
{{.Content}}
</article>
<ul>
<!-- Ranges through content/posts/*.md -->
{{ range .Pages }}
<li>
<a href="{{.Permalink}}">{{.Date.Format "2006-01-02"}} | {{.Title}}</a>
</li>
{{ end }}
</ul>
</main>
{{ end }}
Put Hugo Single Page Templates into themes/onomy/layouts/_default/single.html
# cat themes/onomy/layouts/_default/single.html
{{ define "main" }}
<section id="main">
<h1 id="title">{{ .Title }}</h1>
<div>
<article id="content">
{{ .Content }}
</article>
</div>
</section>
<aside id="meta">
<div>
<section>
<h4 id="date"> {{ .Date.Format "Mon Jan 2, 2006" }} </h4>
<h5 id="wordcount"> {{ .WordCount }} Words </h5>
</section>
{{ with .Params.topics }}
<ul id="topics">
{{ range . }}
<li><a href="{{ "topics" | absURL}}{{ . | urlize }}">{{ . }}</a> </li>
{{ end }}
</ul>
{{ end }}
{{ with .Params.tags }}
<ul id="tags">
{{ range . }}
<li> <a href="{{ "tags" | absURL }}{{ . | urlize }}">{{ . }}</a> </li>
{{ end }}
</ul>
{{ end }}
</div>
<div>
{{ with .PrevInSection }}
<a class="previous" href="{{.Permalink}}"> {{.Title}}</a>
{{ end }}
{{ with .NextInSection }}
<a class="next" href="{{.Permalink}}"> {{.Title}}</a>
{{ end }}
</div>
</aside>
For the hompage template alias index.html
to work properly, we have to add one additional template themes/onomy/layouts/_default/summary.html
to the mix, just to satify the referenced {{ .Render "summary"}}
.
# touch themes/onomy/layouts/_default/summary.html
Put HUGO Content View Templates into themes/onomy/layouts/_default/summary.html
# cat themes/onomy/layouts/_default/summary.html
<article class="post">
<header>
<h2><a href='{{ .Permalink }}'> {{ .Title }}</a> </h2>
<div class="post-meta">{{ .Date.Format "Mon, Jan 2, 2006" }} - {{ .FuzzyWordCount }} Words </div>
</header>
{{ .Summary }}
<footer>
<a href='{{ .Permalink }}'><nobr>Read more →</nobr></a>
</footer>
</article>
I suggest a hompage, two pages (an imprint and a privacy policy) and three posts are enough for starters and to test the pagination, lists and the summary.
This is where the magic happens, than this file triggers the invocation of the index.html
template which renered by the baseof.html
.
# hugo new "_index.md" --source themes/onomy/exampleSite/
/home/westenfalke/CAHTFS/themes/onomy/exampleSite/content/_index.md created
# hugo new "imprint.md" --source themes/onomy/exampleSite/
/home/westenfalke/CAHTFS/themes/onomy/exampleSite/content/imprint.md created
# hugo new "privacy policy.md" --source themes/onomy/exampleSite/
/home/westenfalke/CAHTFS/themes/onomy/exampleSite/content/privacy policy.md created
#hugo new "posts/the first post.md" --source themes/onomy/exampleSite/
/home/westenfalke/CAHTFS/themes/onomy/exampleSite/content/posts/the first post.md created
# hugo new "posts/the second post.md" --source themes/onomy/exampleSite/
/home/westenfalke/CAHTFS/themes/onomy/exampleSite/content/posts/the second post.md created
# hugo new "posts/the third post.md" --source themes/onomy/exampleSite/
/home/westenfalke/CAHTFS/themes/onomy/exampleSite/content/posts/the third post.md created
Hence this all of our documents are drafts, the flag --buildDrafts
will do the trick of rendering all the drafts for the moment, so there is no need to change our newly created content yet.
# hugo server --source ./themes/onomy/exampleSite/ --themesDir ../../../themes/ --theme onomy --buildDrafts
| EN
-------------------+-----
Pages | 13
Paginator pages | 0
Non-page files | 0
Static files | 0
Processed images | 0
Aliases | 0
Sitemaps | 1
Cleaned | 0
Built in 18 ms
Watching for changes in /home/westenfalke/CAHTFS/themes/onomy/{archetypes,exampleSite,layouts,static}
Watching for config changes in themes/onomy/exampleSite/config.toml
Environment: "development"
Serving pages from memory
Running in Fast Render Mode. For full rebuilds on change: hugo server --disableFastRender
Web Server is available at http://localhost:1313/ (bind address 127.0.0.1)
Press Ctrl+C to stop
I'm rather keen to see the http://localhost:1313/sitemap.xml and than have a look at http://localhost:1313/.
<?xml version="1.0" encoding="utf-8" standalone="yes"?>
<urlset xmlns="http://www.sitemaps.org/schemas/sitemap/0.9"
xmlns:xhtml="http://www.w3.org/1999/xhtml">
<url>
<loc>http://localhost:1313/posts/</loc>
<lastmod>2021-01-02T23:30:43+01:00</lastmod>
</url>
<url>
<loc>http://localhost:1313/posts/the-third-post/</loc>
<lastmod>2021-01-02T23:30:43+01:00</lastmod>
</url>
<url>
<loc>http://localhost:1313/posts/the-second-post/</loc>
<lastmod>2021-01-02T23:30:18+01:00</lastmod>
</url>
<url>
<loc>http://localhost:1313/posts/the-first-post/</loc>
<lastmod>2021-01-02T23:29:56+01:00</lastmod>
</url>
<url>
<loc>http://localhost:1313/privacy-policy/</loc>
<lastmod>2021-01-02T23:29:14+01:00</lastmod>
</url>
<url>
<loc>http://localhost:1313/imprint/</loc>
<lastmod>2021-01-02T23:28:55+01:00</lastmod>
</url>
<url>
<loc>http://localhost:1313/</loc>
<lastmod>2021-01-02T23:15:42+01:00</lastmod>
</url>
<url>
<loc>http://localhost:1313/categories/</loc>
</url>
<url>
<loc>http://localhost:1313/readme/</loc>
</url>
<url>
<loc>http://localhost:1313/tags/</loc>
</url>
</urlset>
Just to proofe the above assumtion about the invocation of the index.html
template which renders the baseof.html
right and to leafe no templates empty (except themes/onomy/layouts/404.html
), make this few little upates.
# echo '{{ "<!-- " | safeHTML }}head.html{{ " -->" | safeHTML }}<head><meta charset="utf-8"></head>' >> themes/onomy/layouts/partials/head.html
# echo '<div>FOOTER</div>' >> themes/onomy/layouts/partials/footer.html
# echo '<div>HEADER</div>' >> themes/onomy/layouts/partials/header.html
Before any further changes I'm going to commit this piece of work to git
with some specific preparation, you can skip if you are not invested in version control jump direcly to Second Half OF Creating A HUGO Theme From Scratch
Telling git to ignore some files and folder comcleatly.
# touch themes/onomy/.gitignore
# echo ".gitignore
**/resorces
**/public
**/public_html
**/*.ORI" >> touch themes/onomy/.gitignore
Finding and adding all files we creates so far witout the meta data in .git
# find . -type f | grep -v '.git' | xargs -n1 -d'\n' echo | cut -d'/' -f 4- |xargs -d'\n' -n1 git --work-tree=themes/onomy/ --git-dir=themes/onomy/.git/ add
# git --work-tree=themes/onomy/ --git-dir=themes/onomy/.git add -f .gitignore
# git --work-tree=themes/onomy/ --git-dir=themes/onomy/.git/ status
On branch master
No commits yet
Changes to be committed:
(use "git rm --cached <file>..." to unstage)
new file: .gitignore
new file: LICENSE
new file: README.md
new file: archetypes/default.md
new file: exampleSite/archetypes/default.md
new file: exampleSite/config.toml
new file: exampleSite/content/README.md
new file: exampleSite/content/_index.md
new file: exampleSite/content/imprint.md
new file: exampleSite/content/posts/the first post.md
new file: exampleSite/content/posts/the second post.md
new file: exampleSite/content/posts/the third post.md
new file: exampleSite/content/privacy policy.md
new file: exampleSite/static/img/screenshot/hugo-parameter.png
new file: layouts/404.html
new file: layouts/_default/baseof.html
new file: layouts/_default/list.html
new file: layouts/_default/single.html
new file: layouts/_default/summary.html
new file: layouts/index.html
new file: layouts/partials/footer.html
new file: layouts/partials/head.html
new file: layouts/partials/header.html
new file: static/img/screenshot/hugo-parameter.png
new file: theme.toml
# git --work-tree=themes/onomy/ --git-dir=themes/onomy/.git commit -m'initial configuration, README, LICENSE and templates'
[master (root-commit) d8101fb] initial configuration, README, LICENSE and templates
25 files changed, 1605 insertions(+)
create mode 100644 .gitignore
create mode 100644 LICENSE
create mode 100644 README.md
create mode 100644 archetypes/default.md
create mode 100644 exampleSite/archetypes/default.md
create mode 100644 exampleSite/config.toml
create mode 100644 exampleSite/content/README.md
create mode 100644 exampleSite/content/_index.md
create mode 100644 exampleSite/content/imprint.md
create mode 100644 exampleSite/content/posts/the first post.md
create mode 100644 exampleSite/content/posts/the second post.md
create mode 100644 exampleSite/content/posts/the third post.md
create mode 100644 exampleSite/content/privacy policy.md
create mode 100644 exampleSite/static/img/screenshot/hugo-parameter.png
create mode 100644 layouts/404.html
create mode 100644 layouts/_default/baseof.html
create mode 100644 layouts/_default/list.html
create mode 100644 layouts/_default/single.html
create mode 100644 layouts/_default/summary.html
create mode 100644 layouts/index.html
create mode 100644 layouts/partials/footer.html
create mode 100644 layouts/partials/head.html
create mode 100644 layouts/partials/header.html
create mode 100644 static/img/screenshot/hugo-parameter.png
create mode 100644 theme.toml
# git --work-tree=themes/onomy/ --git-dir=themes/onomy/.git push
Enumerating objects: 37, done.
Counting objects: 100% (37/37), done.
Delta compression using up to 4 threads
Compressing objects: 100% (28/28), done.
Writing objects: 100% (37/37), 20.80 KiB | 1.39 MiB/s, done.
Total 37 (delta 1), reused 0 (delta 0)
remote: Resolving deltas: 100% (1/1), done.
To https://github.com/westenfalke/onomy-hugo-theme.git
* [new branch] master -> master
- add tanonomy templates
- link to a page about LICENSE selection on gitgub
- link to a page about LICENSE selection with CC License Chooser
- Installing gh on Linux and FreeBSD by Mislav Marohnić
- Git Submodules basic explanation by Rik Gitaarik
- Markdown: Syntax by John Gruber
- The hugo-sandbox by Kaushal Modi
- GitHUB distributed version control system
- Git distributed version control system