HUGO Relearn Theme Customization
The number of Themes has been increased rapidly over time and since the very beginning of HUGO - a static web site generator.
The HUGO Relearn theme has one outstanding feature as it fully supports file mode, which means that there is no web server required to get access to the web site content.
Once a site has been generated, it is only required to double click on
the main <my-user-directory>\public\index.html
file.
All dependent files are served from the file system via the file://
protocol
which is supported by all modern browsers. Dependencies are relative to the main root
and so there is also no problem with CORS policy whatsoever.
When rendering a site:
- the BaseUrl which usually is the web server address name
(like e.g.
https://johann-oberdorfer.eu
) - is replaced by
./
- and subsequently all articles are represented by a trailing
index.html
With this nomenclature in place, there is no name resolving needed (which usually is done by a web server). In the underlying theme template files, relative notation for urls is used.
Beside other great features, for me this was one of the main reasons for choosing this theme (to build a local documentation system).
Use cases
The theme claims to be a “documentation” theme. In this respect it fits perfectly to build a personal or a local documentation system (intranet) for example.
Benefits:
Personal or company information is pure text (markdown) and thus future prove, stored locally without the need of a database and ready for serverless operation.
Markdown editing (no proprietary data and software required).
File structure in the content directory is used to build up the site navigation menu, a really nice feature most useful for documentations etc,…
Build in search functionality.
Text and layout is separated (except for the shortcodes used, they need to be embedded in markdown).
Easily maintainable and expandable (if required).
Attachments of existing PDF’s can be added to an article quite easy.
Print capability to print one article, chapters or even the whole site articles at once.
Overall, a modern way to edit, manage and present information.
Theme Customizations
To make the theme even more useful I added some more functionality to it.
Functions added so far:
Categories and Tags Cloud
{{/*
--- shortcode/widgets/tag_cloud.html
Credits:
This code was inspired by the academic theme.
Thanks to: George Cushen
Purpose:
This function does not take any arguments into account.
Read all "tags" and "categories" from the site's taxonomies
and create a test cloud which can be used to navigate the site.
(c) 2020, Johann Oberdorfer - Engineering Support | CAD | Software
www.johann-oberdorfer.eu
This source file is distributed under the MIT license.
This program is distributed in the hope that it will be useful,
but WITHOUT ANY WARRANTY; without even the implied warranty of
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.
See the MIT License for more details.
*/}}
{{ $read_tags := true }}
{{ $fontSmall := .Page.Params.design.font_size_min | default 0.9 }}
{{ $fontBig := .Page.Params.design.font_size_max | default 1.5 }}
{{ $fontDelta := sub $fontBig $fontSmall }}
{{/* read "categories"... */}}
{{ $data := .Site.Taxonomies.categories.ByCount }}
{{/* read "tags"
and add this information to the data array... */}}
{{ if $read_tags }}
{{ $tags := .Site.Taxonomies.tags.ByCount }}
{{ if ne (len $tags) 0 }}
{{ $data = $data | append $tags }}
{{ end }}
{{ end }}
{{/* output goes here... */}}
{{ if ne (len $data) 0 }}
<style>
.tag-cloud-tags {
display: block;
margin: 0;
list-style-type: none;
}
.tag-cloud-tags a {
display: inline-block;
padding: 0.3rem 0.3rem 0rem 0.3rem;
line-height: 1em;
word-break: break-word;
white-space: normal;
text-decoration: none;
border-bottom: 1px solid LightGrey;
}
.tag-cloud-tags a:nth-child(5n+1) { transform: rotate(-2deg); }
.tag-cloud-tags a:nth-child(3n+3) { transform: rotate(1deg); }
.tag-cloud-tags a:nth-child(3n+4) { transform: rotate(-3deg); }
.tag-cloud-tags a:nth-child(5n+5) { transform: rotate(3deg); }
/* underline animation from left */
.hover-animate-from-left {
display: inline-block;
vertical-align: middle;
-webkit-transform: perspective(1px) translateZ(0);
transform: perspective(1px) translateZ(0);
box-shadow: 0 0 1px rgba(0, 0, 0, 0);
position: relative;
overflow: hidden;
}
.hover-animate-from-left:before {
content: "";
position: absolute;
z-index: -1;
left: 0;
right: 100%;
bottom: -1px;
background: #2098D1;
height: 3px;
-webkit-transition-property: right;
transition-property: right;
-webkit-transition-duration: 0.8s;
transition-duration: 0.8s;
-webkit-transition-timing-function: ease-out;
transition-timing-function: ease-out;
}
.hover-animate-from-left:hover:before,
.hover-animate-from-left:focus:before,
.hover-animate-from-left:active:before {
right: 0;
}
</style>
{{/* Warning:
Hugo's `Reverse` function appears to operate in-place,
hence the order of performing $max/$min matters.
*/}}
{{ $max := add (len (index $data 0).Pages) 1 }}
{{ $min := len (index ($data).Reverse 0).Pages }}
{{ $delta := sub $max $min }}
{{ $fontStep := div $fontDelta $delta }}
<div class="row" style="padding:5px"> <!-- background-color:HoneyDew; -->
<div class="heading text-center" style="margin-top:5px">
<div> <!-- class="col-12 text-center" -->
<!-- <span>Categories & Tags</span> -->
<div class="tag-cloud-tags">
{{ range $item := (sort $data ".Page.Title" "asc") }}
{{ $tagCount := len $item.Pages }}
{{ $weight := div (sub (math.Log $tagCount) (math.Log $min)) (sub (math.Log $max) (math.Log $min)) }}
{{ $fontSize := add $fontSmall (mul (sub $fontBig $fontSmall) $weight) }}
<!-- class="button button-border button-mini button-border-thin button-blue button-circle" -->
<!--
.Page.RelPermalink changed to
partial "relLangPrettyUglyURL.hugo" (dict "to" .Page)
-->
{{ if ne (upper .Page.Title) "ARCHIVED" }}
<a href='{{ partial "relLangPrettyUglyURL.hugo" (dict "to" .Page) }}'
style="font-size:{{ $fontSize }}rem" class="hover-animate-from-left">
{{- .Page.Title | upper -}}
<span style="color: LightGrey; font-size:{{ $fontSmall }}rem"> ({{ $tagCount }}) </span>
</a>
{{ end }}
{{ end }}
</div>
</div>
</div>
</div>
{{ end }}
Sitemap
{{/*
-- shortcode/sitemap.html shortcode
general note:
make sure a template name is only defined once throughout
the entire repository, otherwise side effects might occure which
are hard do debug !
relearn theme: .RelPermalink needs to be replaced by
partial "relLangPrettyUglyURL.hugo" (dict "to" .)
*/}}
<style>
/* quick fix,
so that the structured list looks o.k.
*/
li.no-bullets, ul.no-bullets {
list-style-type: none;
}
</style>
{{- $currentNode := . }}
{{- if eq .Site.Params.ordersectionsby "title"}}
{{- range .Site.Home.Sections.ByTitle}}
{{- template "sitemap-tree-nav" dict "sect" . "currentnode" $currentNode }}
{{- end}}
{{- else}}
{{- range .Site.Home.Sections.ByWeight}}
{{- template "sitemap-tree-nav" dict "sect" . "currentnode" $currentNode }}
{{- end}}
{{- end}}
<!-- templates -->
{{- define "sitemap-tree-nav" }}
{{- $currentNode := .currentnode }}
{{- with .sect}}
{{- if and .IsSection (or (not .Params.hidden) $.showhidden)}}
{{- $numberOfPages := (add (len .Pages) (len .Sections)) }}
{{- safeHTML .Params.head}}
<li class="no-bullets">
<div>
<a href='{{ partial "relLangPrettyUglyURL.hugo" (dict "to" .) }}'>
{{safeHTML .Params.Pre}}{{.Title}}{{safeHTML .Params.Post}}
</a>
</div>
{{- if ne $numberOfPages 0 }}
<ul class="no-bullets">
{{- .Scratch.Set "pages" .Pages }}
{{- if .Sections}}
{{- .Scratch.Set "pages" (.Pages | union .Sections) }}
{{- end}}
{{- $pages := (.Scratch.Get "pages") }}
{{- if eq .Site.Params.ordersectionsby "title"}}
{{- range $pages.ByTitle }}
{{- if and .Params.hidden (not $.showhidden) }}
{{- else}}
{{- template "sitemap-tree-nav" dict "sect" . "currentnode" $currentNode }}
{{- end}}
{{- end}}
{{- else}}
{{- range $pages.ByWeight }}
{{- if and .Params.hidden (not $.showhidden) }}
{{- else}}
{{- template "sitemap-tree-nav" dict "sect" . "currentnode" $currentNode }}
{{- end}}
{{- end}}
{{- end}}
</ul>
{{- end}}
</li>
{{- else}}
{{- if not .Params.Hidden }}
<li class="no-bullets">
<div>
<a href='{{ partial "relLangPrettyUglyURL.hugo" (dict "to" .) }}'>
{{safeHTML .Params.Pre}}{{.LinkTitle}}{{safeHTML .Params.Post}}
</a>
</div>
</li>
{{- end}}
{{- end}}
{{- end}}
{{- end}}
To render the sitemap, create a new index.md file in the root of the document contents section using the following code:
+++
title = " "
weight = 6
hidden = false
disableToc = true
headingPost = "<h2><i class='far fa-fw fa-map'></i> Sitemap</h2>"
menuPre = "<br/><i class='far fa-fw fa-map'></i> Sitemap"
+++
<style>
#R-body {background-color: var(--CODE-INLINE-BG-color);}
</style>
{{/* < custom/sitemap > */}}
Columns Shortcode
<style>
.flex{display:flex}
.flex-auto{flex:auto}
.flex-even{flex:1 1}
.flex-wrap{flex-wrap:wrap}
.book-columns>div{margin:1rem 0;min-width:10rem;padding:0 1rem}
.markdown-inner>:first-child{margin-top:0}
.markdown-inner>:last-child{margin-bottom:0}
</style>
<div class="book-columns flex flex-wrap">
{{ range split .Inner "<--->" }}
<div class="flex-even markdown-inner">
{{ . | $.Page.RenderString }}
</div>
{{ end }}
</div>
support for custom.css
<!--
<style>
#body img.bg-white {background-color: white;}
</style>
-->
{{- $assetBusting := not .Site.Params.disableAssetsBusting }}
<link href='{{"css/custom.css" | relURL}}{{ if $assetBusting }}?{{ now.Unix }}{{ end }}' rel="stylesheet">
footer bar with previous and next article
<style>
.bottom-container {
height:45px;
clear:both;
position:absolute;
bottom:0;
width:100%;
color:var(--MENU-SECTIONS-LINK-color);
background-color:var(--MENU-SECTION-ACTIVE-CATEGORY-BG-color);
/* test: background-color:green; */
}
</style>
{{- if or .PrevInSection .NextInSection }}
<div class="bottom-container">
{{- if .NextInSection }}
{{ $next := .NextInSection.Permalink }}
<div class="btn cstyle transparent w3-display-left w3-margin-left">
<a href='{{ partial "relLangPrettyUglyURL.hugo" (dict "to" .NextInSection) }}'>
<i class="fa fa-angle-left"></i>
{{- with .NextInSection }}
{{ .Title | truncate 30 }}
{{- end }}
</a>
</div>
{{- end }}
{{- if .PrevInSection }}
{{ $prev := .PrevInSection.Permalink }}
<div class="btn cstyle transparent w3-display-right w3-margin-right">
<a href='{{ partial "relLangPrettyUglyURL.hugo" (dict "to" .PrevInSection) }}'>
{{- with .PrevInSection }}
{{ .Title | truncate 30 }}
{{- end }}
<i class="fa fa-angle-right"></i>
</a>
</div>
{{- end }}
</div>
{{- end }}
Page Taxonomy in the header for each article
<style>
.entry-meta {
font-style:italic;
font-size:small;
display: flex;
justify-content: center;
margin-top:0px;
padding:0px;
font-weight:300;
}
</style>
{{- if not (eq .Params.archetype "home") }}
{{- if eq .Params.type "docs" }}
<div class="entry-meta box cstyle transparent w3-animate-fading-mod1">
<small>
<span>
<i class="far fa-calendar" style="color:var(--CODE-BLOCK-BORDER-color);"></i>
Posted:
{{ .Date.Format .Site.Params.date_format }}
</span>
{{ if ne .Date .Lastmod }}
<span>
,
<i class="far fa-calendar" style="color:var(--CODE-BLOCK-BORDER-color);"></i>
Last modified:
{{ .Params.lastmod.Format .Site.Params.date_format }}
</span>
{{ end }}
{{ if isset .Params "author" }}
<span>
,
<i class="far fa-user-circle" style="color:var(--CODE-BLOCK-BORDER-color);"></i>
{{ i18n "author" }}{{ .Params.author }}
</span>
{{ else }}
<span>
,
<i class="far fa-user-circle" style="color:var(--CODE-BLOCK-BORDER-color);"></i>
Admin
</span>
{{ end }}
<!-- hans: copied over from tags.html -->
{{- $page := . }}
{{- if .Params.categories }}
<br/>
<span class="categories">
{{- $categories := slice | append .Params.categories }}
{{- range sort $categories }}
{{- if gt (len .) 0 }}
{{- $category := . }}
{{- with $page.Site.GetPage (printf "%s%s" ("/categories/" | relURL ) ( $category | anchorize ) ) }}
{{- $to := . }}
<span>
<i class="fa fa-tags" style="color:var(--CODE-BLOCK-BORDER-color);"></i>
<a href='{{ partial "relLangPrettyUglyURL.hugo" (dict "to" $to) }}'>{{ $category | upper }}</a>
</span>
{{- end }}
{{- end }}
{{- end }}
</span>
{{- end }}
<!-- hans: copied over from tags.html -->
{{- if .Params.tags }}
<span class="tags">
{{- $tags := slice | append .Params.tags }}
{{- range sort $tags }}
{{- if gt (len .) 0 }}
{{- $tag := . }}
{{- with $page.Site.GetPage (printf "%s%s" ("/tags/" | relURL ) ( $tag | anchorize ) ) }}
{{- $to := . }}
<span>
<i class="fa fa-hashtag" style="color:var(--CODE-BLOCK-BORDER-color);"></i>
<a href='{{ partial "relLangPrettyUglyURL.hugo" (dict "to" $to) }}'>{{ $tag | upper }}</a>
</span>
{{- end }}
{{- end }}
{{- end }}
</span>
{{- end }}
<span>
,
<i class="far fa-clock" style="color:var(--CODE-BLOCK-BORDER-color);"> </i>
Time to read:
{{ .ReadingTime }} min.
</span>
</small>
</div>
{{- end }}
{{- end }}
Animated scroll-up buttom
<!-- scroll to top -->
<button onclick='topFunction()' id='myBtn' title='Go to top'></button>
<style>
#myBtn {
display:none;position:fixed;bottom:45px;right:30px;z-index:99;
border:2px;outline:none;background-color:HoneyDew;cursor:pointer;padding:15px;
border-radius:15px;width:40px;height:40px; background:url(),no-repeat;
}
#myBtn:hover {
background-color: var(--INTERNAL-MENU-SECTIONS-LINK-HOVER-color);
}
</style>
<script type="text/javascript">
let mybutton = document.getElementById('myBtn');
let elmnt = document.getElementById("R-body-inner");
elmnt.style.scrollBehavior = 'smooth';
function scrollFunction() {
// probably deactivated/overwritten by perfect-scrollbar.js:
// document.body.scrollTop, document.documentElement.scrollTop
// let y = elmnt.scrollTop;
// console.log(y);
if (elmnt.scrollTop > 20 ) {
mybutton.style.display = 'block';
} else {
mybutton.style.display = 'none';
}
}
function topFunction() {
elmnt.scrollTop = 0; // for Safari
// document.documentElement.scrollTop = 0; // for Chrome, Firefox, IE and Opera
}
// window.onscroll = function() {scrollFunction()};
// the following works:
window.addEventListener('scroll', function() {
scrollFunction();
}, true);
</script>
<!-- scroll to top -->
The shortcode goes here:
{{- partial "goto_top_rocket.html" . }}
{{- $page := .page }}
{{- $content := .content }}
{{- $outputFormat := partial "output-format.hugo" .page }}
{{- partial "single-article.hugo" (dict "page" $page "content" $content "outputFormat" $outputFormat) }}
Support for Custom Style
Custom style support with some more font size adjustments (custom.css):
/* Hans:
partly copied from the theme ksu-cs-textbooks-hugo-theme-relearn-main
to override css properties declared in theme.css
*/
.brown {
color:Brown;
font-weight:bold;
}
.lbrown {
color:Bisque;
font-weight:bold;
text-shadow:0px 0px 4px #000000;
}
.lred {
color:Crimson;
font-weight:bold;
}
.yellow {
color:Yellow;
font-weight:bold;
text-shadow:0px 0px 4px #000000;
}
table {
/* width:auto;
height:auto; */
line-height:12px;
}
.w3-display-container{position:relative;}
.w3-top,.w3-bottom{position:fixed;width:100%;z-index:1}.w3-top{top:0}.w3-bottom{bottom:0}
.w3-display-container:hover .w3-display-hover{display:block}.w3-display-container:hover span.w3-display-hover{display:inline-block}.w3-display-hover{display:none}
.w3-display-left{position:absolute;top:50%;left:0%;transform:translate(0%,-50%);-ms-transform:translate(-0%,-50%)}
.w3-display-right{position:absolute;top:50%;right:0%;transform:translate(0%,-50%);-ms-transform:translate(0%,-50%)}
.w3-margin-left{margin-left:16px!important}.w3-margin-right{margin-right:16px!important}
.w3-display-position{position:absolute}
.w3-center .w3-bar{display:inline-block;width:auto}
.w3-animate-zoom {animation:animatezoom 0.6s}@keyframes animatezoom{from{transform:scale(0)} to{transform:scale(1)}}
.w3-opacity,.w3-hover-opacity:hover{opacity:0.60}.w3-opacity-off,.w3-hover-opacity-off:hover{opacity:1}
.w3-opacity-max{opacity:0.25}.w3-opacity-min{opacity:0.75}
.w3-padding-16{padding-top:16px!important;padding-bottom:16px!important}
.w3-padding-24{padding-top:24px!important;padding-bottom:24px!important}
.w3-padding-32{padding-top:32px!important;padding-bottom:32px!important}
.w3-padding-48{padding-top:48px!important;padding-bottom:48px!important}
.w3-animate-fading-mod {
animation:fading 4s linear
}
@keyframes fading {
0%{opacity:0}
25%{opacity:0.25}
50%{opacity:0.5}
75%{opacity:0.75}
100%{opacity:1}
}
.w3-animate-fading-mod1 {
animation:fading1 1s linear
}
@keyframes fading1 {
0%{opacity:0}
50%{opacity:0.5}
100%{opacity:1}
}
/*
-------------------
theme.css overrides
-------------------
*/
/* EMBED page should be 100% width */
.embed {
width: 100%;
}
/* Fix Font Weights for Lato */
body {
font-size: 1.1rem; /* --- 1.015625rem; --- */
font-weight: 400;
line-height: 1.2; /* --- 1.574; --- */
}
h1 {
font-size: 1.85rem; /* --- 3.25rem; --- */
}
h2 {
font-size: 1.8rem; /* --- 2.2rem; --- */
font-weight: 500;
}
h3, .article-subheading {
font-size: 1.5rem; /* --- 1.8rem; --- */
font-weight: 500;
}
#R-body h2 {
font-weight: 700;
font-size: 1.5rem;
}
#R-body h3 {
font-weight: 600;
font-size: 1.3rem;
}
#R-body h4 {
font-weight: 600;
font-size: 1.3rem;
}
#R-sidebar {
font-size: .953125rem; /* --- .953125rem; --- */
line-height: 1.2; /* 1.574; */
}
#R-sidebar ul li {
line-height: 1.0;
}
#R-sidebar ul.collapsible-menu li.active > a {
border-inline-end-color: var(--INTERNAL-MENU-BORDER-color);
}
#R-sidebar ul.collapsible-menu > li > label {
margin-left: 2px;
}
/* Sidebar Titles */
#R-sidebar ul.topics > li > a b {
font-weight: 400;
opacity: 0.5;
line-height: 1;
}
#R-body .flex-block-wrapper {
/*
margin-left: auto;
margin-right: auto;
*/
max-width: /* calc( 81.25rem - 18.75rem - 2 * 3.25rem ); */
calc( 88.25rem - 18.75rem - 2 * 1.25rem );
}
/* we limit width if we have large screens */
@media screen and ( min-width: 88.25rem ){ /* #R-sidebar/width + ./max-width */
#R-body .flex-block-wrapper {
width: calc( 88.25rem - var(--INTERNAL-MENU-WIDTH-L) - 2 * 3.25rem );
}
body:not(.print) #R-body .narrow .flex-block-wrapper {
width: calc( 88.25rem - var(--INTERNAL-MENU-WIDTH-L) - 2 * 9.75rem );
}
}
Full Page Reader View - experimental
In this view, the side menu will not appear so that the focus is purely on the article itself.
In order to achive this, each single page will be rendered in another format, watch out for
the “embed” keyword in the customized configuration section. The source code is provided at
at the end of this article and can be downloaded as a zip file.
If there are already some customized files available, make sure not to overwrite
them by mistake (as files in the zip file have the same names acc. to the overload mechanism
declared by the hugo static file generator).
Global Configuration settings
In the overall configuration file (config.toml), the following declarations are needed to support local mode and also to tell hugo to generate required “embed” type output files:
# true -> all relative URLs would instead be canonicalized using baseURL
canonifyURLs = true
# required value to serve this page from a webserver AND the file system;
# if you don't want to serve your page from the file system, you can also set this value to false
# true -> rewrite all relative URLs to be relative to the current content
relativeURLs = true
# if you set uglyURLs to false, this theme will append 'index.html' to any branch bundle link
# so your page can be also served from the file system; if you don't want that,
# set disableExplicitIndexURLs=true in the [params] section
# true -> basic/index.html -> basic.html
uglyURLs = false
[outputs]
home = ["HTML", "RSS", "PRINT", "SEARCH", "SEARCHPAGE"]
section = ["HTML", "RSS", "PRINT"]
page = ["HTML", "RSS", "PRINT", "EMBED"]
[outputFormats]
[outputFormats.EMBED]
name= "EMBED"
baseName = "index.embed"
isHTML = true
mediaType = 'text/html'
permalinkable = true
[languages]
[[languages.en.menu.shortcuts]]
name = "<br/><i class='far fa-fw fa-map'></i> Sitemap"
pageRef = "sitemap/"
weight = 10
[params]
date_format = "2006-01-02"
showVisitedLinks = false
collapsibleMenu = true
disableBreadcrumb = false
disableInlineCopyToClipBoard = true
disableNextPrev = false
disableLandingPageButton = true
breadcrumbSeparator = ">"
titleSeparator = "::"
disableAssetsBusting = true
themeVariant = [
{ identifier = "zen-light", name = "Light" },
{ identifier = "zen-dark", name = "Dark" },
{ identifier = "relearn-bright", name = "Bright" },
{ identifier = "neon" },
{ identifier = "blue" },
{ identifier = "green" },
]
Software Versions needed
Software versions needed (minimum required) are as following:
In the time of writing this article, the actual theme version is:
relearn-theme 6.0.0
hugo version which is also the minimum version required to run the relearn-theme 6.0.0:
v0.121.0-e321c3502aa8e80a7a7c951359339a985f082757+extended windows/amd64
File name: | Size / byte: | |
---|---|---|
relearn-theme-6.0.0-customization.zip | 26785 |
Credits:
- gohugo.io: Hugo static site generator
- themes.gohugo.io: HUGO-theme-relearn