CraftGears

Creating a Hugo Theme - Part 4: List Layout

· 3 min read
Creating a Hugo Theme - Part 4: List Layout

In this post, we’ll improve the list.html file. Here’s how it currently looks—completely unstyled:

Unstyled Page

Styling Plan #

The plan is to display the list of posts in two columns. On the left, there will be a simple thumbnail, and on the right, the title and description. Each page will show ten posts with pagination available.

Here’s how the finished layout looks. (For demonstration, the page is set to display six posts due to limited content.)

Styled Desktop Styled Mobile 1 Styled Mobile 2


list.html #

While list.html can include posts, tags, and more, we’ll focus here solely on styling the post list. Tags are handled separately since they don’t have images.

This file also naturally integrates with the “More” button on the homepage, enabling seamless navigation.

{{ define "main" }}
  <div class="flex flex-col md:max-w-7xl mx-auto w-full p-4 md:p-6">
    <h2 class="text-3xl font-bold py-4">{{ .Title | humanize }}</h2>

    <div class="grid grid-cols-1 lg:grid-cols-2 gap-4 lg:gap-8">
      {{ range ( .Paginator 6 ).Pages }}
        {{- partial "content/slim-card.html" . -}}
      {{ end }}
    </div>

    <div class="flex items-center mt-10">
      {{- if .Paginator.HasPrev -}}
      <a class="text-dark1-color bg-gray-200 hover:bg-gray-300 rounded-full py-1 md:py-1.5 mr-auto md:py-1.5 px-4 md:px-6 duration-300"
          href="{{ .Paginator.Prev.URL }}">Previous</a>
      {{- end -}}

      {{- if .Paginator.HasNext -}}
      <a class="text-dark1-color bg-gray-200 hover:bg-gray-300 rounded-full py-1 md:py-1.5 ml-auto md:py-1.5 px-4 md:px-6 duration-300"
          href="{{ .Paginator.Next.URL }}">Next</a>
      {{- end -}}
    </div>
  </div>
{{ end }}

The implementation is straightforward, and the individual cards are handled by the slim-card.html partial.


slim-card.html #

This card design is simpler than the homepage. It features an image on the left and the description on the right.

<a href="{{ .Permalink }}">
  <article class="relative flex group rounded-3xl duration-300 bg-gray-100 h-40 p-4">
    {{- if .Params.image }}
      {{- if (strings.HasPrefix .Params.image "http") }}
        <img
          class="rounded-3xl aspect-square object-cover group-hover:scale-105 duration-300"
          src="{{ .Params.image }}"
          alt="{{ $.Name }}">
      {{- else }}
        {{- with $coverImage := .Resources.Get .Params.image -}}
          {{- $coverImage := $coverImage.Process "resize 700x" -}}
          <img
            class="rounded-3xl aspect-square object-cover group-hover:scale-105 duration-300"
            src="{{ $coverImage.Permalink }}"
            alt="{{ $.Name }}">
        {{- end }}
      {{- end }}
    {{- end }}
    <div class="flex flex-col px-4">
      <h3 class="mb-4 text-dark1-color text-lg font-bold line-clamp-1">{{ .LinkTitle }}</h3>
      <p class="flex-1 mb-4 text-dark2-color text-normal leading-normal line-clamp-2">{{ .Params.description }}</p>
      {{ $dateMachine := .Date | time.Format "2006-01-02T15:04:05-07:00" }}
      {{ $dateHuman := .Date | time.Format ":date_long" }}
      <time class="text-dark3-color" datetime="{{ $dateMachine }}">{{ $dateHuman }}</time>
    </div>
  </article>
</a>

terms.html #

Tags and categories are processed in the terms.html file. This is a general-purpose template as it handles the final lookup stage in Hugo’s rendering pipeline.

For a detailed customization guide, refer to the official documentation: Hugo Taxonomy Templates

{{ define "main" }}
  <div class="flex flex-col md:max-w-7xl mx-auto w-full p-4 md:p-6">
    <h2 class="text-3xl font-bold py-4">{{ .Title | humanize }}</h2>

    <div class="flex flex-wrap gap-2 pb-10">
      {{ range .Pages }}
        <a class="px-4 py-1 md:py-1.5 rounded-full bg-gray-100 hover:bg-gray-200 duration-300" href="{{ .RelPermalink }}">
          {{ .LinkTitle }}
        </a>
      {{ end }}
    </div>
  </div>
{{ end }}

Conclusion #

The basic structure of the site is now in place. Only the header and footer remain. For the header, adding a folding feature for mobile would be ideal.

We’ll tackle that in the next installment.

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