CraftGears

Creating a Hugo Theme - Part 6: Advanced Single View

· 10 min read
Creating a Hugo Theme - Part 6: Advanced Single View

This time, I decided to make some changes to the single view layout. Initially, I wanted to display related links on the right side, but I changed my mind. Instead, I moved the related posts section to the bottom of each post.

Additionally, I’ve implemented two options for displaying the cover image at the top: wide and inline.

Let’s take a look at the updated design in the screenshots below.

Inline Cover
Inline Cover
Wide Cover
Wide Cover
Social & Related Posts

Hugo Modules #

I also integrated the Hugo Modules provided by Gethugothemes. I added everything I found useful to enhance the site’s functionality.

Below is the module.toml file configuration I used.

[hugoVersion]
extended = true
min = "0.128.0"

[[imports]]
path = "github.com/gethugothemes/hugo-modules/icons/font-awesome"

[[imports]]
path = "github.com/gethugothemes/hugo-modules/components/social-share"

[[imports]]
path = "github.com/gethugothemes/hugo-modules/search"

[[imports]]
path = "github.com/gethugothemes/hugo-modules/gallery-slider"

[[imports]]
path = "github.com/gethugothemes/hugo-modules/images"

[[imports]]
path = "github.com/gethugothemes/hugo-modules/components/preloader"

[[imports]]
path = "github.com/gethugothemes/hugo-modules/gzip-caching"

[[imports]]
path = "github.com/gethugothemes/hugo-modules/modal"

[[imports]]
path = "github.com/gethugothemes/hugo-modules/seo-tools/basic-seo"

[[imports]]
path = "github.com/gethugothemes/hugo-modules/adsense"

[[imports]]
path = "github.com/gethugothemes/hugo-modules/seo-tools/site-verifications"

[[imports]]
path = "github.com/gethugothemes/hugo-modules/videos"

[[imports]]
path = "github.com/gethugothemes/hugo-modules/table-of-contents"

[[imports]]
path = "github.com/gethugothemes/hugo-modules/tab"

[[imports]]
path = "github.com/gethugothemes/hugo-modules/shortcodes/notice"

[[imports]]
path = "github.com/gethugothemes/hugo-modules/accordion"

This also led to significant changes in the params.toml file.

################################################################################
# Hudo Modules

# Accordion
[[plugins.js]]
link = "js/accordion.js"

# Tab
[[plugins.js]]
link = "js/tab.js"

# Video
[[plugins.js]]
link = "plugins/youtube-lite.js"
[[plugins.js]]
link = "plugins/vimeo-lite.js"

# Site Verifications
[site_verification]
google = ""   # Your verification code
bing = ""     # Your verification code
baidu = ""    # Your verification code
facebook = "" # Your verification code
mastodon = "" # Your verification code

# Google Adsense, see https://www.google.com/adsense/
google_adsense = "" # example: ca-pub-XXXXXXXXXX

# Basic SEO: seo meta data for OpenGraph / Twitter Card
[metadata]
keywords = ["Craft", "Gears", "IT", "Tools"]
description = "There is a wealth of information available on CraftGears. Be sure to visit!"
author = "Craft Gears"
image = "images/fallback.jpg" # this image will be used as fallback if a page has no image of its own

# Modal
[[plugins.js]]
link = "js/modal.js"

# Preloader
[preloader]
enable = false
preloader = "" # use jpg, png, svg or gif format.

# Font Awesome
[[plugins.css]]
link = "plugins/font-awesome/v6/brands.css"
[[plugins.css]]
link = "plugins/font-awesome/v6/icons.css"
[[plugins.css]]
link = "plugins/font-awesome/v6/regular.css"
[[plugins.css]]
link = "plugins/font-awesome/v6/solid.css"

# Gallery Slider
[[plugins.css]]
link = "plugins/glightbox/glightbox.css"
lazy = true

[[plugins.css]]
link = "plugins/swiper/swiper-bundle.css"

[[plugins.js]]
link = "plugins/glightbox/glightbox.js"

[[plugins.js]]
link = "js/gallery-slider.js"
lazy = true

[[plugins.js]]
link = "plugins/swiper/swiper-bundle.js"

# Search
[[plugins.js]]
link = "js/search.js"

[search]
enable = true
primary_color = "#ce8460"
include_sections = ["posts"] # if `include_sections` empty, then sections will come from the `mainSections`
# include_all_sections = true # if `include_all_sections` is true, then comment out the `include_sections`
include_list_pages = false
show_image = true
show_description = true
show_tags = true
show_categories = true

# Image
[[plugins.js]]
link = "plugins/lazy-loader.js"

The config.toml file has been updated as well.

[services]
  [services.googleAnalytics]
    id = 'G-MEASUREMENT_ID'

################################################################################
# Markup
[markup.tableOfContents]
startLevel = 2
endLevel = 3
ordered = false

################################################################################
# Outputs
[outputs]
home = ["html", "rss", "SearchIndex"]
page = ['html']
rss = ['rss']
section = ['html', 'rss']
taxonomy = ['html', 'rss']
term = ['html', 'rss']

################################################################################
# Output Format
[outputFormats]
[outputFormats.SearchIndex]
mediaType = "application/json"
baseName = "searchindex"
isPlainText = true
notAlternative = true

While working on these updates, I encountered numerous trial-and-error moments, particularly with the Search functionality.

Here are the key findings:

  • Contrary to the Hugo Modules documentation, you need to add ["html", "rss"] to outputs.home. Without this, the index.html file will not be generated.
  • If the search functionality doesn’t work, ensure the JavaScript file is executed at the end of the <body> tag. (I learned this from Hugoplate—thank you!)

As a result, the baseof.html file has been modified as shown below.

baseof.html

<!DOCTYPE html>
<html lang="{{ site.Language.LanguageCode }}" dir="{{ or site.Language.LanguageDirection `ltr` }}">
<head>
  <!-- head -->
  {{ partial "head.html" . }}

  <!-- style -->
  {{ partial "head/style.html" . }}
</head>
<body>
  <header class="sticky top-0 z-50 md:flex w-full p-4 md:p-6 bg-white/80 backdrop-blur-md border-b border-dark3-color/30">
    {{ partial "header.html" . }}
  </header>

  <!-- hugo-modules: search -->
  {{ partial "search-modal.html" (dict "Context" .) }}

  <main>
    {{ block "main" . }}{{ end }}
  </main>
  <footer class="flex items-center justify-center bg-gray-100 w-full">
    {{ partial "footer.html" . }}
  </footer>

  <!-- script -->
  {{ partialCached "head/script.html" . }}
  </body>
</html>

The original css.html file has been renamed to style.html.

style.html

<!-- DNS preconnect -->
<meta http-equiv="x-dns-prefetch-control" content="on" />
<link rel="preconnect" href="https://use.fontawesome.com" crossorigin />
<link rel="preconnect" href="//cdnjs.cloudflare.com" />
<link rel="preconnect" href="//www.googletagmanager.com" />
<link rel="preconnect" href="//www.google-analytics.com" />
<link rel="dns-prefetch" href="https://use.fontawesome.com" />
<link rel="dns-prefetch" href="//ajax.googleapis.com" />
<link rel="dns-prefetch" href="//cdnjs.cloudflare.com" />
<link rel="dns-prefetch" href="//www.googletagmanager.com" />
<link rel="dns-prefetch" href="//www.google-analytics.com" />
<link rel="dns-prefetch" href="//fonts.googleapis.com" />
<link rel="dns-prefetch" href="//connect.facebook.net" />
<link rel="dns-prefetch" href="//platform.linkedin.com" />
<link rel="dns-prefetch" href="//platform.twitter.com" />

<!-- plugins + stylesheet -->
{{ $styles := slice }}
{{ $stylesLazy := slice }}
{{ range site.Params.plugins.css }}
  {{ if findRE "^http" .link }}
    <link
      crossorigin="anonymous"
      media="all"
      rel="stylesheet"
      href="{{ .link | relURL }}"
      {{ .attributes | safeHTMLAttr }} />
  {{ else }}
    {{ if not .lazy }}
      {{ $styles = $styles | append (resources.Get .link) }}
    {{ else }}
      {{ $stylesLazy = $stylesLazy | append (resources.Get .link) }}
    {{ end }}
  {{ end }}
{{ end }}

{{/* main style */}}
{{ $styles = $styles | append (resources.Get "scss/main.scss" | toCSS) }}
{{ $styles = $styles | resources.Concat "css/style.css" }}

{{ $stylesLazy = $stylesLazy | resources.Concat "css/style-lazy.css" }}

{{ if hugo.IsProduction }}
  {{ $styles = $styles | resources.ExecuteAsTemplate "css/style.css" . | minify | fingerprint | resources.PostProcess }}
  {{ $stylesLazy = $stylesLazy | resources.ExecuteAsTemplate "css/style-lazy.css" . | minify | fingerprint | resources.PostProcess }}
{{ else }}
  {{ $styles = $styles | resources.ExecuteAsTemplate "css/style.css" . }}
  {{ $stylesLazy = $stylesLazy | resources.ExecuteAsTemplate "css/style-lazy.css" . }}
{{ end }}

{{/* styles */}}
<link
  href="{{ $styles.RelPermalink }}"
  integrity="{{ $styles.Data.Integrity }}"
  rel="stylesheet" />

{{/* styles lazy */}}
<link
  defer
  async
  rel="stylesheet"
  href="{{ $stylesLazy.RelPermalink }}"
  integrity="{{ $stylesLazy.Data.Integrity }}"
  media="print"
  onload="this.media='all'; this.onload=null;" />

{{/*  TailwindCSS  */}}
{{ with resources.Get "css/main.css" }}
  {{ $opts := dict "minify" true }}
  {{ with . | css.TailwindCSS $opts }}
    {{ if hugo.IsDevelopment }}
      <link rel="stylesheet" href="{{ .RelPermalink }}">
    {{ else }}
      {{ with . | fingerprint }}
        <link
          rel="stylesheet"
          href="{{ .RelPermalink }}"
          integrity="{{ .Data.Integrity }}"
          crossorigin="anonymous">
      {{ end }}
    {{ end }}
  {{ end }}
{{ end }}

script.html

<!-- JS Plugins + Main script -->
{{ $scripts := slice }}
{{ $scriptsLazy := slice }}
{{ range site.Params.plugins.js }}
  {{ if findRE "^http" .link }}
    <script
      src="{{ .link | relURL }}"
      type="application/javascript"
      {{ .attributes | safeHTMLAttr }}></script>
  {{ else }}
    {{ if not .lazy }}
      {{ $scripts = $scripts | append (resources.Get .link) }}
    {{ else }}
      {{ $scriptsLazy = $scriptsLazy | append (resources.Get .link) }}
    {{ end }}
  {{ end }}
{{ end }}


<!-- main script -->
{{ $scripts = $scripts | append (resources.Get "js/main.js") }}
{{ $scripts = $scripts | resources.Concat "js/script.js" }}

{{ $scriptsLazy = $scriptsLazy | resources.Concat "js/script-lazy.js" }}

{{ if hugo.IsProduction }}
  {{ $scripts = $scripts | minify | fingerprint }}
  {{ $scriptsLazy = $scriptsLazy | minify | fingerprint }}
{{ end }}

{{/* scripts */}}
<script
  crossorigin="anonymous"
  integrity="{{ $scripts.Data.Integrity }}"
  src="{{ $scripts.RelPermalink }}"></script>

{{/* scripts lazy */}}
<script
  defer
  async
  crossorigin="anonymous"
  integrity="{{ $scriptsLazy.Data.Integrity }}"
  src="{{ $scriptsLazy.RelPermalink }}"></script>

To customize the styles of the newly added Hugo Modules, I created two additional files. Following the Hugo Modules guide, these files were added as necessary.

assets/scss/main.scss

@import 'social-share';
@import 'search';
@import 'gallery-slider';
@import 'images';
@import 'preloader';
@import 'modal';
@import 'toc';
@import 'tab';
@import 'notice';
@import 'accordion';

@import "hugo-modules"

assets/scss/hugo-modules.scss

Note that styles were not applied to every module—consider this an example of how to make selective customizations.

// Image
figure {
  figcaption {
    text-align: center;
    font-style: italic;
  }
}

// Notice
.notice {
  border-radius: var(--radius-2xl);
}

// ToC
.table-of-content {
  & summary {
    border-radius: var(--radius-2xl);
    border-bottom-right-radius: 0;
    border-bottom-left-radius: 0;
  }

  & nav {
    border-radius: var(--radius-2xl);
    border-top-right-radius: 0;
    border-top-left-radius: 0;
  }

  // open state
  &[open] {
    & summary {
      border-radius: var(--radius-2xl);
      border-bottom-right-radius: 0;
      border-bottom-left-radius: 0;
    }

    & nav {
      border-radius: var(--radius-2xl);
      border-top-right-radius: 0;
      border-top-left-radius: 0;
    }
  }

  // closed state
  &:not([open]) {
    & summary {
      border-radius: var(--radius-2xl);
    }

    & nav {
      border-radius: var(--radius-2xl);
    }
  }
}

I almost forgot the search button! It has been added to menu.html as shown below.

...
  <nav class="hidden peer-checked:block absolute top-16 left-0 w-full bg-white sm:bg-transparent shadow-md sm:static sm:shadow-none sm:block">
    <ul class="flex text-sm items-center sm:justify-end flex-col gap-y-2 p-4 sm:flex-row sm:gap-x-2 sm:p-0 sm:text-dark2-color uppercase font-semibold">
      {{- partial "inline/menu/walk.html" (dict "page" $page "menuEntries" .) }}
      <!-- Insert code -->
      {{ if site.Params.search.enable }}
        <button class="w-6 cursor-pointer" data-target="search-modal">
          <i class="fa-solid fa-magnifying-glass"></i>
        </button>
      {{ end }}
    </ul>
  </nav>
...

In addition to the above, I integrated various Hugo Modules for features like SEO, Adsense, and Site Verification. With these enhancements, the blog feels like a well-rounded theme. For detailed guidance, refer to the Hugo Modules documentation.


single.html #

Now, let’s refocus on the single view.

{{ define "main" }}
  <div class="md:flex md:flex-col mx-auto gap-4 md:gap-6 py-10">
    <article class="prose max-w-none">
      <h1 class="text-3xl md:text-4xl text-center px-4 md:px-6">{{ .Title }}</h1>

      <div class="flex items-center justify-center gap-x-2">
        {{ $dateMachine := .Date | time.Format "2006-01-02T15:04:05-07:00" }}
        {{ $dateHuman := .Date | time.Format ":date_long" }}
        <i class="fa-regular fa-calendar"></i>
        <time datetime="{{ $dateMachine }}">{{ $dateHuman }}</time>
        ·
        <i class="fa-regular fa-clock"></i>
        <span>
          {{ .ReadingTime }} min read
        </span>
      </div>

      {{- if .Params.image }}
        {{ $coverLink := "" }}
        {{- if (strings.HasPrefix .Params.image "http") }}
          {{ $coverLink = .Params.image }}
        {{- else }}
          {{- with $coverImage := .Resources.Get .Params.image -}}
            {{ $coverImage := $coverImage.Process "resize 1500x" }}
            {{ $coverLink = $coverImage.Permalink }}
          {{- end }}
        {{- end }}

        {{ if eq .Params.coverMode "inline" }}
          <div class="px-4">
            <img class="md:mx-auto md:max-w-3xl bg-red-500 rounded-3xl" src="{{ $coverLink }}" alt="{{.Title}}" />
          </div>
        {{ else if eq .Params.coverMode "wide" }}
          <img class="w-screen mx-auto md:max-w-[1900] aspect-video object-cover" src="{{ $coverLink }}" alt="{{.Title}}" />
        {{ end }}
      {{- end }}


      <div class="prose mx-auto md:max-w-3xl px-4 md:px-6 py-6">
        {{ .Content }}
      </div>
    </article>

    <div class="mx-auto md:max-w-3xl p-4 md:p-6 w-full">
      {{ partial "terms.html" (dict "taxonomy" "tags" "page" .) }}
    </div>

    <div class="md:mx-auto md:max-w-3xl p-4 md:px-6 w-full">
      <div class="flex flex-col items-center bg-gray-100 rounded-3xl gap-4 text-dark2-color md:mx-auto md:max-w-3xl p-4 md:p-6 w-full">
        <span class="text-center">Did you find this post helpful?<br/>Share it with others!</span>
        <!-- social share -->
        {{ partial "social-share.html" (dict "Context" . ) }}
      </div>
    </div>

    <!-- Related posts -->
    {{ $related := (where site.RegularPages "Section" "in" site.Params.mainSections) | intersect (where site.RegularPages ".Title" "!=" .Title) | union (site.RegularPages.Related . ) }}
    {{ $related = $related | shuffle | first 4 }}
    {{ with $related }}
      <div class="md:mx-auto md:max-w-3xl px-4 md:px-6 w-full">
        <h2 class="text-2xl font-bold text-dark1-color pb-4">Related Posts</h2>
        <div class="flex flex-col gap-4">
          {{ range . }}
            {{ partial "content/slim-card.html" . }}
          {{ end }}
        </div>
      </div>
    {{ end }}
  </div>
{{ end }}

Here are four key updates compared to the previous design:

  • Reading Time
  • Cover Images
  • Social Sharing
  • Related Posts

Adding Links to Subtitles If you look at the post, you’ll notice a # link next to the chapter titles. This is implemented using Hugo’s render functionality.

To enable this, add a render-heading.html file under: themes/<theme-name>/layouts/_default/_markup/render-heading.html.

<h{{ .Level }} id="{{ .Anchor }}">
  {{ .Text }}
  {{ if le .Level 2 }}
    <a class="text-dark3-color no-underline" href="#{{ .Anchor }}">#</a>
  {{ end }}
</h{{ .Level }}>

Conclusion #

Since there were so many detailed changes, I may have missed mentioning a few files. However, this should give you a good overview of the main updates.

This wraps up my post on creating a theme. While working on this site and documenting the process, I realized the order might feel a bit chaotic. In the future, I’ll come back with more structured and polished content. Stay tuned!

Did you find this post helpful?
Share it with others!