CraftGears

Creating a Hugo Theme - Part 3: Home Layout

· 5 min read
Creating a Hugo Theme - Part 3: Home Layout

Below is the current state of the Home layout:

Unstyled Home

It’s hard to tell what this setup is all about. Let’s shape it step by step.


Styling Plan #

The Home Layout is where design sense matters. I am not a professional designer, so the results may not be exceptional. Still, I’ll do my best and draw inspiration from Apple’s Newsroom design.


Overall Structure

  • Sticky header with a backdrop effect when scrolling
  • Category-based list (3 latest posts per category) - white card style
  • Recent posts list (6 latest posts) - black card style

That’s it for simplicity. This setup requires minor modifications to some previously created files.

Instead of just explaining, let me show you how it looks after the updates:

Styled Desktop 1 Styled Desktop 2 Styled Mobile


Home Layout #

baseof.html

<!DOCTYPE html>
<html lang="{{ site.Language.LanguageCode }}" dir="{{ or site.Language.LanguageDirection `ltr` }}">
<head>
  {{ partial "head.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">
    {{ partial "header.html" . }}
  </header>
  <main>
    {{ block "main" . }}{{ end }}
  </main>
  <footer class="flex items-center justify-center bg-gray-100 w-full h-52">
    {{ partial "footer.html" . }}
  </footer>
</body>
</html>

In this file, the header and footer sections have additional classes. The header now supports sticky functionality and has a backdrop background effect, while the footer received background color and size adjustments.

home.html

{{ define "main" }}
  <!-- Hugo Section -->
  <div class="bg-gray-100 w-full p-4 md:px-8">
    {{ $categories := site.Params.categories }}
    {{ range $categories }}
      <section class="md:flex md:flex-col md:max-w-7xl mx-auto md:p-6">
        <div class="flex items-center mb-4 md:mb-8">
          <h2 class="font-bold text-3xl md:text-4xl text-dark1-color mr-auto">{{ . | humanize }}</h2>
          <a class="text-dark1-color bg-gray-200 hover:bg-gray-300 rounded-full py-1 md:py-1.5 px-4 md:px-6 duration-300"
            href="{{ printf "/categories/%s" . | absURL }}">More</a>
        </div>

        <div class="grid grid-cols-1 md:grid-cols-2 lg:grid-cols-3 gap-4 md:gap-8">
          {{ range where $.Site.RegularPages "Params.categories" "intersect" (slice . ) | first 3 }}
            {{- partial "content/white-card" . -}}
          {{ end }}
        </div>
      </section>
    {{ end }}
  </div>

  <!-- Recent Posts -->
  <div class="bg-white w-full p-4 md:p-8 md:py-12">
    <section class="md:flex md:flex-col md:max-w-7xl mx-auto md:px-6">
      <div class="flex items-center mb-4 md:mb-8">
        <h2 class="font-bold text-3xl md:text-4xl text-dark1-color mr-auto">Recent Posts</h2>
        <a class="text-dark1-color bg-gray-200 hover:bg-gray-300 rounded-full py-1 md:py-1.5 px-4 md:px-6 duration-300"
          href="{{ "/posts" | absURL }}">More</a>
      </div>

      <div class="grid grid-cols-1 md:grid-cols-2 lg:grid-cols-3 gap-4 md:gap-8">
        {{ range site.RegularPages | first 6 }}
          {{- partial "content/dark-card" . -}}
        {{ end }}
      </div>
    </section>
  </div>
{{ end }}

The layout first displays the latest 3 posts for each category, followed by 6 recent posts overall. Categories can grow dynamically, and their number is defined in the config.toml as follows:

[params]
categories = ["hugo"]

Both sections share a similar structure, and the More button links to the corresponding lists.


Card Components

The card designs are split into white-card.html and dark-card.html. For post images, the cover value from each post’s Front Matter is used.

white-card.html

<article class="group rounded-3xl duration-300 bg-white">
  <a class="flex flex-col h-full" href="{{ .Permalink }}">
    <figure class="shrink-0 w-full aspect-video overflow-hidden rounded-3xl rounded-b-none">
      {{- if .Params.image }}
        {{- if (strings.HasPrefix .Params.image "http") }}
          <img
            class="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="object-cover group-hover:scale-105 duration-300"
              src="{{ $coverImage.Permalink }}"
              alt="{{ $.Name }}">
          {{- end }}
        {{- end }}
      {{- end }}
      </figure>
    <div class="flex flex-col flex-1 p-6">
      <h3 class="mb-4 text-dark1-color text-2xl font-bold">{{ .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>
  </a>
</article>

dark-card.html

<article class="relative group rounded-3xl duration-300 bg-black">
  <a class="relative" href="{{ .Permalink }}">
    <figure class="w-full aspect-square overflow-hidden rounded-3xl">
      {{- if .Params.image }}
        {{- if (strings.HasPrefix .Params.image "http") }}
          <img
            class="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="object-cover group-hover:scale-105 duration-300"
              src="{{ $coverImage.Permalink }}"
              alt="{{ $.Name }}">
          {{- end }}
        {{- end }}
      {{- end }}
    </figure>
    <div class="absolute flex items-end bg-gradient-to-b from-transparent to-black top-0 w-full h-4/5 mb-20"></div>
    <div class="absolute flex items-end bg-transparent top-0 w-full h-full mb-20">
      <div class="p-6">
        <h3 class="mb-4 text-light1-color text-2xl font-bold">{{ .LinkTitle }}</h3>
        <p class="mb-4 text-light2-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-light3-color" datetime="{{ $dateMachine }}">{{ $dateHuman }}</time>
      </div>
    </div>
  </a>
</article>

While similar, dark-card uses a black gradient background with lighter text for contrast.

Adjusting Single Post View

However, modifying baseof.html caused the content layout of individual posts to stretch too wide. I will fix this issue.

{{ define "main" }}
  <div class="md:flex md:flex-col md:max-w-7xl mx-auto md:p-6">
    <div class="grid grid-cols-1 lg:grid-cols-3 gap-6">
      <div class="col-span-2">
        <article class="prose max-w-none">
          <h1>{{ .Title }}</h1>

          {{ $dateMachine := .Date | time.Format "2006-01-02T15:04:05-07:00" }} {{
          $dateHuman := .Date | time.Format ":date_long" }}
          <time datetime="{{ $dateMachine }}">{{ $dateHuman }}</time>

          {{ .Content }}
        </article>

        {{ partial "terms.html" (dict "taxonomy" "tags" "page" .) }}
      </div>
      <div class="bg-gray-100 w-full h-full">
        aside area
      </div>
    </div>
  </div>
{{ end }}

TailwindCSS v4 Variables #

As you might have noticed from the code above, I used several color variables. The syntax feels unfamiliar compared to traditional JavaScript files, but here’s how I adjusted it.

@import "tailwindcss";

@plugin "@tailwindcss/typography";

@theme {
  --color-dark1-color: var(--color-gray-900);
  --color-dark2-color: var(--color-gray-700);
  --color-dark3-color: var(--color-gray-500);
  --color-light1-color: var(--color-gray-100);
  --color-light2-color: var(--color-gray-300);
  --color-light3-color: var(--color-gray-500);

  --color-primary: var(--color-sky-500);
}

Dark shades (dark1~3) are used on light backgrounds, while light shades (light1~3) complement dark backgrounds.


Conclusion #

The home layout serves as the face of the blog, often changing frequently. Using this approach, you can create your unique layout and refine it to perfection.

In the next part, we’ll dive into updating the list sections.

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