mirror of
https://github.com/imfing/hextra.git
synced 2025-07-01 16:27:13 -04:00
Compare commits
17 Commits
Author | SHA1 | Date | |
---|---|---|---|
2df3c563bf | |||
ec02eb34fe | |||
46dea718e6 | |||
adf5a113fc | |||
6a19ac31c0 | |||
3c4ede96df | |||
01f7e3a425 | |||
da5a087891 | |||
79883dc7cc | |||
b283227046 | |||
0e9cf1a519 | |||
fdc30c6cd5 | |||
3632294706 | |||
929578192b | |||
c18d5def26 | |||
4e63aa4f14 | |||
b51bfa3177 |
2
.github/workflows/pages.yml
vendored
2
.github/workflows/pages.yml
vendored
@ -34,7 +34,7 @@ jobs:
|
|||||||
HUGO_VERSION: 0.117.0
|
HUGO_VERSION: 0.117.0
|
||||||
steps:
|
steps:
|
||||||
- name: Checkout
|
- name: Checkout
|
||||||
uses: actions/checkout@v3
|
uses: actions/checkout@v4
|
||||||
with:
|
with:
|
||||||
fetch-depth: 0 # Fetch all history for .GitInfo and .Lastmod
|
fetch-depth: 0 # Fetch all history for .GitInfo and .Lastmod
|
||||||
- name: Setup Go
|
- name: Setup Go
|
||||||
|
@ -613,6 +613,9 @@ video {
|
|||||||
.mb-4 {
|
.mb-4 {
|
||||||
margin-bottom: 1rem;
|
margin-bottom: 1rem;
|
||||||
}
|
}
|
||||||
|
.mb-6 {
|
||||||
|
margin-bottom: 1.5rem;
|
||||||
|
}
|
||||||
.mb-8 {
|
.mb-8 {
|
||||||
margin-bottom: 2rem;
|
margin-bottom: 2rem;
|
||||||
}
|
}
|
||||||
@ -779,6 +782,9 @@ video {
|
|||||||
.max-w-\[min\(calc\(100vw-2rem\)\2c calc\(100\%\+20rem\)\)\] {
|
.max-w-\[min\(calc\(100vw-2rem\)\2c calc\(100\%\+20rem\)\)\] {
|
||||||
max-width: min(calc(100vw - 2rem),calc(100% + 20rem));
|
max-width: min(calc(100vw - 2rem),calc(100% + 20rem));
|
||||||
}
|
}
|
||||||
|
.max-w-full {
|
||||||
|
max-width: 100%;
|
||||||
|
}
|
||||||
.max-w-none {
|
.max-w-none {
|
||||||
max-width: none;
|
max-width: none;
|
||||||
}
|
}
|
||||||
@ -999,17 +1005,9 @@ video {
|
|||||||
--tw-bg-opacity: 1;
|
--tw-bg-opacity: 1;
|
||||||
background-color: rgb(254 252 232 / var(--tw-bg-opacity));
|
background-color: rgb(254 252 232 / var(--tw-bg-opacity));
|
||||||
}
|
}
|
||||||
.bg-gradient-to-b {
|
|
||||||
background-image: linear-gradient(to bottom, var(--tw-gradient-stops));
|
|
||||||
}
|
|
||||||
.bg-gradient-to-r {
|
.bg-gradient-to-r {
|
||||||
background-image: linear-gradient(to right, var(--tw-gradient-stops));
|
background-image: linear-gradient(to right, var(--tw-gradient-stops));
|
||||||
}
|
}
|
||||||
.from-gray-800 {
|
|
||||||
--tw-gradient-from: #1f2937 var(--tw-gradient-from-position);
|
|
||||||
--tw-gradient-to: rgb(31 41 55 / 0) var(--tw-gradient-to-position);
|
|
||||||
--tw-gradient-stops: var(--tw-gradient-from), var(--tw-gradient-to);
|
|
||||||
}
|
|
||||||
.from-gray-900 {
|
.from-gray-900 {
|
||||||
--tw-gradient-from: #111827 var(--tw-gradient-from-position);
|
--tw-gradient-from: #111827 var(--tw-gradient-from-position);
|
||||||
--tw-gradient-to: rgb(17 24 39 / 0) var(--tw-gradient-to-position);
|
--tw-gradient-to: rgb(17 24 39 / 0) var(--tw-gradient-to-position);
|
||||||
@ -1113,12 +1111,18 @@ video {
|
|||||||
.pl-\[max\(env\(safe-area-inset-left\)\2c 1\.5rem\)\] {
|
.pl-\[max\(env\(safe-area-inset-left\)\2c 1\.5rem\)\] {
|
||||||
padding-left: max(env(safe-area-inset-left),1.5rem);
|
padding-left: max(env(safe-area-inset-left),1.5rem);
|
||||||
}
|
}
|
||||||
|
.pr-2 {
|
||||||
|
padding-right: 0.5rem;
|
||||||
|
}
|
||||||
.pr-4 {
|
.pr-4 {
|
||||||
padding-right: 1rem;
|
padding-right: 1rem;
|
||||||
}
|
}
|
||||||
.pr-\[calc\(env\(safe-area-inset-right\)-1\.5rem\)\] {
|
.pr-\[calc\(env\(safe-area-inset-right\)-1\.5rem\)\] {
|
||||||
padding-right: calc(env(safe-area-inset-right) - 1.5rem);
|
padding-right: calc(env(safe-area-inset-right) - 1.5rem);
|
||||||
}
|
}
|
||||||
|
.pr-\[max\(env\(safe-area-inset-left\)\2c 1\.5rem\)\] {
|
||||||
|
padding-right: max(env(safe-area-inset-left),1.5rem);
|
||||||
|
}
|
||||||
.pr-\[max\(env\(safe-area-inset-right\)\2c 1\.5rem\)\] {
|
.pr-\[max\(env\(safe-area-inset-right\)\2c 1\.5rem\)\] {
|
||||||
padding-right: max(env(safe-area-inset-right),1.5rem);
|
padding-right: max(env(safe-area-inset-right),1.5rem);
|
||||||
}
|
}
|
||||||
@ -1197,6 +1201,9 @@ video {
|
|||||||
.leading-7 {
|
.leading-7 {
|
||||||
line-height: 1.75rem;
|
line-height: 1.75rem;
|
||||||
}
|
}
|
||||||
|
.leading-none {
|
||||||
|
line-height: 1;
|
||||||
|
}
|
||||||
.leading-tight {
|
.leading-tight {
|
||||||
line-height: 1.25;
|
line-height: 1.25;
|
||||||
}
|
}
|
||||||
@ -1363,6 +1370,9 @@ video {
|
|||||||
.duration-200 {
|
.duration-200 {
|
||||||
transition-duration: 200ms;
|
transition-duration: 200ms;
|
||||||
}
|
}
|
||||||
|
.ease-in {
|
||||||
|
transition-timing-function: cubic-bezier(0.4, 0, 1, 1);
|
||||||
|
}
|
||||||
.ease-in-out {
|
.ease-in-out {
|
||||||
transition-timing-function: cubic-bezier(0.4, 0, 0.2, 1);
|
transition-timing-function: cubic-bezier(0.4, 0, 0.2, 1);
|
||||||
}
|
}
|
||||||
@ -3344,11 +3354,6 @@ body {
|
|||||||
align-items: flex-start;
|
align-items: flex-start;
|
||||||
}
|
}
|
||||||
|
|
||||||
.sm\:px-4 {
|
|
||||||
padding-left: 1rem;
|
|
||||||
padding-right: 1rem;
|
|
||||||
}
|
|
||||||
|
|
||||||
.sm\:text-xl {
|
.sm\:text-xl {
|
||||||
font-size: 1.25rem;
|
font-size: 1.25rem;
|
||||||
}
|
}
|
||||||
|
@ -1,5 +1,16 @@
|
|||||||
// Search functionality using FlexSearch.
|
// Search functionality using FlexSearch.
|
||||||
|
|
||||||
|
// Change shortcut key to cmd+k on Mac, iPad or iPhone.
|
||||||
|
document.addEventListener("DOMContentLoaded", function () {
|
||||||
|
if (/iPad|iPhone|Macintosh/.test(navigator.userAgent)) {
|
||||||
|
// select the kbd element under the .search-wrapper class
|
||||||
|
const keys = document.querySelectorAll(".search-wrapper kbd");
|
||||||
|
keys.forEach(key => {
|
||||||
|
key.innerHTML = '<span class="text-xs">⌘</span>K';
|
||||||
|
});
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
// Render the search data as JSON.
|
// Render the search data as JSON.
|
||||||
// {{ $searchDataFile := printf "%s.search-data.json" .Language.Lang }}
|
// {{ $searchDataFile := printf "%s.search-data.json" .Language.Lang }}
|
||||||
// {{ $searchData := resources.Get "json/search-data.json" | resources.ExecuteAsTemplate $searchDataFile . }}
|
// {{ $searchData := resources.Get "json/search-data.json" | resources.ExecuteAsTemplate $searchDataFile . }}
|
||||||
|
3
dev.toml
3
dev.toml
@ -18,6 +18,9 @@
|
|||||||
target = '$1'
|
target = '$1'
|
||||||
|
|
||||||
[module]
|
[module]
|
||||||
|
[[module.mounts]]
|
||||||
|
source = "assets"
|
||||||
|
target = "assets"
|
||||||
[[module.mounts]]
|
[[module.mounts]]
|
||||||
source = "hugo_stats.json"
|
source = "hugo_stats.json"
|
||||||
target = "assets/watching/hugo_stats.json"
|
target = "assets/watching/hugo_stats.json"
|
||||||
|
@ -2,3 +2,75 @@
|
|||||||
title: Hextra Theme
|
title: Hextra Theme
|
||||||
layout: hextra-home
|
layout: hextra-home
|
||||||
---
|
---
|
||||||
|
|
||||||
|
{{< hextra/hero-badge >}}
|
||||||
|
<div class="w-2 h-2 rounded-full bg-primary-400"></div>
|
||||||
|
<span>Free, open source</span>
|
||||||
|
{{< icon name="arrow-circle-right" attributes="height=14" >}}
|
||||||
|
{{< /hextra/hero-badge >}}
|
||||||
|
|
||||||
|
<div class="mt-6 mb-6">
|
||||||
|
{{< hextra/hero-headline >}}
|
||||||
|
Build modern websites <br class="sm:block hidden" />with Markdown and Hugo
|
||||||
|
{{< /hextra/hero-headline >}}
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="mb-12">
|
||||||
|
{{< hextra/hero-subtitle >}}
|
||||||
|
Fast, batteries-included Hugo theme <br class="sm:block hidden" />for creating beautiful static websites
|
||||||
|
{{< /hextra/hero-subtitle >}}
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="mb-6">
|
||||||
|
{{< hextra/hero-button text="Get Started" link="docs" >}}
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="mt-6"></div>
|
||||||
|
|
||||||
|
{{< hextra/feature-grid >}}
|
||||||
|
{{< hextra/feature-card
|
||||||
|
title="Fast and Full-featured"
|
||||||
|
subtitle="Simple and easy to use, yet powerful and feature-rich."
|
||||||
|
class="aspect-auto md:aspect-[1.1/1] max-md:min-h-[340px]"
|
||||||
|
image="images/hextra-doc.webp"
|
||||||
|
imageClass="top-[40%] left-[24px] w-[180%] sm:w-[110%] dark:opacity-80"
|
||||||
|
style="background: radial-gradient(ellipse at 50% 80%,rgba(194,97,254,0.15),hsla(0,0%,100%,0));"
|
||||||
|
>}}
|
||||||
|
{{< hextra/feature-card
|
||||||
|
title="Markdown is All You Need"
|
||||||
|
subtitle="Compose with just Markdown. Enrich with Shortcode components."
|
||||||
|
class="aspect-auto md:aspect-[1.1/1] max-lg:min-h-[340px]"
|
||||||
|
image="images/hextra-markdown.webp"
|
||||||
|
imageClass="top-[40%] left-[36px] w-[180%] sm:w-[110%] dark:opacity-80"
|
||||||
|
style="background: radial-gradient(ellipse at 50% 80%,rgba(142,53,74,0.15),hsla(0,0%,100%,0));"
|
||||||
|
>}}
|
||||||
|
{{< hextra/feature-card
|
||||||
|
title="Full Text Search"
|
||||||
|
subtitle="Built-in full text search with FlexSearch, no extra setup required."
|
||||||
|
class="aspect-auto md:aspect-[1.1/1] max-md:min-h-[340px]"
|
||||||
|
image="images/hextra-search.webp"
|
||||||
|
imageClass="top-[40%] left-[36px] w-[110%] sm:w-[110%] dark:opacity-80"
|
||||||
|
style="background: radial-gradient(ellipse at 50% 80%,rgba(221,210,59,0.15),hsla(0,0%,100%,0));"
|
||||||
|
>}}
|
||||||
|
{{< hextra/feature-card
|
||||||
|
title="Lightweight as a Feather"
|
||||||
|
subtitle="No dependency or Node.js is needed to use Hextra. Powered by Hugo, one of *the fastest* static site generators, building your site in just seconds with a single binary."
|
||||||
|
>}}
|
||||||
|
{{< hextra/feature-card
|
||||||
|
title="Reponsive with Dark Mode Included"
|
||||||
|
subtitle="Looks great on different screen sizes. Built-in dark mode support, with auto-switching based on user's system preference."
|
||||||
|
>}}
|
||||||
|
{{< hextra/feature-card
|
||||||
|
title="Build and Host for Free"
|
||||||
|
subtitle="Build with GitHub Actions, and host for free on GitHub Pages. Alternatively it can be hosted on any static hosting service."
|
||||||
|
>}}
|
||||||
|
{{< hextra/feature-card
|
||||||
|
title="Multi-Language Made Easy"
|
||||||
|
subtitle="Create multi-language pages by just adding locales suffix to the Markdown file. Adding i18n support to your site is intuitive."
|
||||||
|
>}}
|
||||||
|
{{< hextra/feature-card
|
||||||
|
title="And Much More..."
|
||||||
|
icon="sparkles"
|
||||||
|
subtitle="Syntax highlighting / Table of contents / SEO / RSS / LaTeX / Mermaid / Customizable / and more..."
|
||||||
|
>}}
|
||||||
|
{{< /hextra/feature-grid >}}
|
||||||
|
@ -12,4 +12,5 @@ This section covers some advanced topics of the theme.
|
|||||||
{{< cards >}}
|
{{< cards >}}
|
||||||
{{< card link="multi-language" title="Multi-language" icon="translate" >}}
|
{{< card link="multi-language" title="Multi-language" icon="translate" >}}
|
||||||
{{< card link="customization" title="Customization" icon="pencil" >}}
|
{{< card link="customization" title="Customization" icon="pencil" >}}
|
||||||
|
{{< card link="comments" title="Comments System" icon="chat-alt" >}}
|
||||||
{{< /cards >}}
|
{{< /cards >}}
|
||||||
|
39
exampleSite/content/docs/advanced/comments.md
Normal file
39
exampleSite/content/docs/advanced/comments.md
Normal file
@ -0,0 +1,39 @@
|
|||||||
|
---
|
||||||
|
title: Comments System
|
||||||
|
linkTitle: Comments
|
||||||
|
---
|
||||||
|
|
||||||
|
Hextra supports adding comments system to your site.
|
||||||
|
Currently [giscus](https://giscus.app/) is supported.
|
||||||
|
|
||||||
|
<!--more-->
|
||||||
|
|
||||||
|
## giscus
|
||||||
|
|
||||||
|
[giscus](https://giscus.app/) is a comments system powered by [GitHub Discussions](https://docs.github.com/en/discussions). It is free and open source.
|
||||||
|
|
||||||
|
To enable giscus, you need to add the following to the site configuration file:
|
||||||
|
|
||||||
|
```yaml {filename="hugo.yaml"}
|
||||||
|
params:
|
||||||
|
comments:
|
||||||
|
enable: false
|
||||||
|
type: giscus
|
||||||
|
|
||||||
|
giscus:
|
||||||
|
repo: <repository>
|
||||||
|
repoId: <repository ID>
|
||||||
|
category: <category>
|
||||||
|
categoryId: <category ID>
|
||||||
|
```
|
||||||
|
|
||||||
|
The giscus configurations can be constructed from the [giscus.app](https://giscus.app/) website. More details can also be found there.
|
||||||
|
|
||||||
|
Comments can be enabled or disabled for a specific page in the page front matter:
|
||||||
|
|
||||||
|
```yaml {filename="content/docs/about.md"}
|
||||||
|
---
|
||||||
|
title: About
|
||||||
|
comments: true
|
||||||
|
---
|
||||||
|
```
|
@ -180,3 +180,29 @@ Include both `favicon.ico` and `favicon.svg` files in your project to ensure you
|
|||||||
|
|
||||||
While `favicon.ico` is generally for older browsers, `favicon.svg` is supported by modern ones. The optional `favicon-dark.svg` can be included for a tailored experience in dark mode.
|
While `favicon.ico` is generally for older browsers, `favicon.svg` is supported by modern ones. The optional `favicon-dark.svg` can be included for a tailored experience in dark mode.
|
||||||
Feel free to use tools like [favicon.io](https://favicon.io/) or [favycon](https://github.com/ruisaraiva19/favycon) to generate these icons.
|
Feel free to use tools like [favicon.io](https://favicon.io/) or [favycon](https://github.com/ruisaraiva19/favycon) to generate these icons.
|
||||||
|
|
||||||
|
|
||||||
|
### Page Width
|
||||||
|
|
||||||
|
The width of the page can be customized by the `params.page.width` parameter in the config file:
|
||||||
|
|
||||||
|
```yaml {filename="hugo.yaml"}
|
||||||
|
params:
|
||||||
|
page:
|
||||||
|
# full (100%), wide (90rem), normal (1280px)
|
||||||
|
width: wide
|
||||||
|
```
|
||||||
|
|
||||||
|
There are three available options: `full`, `wide`, and `normal`.
|
||||||
|
By default, the page width is set to `normal`.
|
||||||
|
|
||||||
|
Similarly, the width of the navbar and footer can be customized by the `params.navbar.width` and `params.footer.width` parameters.
|
||||||
|
|
||||||
|
|
||||||
|
### Google Analytics
|
||||||
|
|
||||||
|
To enable Google Analytics, set the `googleAnalytics` parameter in the config file:
|
||||||
|
|
||||||
|
```yaml {filename="hugo.yaml"}
|
||||||
|
googleAnalytics: G-XXXXXXXXXX
|
||||||
|
```
|
||||||
|
@ -63,3 +63,63 @@ weight: 2
|
|||||||
## Configure Content Directory
|
## Configure Content Directory
|
||||||
|
|
||||||
If we need to use a different directory for our content, we can do so by setting the [`contentDir`](https://gohugo.io/getting-started/configuration/#contentdir) parameter in our site configuration file.
|
If we need to use a different directory for our content, we can do so by setting the [`contentDir`](https://gohugo.io/getting-started/configuration/#contentdir) parameter in our site configuration file.
|
||||||
|
|
||||||
|
## Add Images
|
||||||
|
|
||||||
|
To add images, the easiest way is to put the image files in the same directory as the Markdown file.
|
||||||
|
For example, add an image file `image.png` alongside the `my-page.md` file:
|
||||||
|
|
||||||
|
{{< filetree/container >}}
|
||||||
|
{{< filetree/folder name="content" >}}
|
||||||
|
{{< filetree/folder name="docs" >}}
|
||||||
|
{{< filetree/file name="my-page.md" >}}
|
||||||
|
{{< filetree/file name="image.png" >}}
|
||||||
|
{{< /filetree/folder >}}
|
||||||
|
{{< /filetree/folder >}}
|
||||||
|
{{< /filetree/container >}}
|
||||||
|
|
||||||
|
Then, we can use the following Markdown syntax to add the image to the content:
|
||||||
|
|
||||||
|
```markdown {filename="content/docs/my-page.md"}
|
||||||
|

|
||||||
|
```
|
||||||
|
|
||||||
|
We can also utilize the [page bundles][page-bundles] feature of Hugo to organize the image files together with the Markdown file. To achieve that, turn the `my-page.md` file into a directory `my-page` and put the content into a file named `index.md`, and put the image files inside the `my-page` directory:
|
||||||
|
|
||||||
|
{{< filetree/container >}}
|
||||||
|
{{< filetree/folder name="content" >}}
|
||||||
|
{{< filetree/folder name="docs" >}}
|
||||||
|
{{< filetree/folder name="my-page" >}}
|
||||||
|
{{< filetree/file name="index.md" >}}
|
||||||
|
{{< filetree/file name="image.png" >}}
|
||||||
|
{{< /filetree/folder >}}
|
||||||
|
{{< /filetree/folder >}}
|
||||||
|
{{< /filetree/folder >}}
|
||||||
|
{{< /filetree/container >}}
|
||||||
|
|
||||||
|
```markdown {filename="content/docs/my-page/index.md"}
|
||||||
|

|
||||||
|
```
|
||||||
|
|
||||||
|
Alternatively, we can also put the image files in the `static` directory, which will make the images available for all pages:
|
||||||
|
|
||||||
|
{{< filetree/container >}}
|
||||||
|
{{< filetree/folder name="static" >}}
|
||||||
|
{{< filetree/folder name="images" >}}
|
||||||
|
{{< filetree/file name="image.png" >}}
|
||||||
|
{{< /filetree/folder >}}
|
||||||
|
{{< /filetree/folder >}}
|
||||||
|
{{< filetree/folder name="content" >}}
|
||||||
|
{{< filetree/folder name="docs" >}}
|
||||||
|
{{< filetree/file name="my-page.md" >}}
|
||||||
|
{{< /filetree/folder >}}
|
||||||
|
{{< /filetree/folder >}}
|
||||||
|
{{< /filetree/container >}}
|
||||||
|
|
||||||
|
Note that the image path begins with a slash `/` and is relative to the static directory:
|
||||||
|
|
||||||
|
```markdown {filename="content/docs/my-page.md"}
|
||||||
|

|
||||||
|
```
|
||||||
|
|
||||||
|
[page-bundles]: https://gohugo.io/content-management/page-bundles/#leaf-bundles
|
||||||
|
@ -12,8 +12,8 @@ linkTitle: Cards
|
|||||||
|
|
||||||
{{< cards >}}
|
{{< cards >}}
|
||||||
{{< card link="/" title="Image Card" image="https://source.unsplash.com/featured/800x600?landscape" subtitle="Unsplash Landscape" >}}
|
{{< card link="/" title="Image Card" image="https://source.unsplash.com/featured/800x600?landscape" subtitle="Unsplash Landscape" >}}
|
||||||
{{< card link="/" title="Local Image" image="/images/space.jpg" subtitle="Image under assets directory, processed by Hugo." method="Resize" options="600x q80 webp" >}}
|
|
||||||
{{< card link="/" title="Local Image" image="/images/card-image-unprocessed.jpg" subtitle="Raw image under static directory." >}}
|
{{< card link="/" title="Local Image" image="/images/card-image-unprocessed.jpg" subtitle="Raw image under static directory." >}}
|
||||||
|
{{< card link="/" title="Local Image" image="images/space.jpg" subtitle="Image under assets directory, processed by Hugo." method="Resize" options="600x q80 webp" >}}
|
||||||
{{< /cards >}}
|
{{< /cards >}}
|
||||||
|
|
||||||
## Usage
|
## Usage
|
||||||
@ -28,8 +28,8 @@ linkTitle: Cards
|
|||||||
```
|
```
|
||||||
{{</* cards */>}}
|
{{</* cards */>}}
|
||||||
{{</* card link="/" title="Image Card" image="https://source.unsplash.com/featured/800x600?landscape" subtitle="Unsplash Landscape" */>}}
|
{{</* card link="/" title="Image Card" image="https://source.unsplash.com/featured/800x600?landscape" subtitle="Unsplash Landscape" */>}}
|
||||||
{{</* card link="/" title="Local Image" image="/images/space.jpg" subtitle="Image under assets directory, processed by Hugo." method="Resize" options="600x q80 webp" */>}}
|
|
||||||
{{</* card link="/" title="Local Image" image="/images/card-image-unprocessed.jpg" subtitle="Raw image under static directory." */>}}
|
{{</* card link="/" title="Local Image" image="/images/card-image-unprocessed.jpg" subtitle="Raw image under static directory." */>}}
|
||||||
|
{{</* card link="/" title="Local Image" image="images/space.jpg" subtitle="Image under assets directory, processed by Hugo." method="Resize" options="600x q80 webp" */>}}
|
||||||
{{</* /cards */>}}
|
{{</* /cards */>}}
|
||||||
```
|
```
|
||||||
|
|
||||||
@ -52,7 +52,7 @@ Additionally, the card supports adding image and processing through these parame
|
|||||||
| `method` | Sets Hugo's image processing method. |
|
| `method` | Sets Hugo's image processing method. |
|
||||||
| `options` | Configures Hugo's image processing options. |
|
| `options` | Configures Hugo's image processing options. |
|
||||||
|
|
||||||
Card supports three kinds of images:
|
Card supports three kinds of images:
|
||||||
|
|
||||||
1. Remote image: the full URL in the `image` parameter.
|
1. Remote image: the full URL in the `image` parameter.
|
||||||
2. Static image: use the relative path in Hugo's `static/` directory.
|
2. Static image: use the relative path in Hugo's `static/` directory.
|
||||||
|
18
exampleSite/content/showcase/index.md
Normal file
18
exampleSite/content/showcase/index.md
Normal file
@ -0,0 +1,18 @@
|
|||||||
|
---
|
||||||
|
title: Showcase
|
||||||
|
description: "Open source projects powered by Hextra."
|
||||||
|
toc: false
|
||||||
|
layout: wide
|
||||||
|
---
|
||||||
|
|
||||||
|
<div class="mt-4"></div>
|
||||||
|
|
||||||
|
<p class="mb-12 text-center text-lg text-gray-500 dark:text-gray-400">
|
||||||
|
Open source projects powered by Hextra.
|
||||||
|
</p>
|
||||||
|
|
||||||
|
{{< cards >}}
|
||||||
|
{{< card link="https://getporter.org/" title="Porter" image="https://repository-images.githubusercontent.com/155893691/aa249c80-fcf3-11ea-93b0-30079e8d7de4" imageStyle="object-fit:cover; aspect-ratio:16/9;" >}}
|
||||||
|
{{< card link="https://lutheranconfessions.org/" title="LutheranConfessions" image="https://github.com/imfing/hextra/assets/5097752/ad6625e4-88cd-4cad-b102-5399997d0359" imageStyle="object-fit:cover; aspect-ratio:16/9;" >}}
|
||||||
|
{{< card link="/" title="Hextra Starter Template" image="https://user-images.githubusercontent.com/5097752/263551418-c403b9a9-a76c-47a6-8466-513d772ef0b7.jpg" imageStyle="object-fit:cover; aspect-ratio:16/9;" >}}
|
||||||
|
{{< /cards >}}
|
@ -7,6 +7,8 @@ enableGitInfo: true
|
|||||||
# enableEmoji: false
|
# enableEmoji: false
|
||||||
hasCJKLanguage: true
|
hasCJKLanguage: true
|
||||||
|
|
||||||
|
# googleAnalytics: G-XXXXXXXXXX
|
||||||
|
|
||||||
outputs:
|
outputs:
|
||||||
home: [HTML]
|
home: [HTML]
|
||||||
page: [HTML]
|
page: [HTML]
|
||||||
@ -24,13 +26,14 @@ languages:
|
|||||||
title: "Hextra テーマ"
|
title: "Hextra テーマ"
|
||||||
zh-cn:
|
zh-cn:
|
||||||
languageName: 简体中文
|
languageName: 简体中文
|
||||||
|
languageCode: zh-CN
|
||||||
weight: 3
|
weight: 3
|
||||||
title: Hextra
|
title: Hextra
|
||||||
|
|
||||||
module:
|
module:
|
||||||
hugoVersion:
|
hugoVersion:
|
||||||
extended: true
|
extended: true
|
||||||
min: "0.111.0"
|
min: "0.112.0"
|
||||||
|
|
||||||
workspace: hugo.work
|
workspace: hugo.work
|
||||||
imports:
|
imports:
|
||||||
@ -51,20 +54,24 @@ menu:
|
|||||||
name: Documentation
|
name: Documentation
|
||||||
pageRef: /docs
|
pageRef: /docs
|
||||||
weight: 1
|
weight: 1
|
||||||
|
- identifier: showcase
|
||||||
|
name: Showcase
|
||||||
|
pageRef: /showcase
|
||||||
|
weight: 2
|
||||||
- identifier: blog
|
- identifier: blog
|
||||||
name: Blog
|
name: Blog
|
||||||
pageRef: /blog
|
pageRef: /blog
|
||||||
weight: 2
|
weight: 3
|
||||||
- identifier: about
|
- identifier: about
|
||||||
name: About
|
name: About
|
||||||
pageRef: /about
|
pageRef: /about
|
||||||
weight: 3
|
|
||||||
- name: Search
|
|
||||||
weight: 4
|
weight: 4
|
||||||
|
- name: Search
|
||||||
|
weight: 5
|
||||||
params:
|
params:
|
||||||
type: search
|
type: search
|
||||||
- name: GitHub
|
- name: GitHub
|
||||||
weight: 5
|
weight: 6
|
||||||
url: "https://github.com/imfing/hextra"
|
url: "https://github.com/imfing/hextra"
|
||||||
params:
|
params:
|
||||||
icon: github
|
icon: github
|
||||||
@ -93,13 +100,19 @@ params:
|
|||||||
logo:
|
logo:
|
||||||
path: images/logo.svg
|
path: images/logo.svg
|
||||||
dark: images/logo-dark.svg
|
dark: images/logo-dark.svg
|
||||||
# link: /
|
|
||||||
# width: 40
|
# width: 40
|
||||||
# height: 20
|
# height: 20
|
||||||
|
# link: /
|
||||||
|
width: wide
|
||||||
|
|
||||||
|
page:
|
||||||
|
# full (100%), wide (90rem), normal (1280px)
|
||||||
|
width: normal
|
||||||
|
|
||||||
footer:
|
footer:
|
||||||
displayCopyright: true
|
displayCopyright: true
|
||||||
displayPoweredBy: true
|
displayPoweredBy: true
|
||||||
|
width: normal
|
||||||
|
|
||||||
displayUpdatedDate: true
|
displayUpdatedDate: true
|
||||||
dateFormat: "January 2, 2006"
|
dateFormat: "January 2, 2006"
|
||||||
@ -110,3 +123,20 @@ params:
|
|||||||
editURL:
|
editURL:
|
||||||
enable: true
|
enable: true
|
||||||
base: "https://github.com/imfing/hextra/edit/main/exampleSite/content"
|
base: "https://github.com/imfing/hextra/edit/main/exampleSite/content"
|
||||||
|
|
||||||
|
comments:
|
||||||
|
enable: false
|
||||||
|
type: giscus
|
||||||
|
|
||||||
|
# https://giscus.app/
|
||||||
|
giscus:
|
||||||
|
repo: imfing/hextra
|
||||||
|
repoId: R_kgDOJ9fJag
|
||||||
|
category: General
|
||||||
|
categoryId: DIC_kwDOJ9fJas4CY7gW
|
||||||
|
# mapping: pathname
|
||||||
|
# strict: 0
|
||||||
|
# reactionsEnabled: 1
|
||||||
|
# emitMetadata: 0
|
||||||
|
# inputPosition: top
|
||||||
|
# lang: en
|
||||||
|
@ -40,7 +40,6 @@
|
|||||||
"path",
|
"path",
|
||||||
"pre",
|
"pre",
|
||||||
"script",
|
"script",
|
||||||
"section",
|
|
||||||
"span",
|
"span",
|
||||||
"strong",
|
"strong",
|
||||||
"style",
|
"style",
|
||||||
@ -94,7 +93,6 @@
|
|||||||
"bg-black/[.05]",
|
"bg-black/[.05]",
|
||||||
"bg-blue-100",
|
"bg-blue-100",
|
||||||
"bg-clip-text",
|
"bg-clip-text",
|
||||||
"bg-gradient-to-b",
|
|
||||||
"bg-gradient-to-r",
|
"bg-gradient-to-r",
|
||||||
"bg-gray-100",
|
"bg-gray-100",
|
||||||
"bg-neutral-50",
|
"bg-neutral-50",
|
||||||
@ -228,6 +226,7 @@
|
|||||||
"data-[state=selected]:text-primary-600",
|
"data-[state=selected]:text-primary-600",
|
||||||
"decoration-from-font",
|
"decoration-from-font",
|
||||||
"duration-200",
|
"duration-200",
|
||||||
|
"ease-in",
|
||||||
"ease-in-out",
|
"ease-in-out",
|
||||||
"filename",
|
"filename",
|
||||||
"first:mt-0",
|
"first:mt-0",
|
||||||
@ -246,7 +245,6 @@
|
|||||||
"footnote-backref",
|
"footnote-backref",
|
||||||
"footnote-ref",
|
"footnote-ref",
|
||||||
"footnotes",
|
"footnotes",
|
||||||
"from-gray-800",
|
|
||||||
"from-gray-900",
|
"from-gray-900",
|
||||||
"gap-1",
|
"gap-1",
|
||||||
"gap-2",
|
"gap-2",
|
||||||
@ -274,6 +272,7 @@
|
|||||||
"hamburger-menu",
|
"hamburger-menu",
|
||||||
"hextra-card",
|
"hextra-card",
|
||||||
"hextra-cards",
|
"hextra-cards",
|
||||||
|
"hextra-feature-card",
|
||||||
"hextra-filetree",
|
"hextra-filetree",
|
||||||
"hextra-filetree-folder",
|
"hextra-filetree-folder",
|
||||||
"hextra-footer",
|
"hextra-footer",
|
||||||
@ -320,6 +319,7 @@
|
|||||||
"leading-5",
|
"leading-5",
|
||||||
"leading-6",
|
"leading-6",
|
||||||
"leading-7",
|
"leading-7",
|
||||||
|
"leading-none",
|
||||||
"leading-tight",
|
"leading-tight",
|
||||||
"left-[24px]",
|
"left-[24px]",
|
||||||
"left-[36px]",
|
"left-[36px]",
|
||||||
@ -367,6 +367,7 @@
|
|||||||
"mb-16",
|
"mb-16",
|
||||||
"mb-2",
|
"mb-2",
|
||||||
"mb-4",
|
"mb-4",
|
||||||
|
"mb-6",
|
||||||
"mb-8",
|
"mb-8",
|
||||||
"md:aspect-[1.1/1]",
|
"md:aspect-[1.1/1]",
|
||||||
"md:h-[calc(100vh-var(--navbar-height)-var(--menu-height))]",
|
"md:h-[calc(100vh-var(--navbar-height)-var(--menu-height))]",
|
||||||
@ -439,8 +440,10 @@
|
|||||||
"pl-[max(env(safe-area-inset-left),1.5rem)]",
|
"pl-[max(env(safe-area-inset-left),1.5rem)]",
|
||||||
"placeholder:text-gray-500",
|
"placeholder:text-gray-500",
|
||||||
"pointer-events-none",
|
"pointer-events-none",
|
||||||
|
"pr-2",
|
||||||
"pr-4",
|
"pr-4",
|
||||||
"pr-[calc(env(safe-area-inset-right)-1.5rem)]",
|
"pr-[calc(env(safe-area-inset-right)-1.5rem)]",
|
||||||
|
"pr-[max(env(safe-area-inset-left),1.5rem)]",
|
||||||
"pr-[max(env(safe-area-inset-right),1.5rem)]",
|
"pr-[max(env(safe-area-inset-right),1.5rem)]",
|
||||||
"print:bg-transparent",
|
"print:bg-transparent",
|
||||||
"print:hidden",
|
"print:hidden",
|
||||||
@ -512,7 +515,6 @@
|
|||||||
"sm:flex",
|
"sm:flex",
|
||||||
"sm:grid-cols-2",
|
"sm:grid-cols-2",
|
||||||
"sm:items-start",
|
"sm:items-start",
|
||||||
"sm:px-4",
|
|
||||||
"sm:text-xl",
|
"sm:text-xl",
|
||||||
"sm:w-[110%]",
|
"sm:w-[110%]",
|
||||||
"sr-only",
|
"sr-only",
|
||||||
|
@ -1,8 +1,27 @@
|
|||||||
{{- if .Title -}}
|
{{- $alt := .PlainText | safeHTML -}}
|
||||||
|
{{- $lazyLoading := .Page.Site.Params.enableImagelazyLoading | default true -}}
|
||||||
|
{{- $dest := .Destination -}}
|
||||||
|
|
||||||
|
{{- $isRemote := not (urls.Parse $dest).Scheme -}}
|
||||||
|
{{- $isPage := and (eq .Page.Kind "page") (not .Page.BundleType) -}}
|
||||||
|
{{- $startsWithSlash := hasPrefix $dest "/" -}}
|
||||||
|
{{- $startsWithRelative := hasPrefix $dest "../" -}}
|
||||||
|
|
||||||
|
{{- if and $dest $isRemote -}}
|
||||||
|
{{- if $startsWithSlash -}}
|
||||||
|
{{/* Images under static directory */}}
|
||||||
|
{{- $dest = (relURL (strings.TrimPrefix "/" $dest)) -}}
|
||||||
|
{{- else if and $isPage (not $startsWithRelative) -}}
|
||||||
|
{{/* Images that are sibling to the individual page file */}}
|
||||||
|
{{ $dest = (printf "../%s" $dest) }}
|
||||||
|
{{- end -}}
|
||||||
|
{{- end -}}
|
||||||
|
|
||||||
|
{{- with .Title -}}
|
||||||
<figure>
|
<figure>
|
||||||
<img src="{{ .Destination | safeURL }}" title="{{ .Title }}" alt="{{ .PlainText | safeHTML }}" loading="lazy" />
|
<img src="{{ $dest | safeURL }}" title="{{ . }}" alt="{{ $alt }}" {{ if $lazyLoading }}loading="lazy"{{ end }} />
|
||||||
<figcaption>{{ .Title }}</figcaption>
|
<figcaption>{{ . }}</figcaption>
|
||||||
</figure>
|
</figure>
|
||||||
{{- else -}}
|
{{- else -}}
|
||||||
<img src="{{ .Destination | safeURL }}" alt="{{ .PlainText | safeHTML }}" loading="lazy" />
|
<img src="{{ $dest | safeURL }}" alt="{{ $alt }}" {{ if $lazyLoading }}loading="lazy"{{ end }} />
|
||||||
{{- end -}}
|
{{- end -}}
|
||||||
|
@ -1,5 +1,5 @@
|
|||||||
{{ define "main" }}
|
{{ define "main" }}
|
||||||
<div class="mx-auto flex max-w-screen-xl">
|
<div class='mx-auto flex {{ partial "utils/page-width" . }}'>
|
||||||
{{ partial "sidebar.html" (dict "context" .) }}
|
{{ partial "sidebar.html" (dict "context" .) }}
|
||||||
{{ partial "toc.html" . }}
|
{{ partial "toc.html" . }}
|
||||||
<article class="w-full break-words flex min-h-[calc(100vh-var(--navbar-height))] min-w-0 justify-center pb-8 pr-[calc(env(safe-area-inset-right)-1.5rem)]">
|
<article class="w-full break-words flex min-h-[calc(100vh-var(--navbar-height))] min-w-0 justify-center pb-8 pr-[calc(env(safe-area-inset-right)-1.5rem)]">
|
||||||
@ -10,6 +10,7 @@
|
|||||||
</div>
|
</div>
|
||||||
<div class="mt-16"></div>
|
<div class="mt-16"></div>
|
||||||
{{ partial "components/last-updated.html" . }}
|
{{ partial "components/last-updated.html" . }}
|
||||||
|
{{ partial "components/comments.html" . }}
|
||||||
</main>
|
</main>
|
||||||
</article>
|
</article>
|
||||||
</div>
|
</div>
|
||||||
|
43
layouts/_default/list.rss.xml
Normal file
43
layouts/_default/list.rss.xml
Normal file
@ -0,0 +1,43 @@
|
|||||||
|
<rss version="2.0" xmlns:atom="http://www.w3.org/2005/Atom">
|
||||||
|
<channel>
|
||||||
|
<title>{{ .Site.Title }} – {{ .Title }}</title>
|
||||||
|
<link>{{ .Permalink }}</link>
|
||||||
|
<description>Recent content {{ if ne .Title .Site.Title }}{{ with .Title }}in {{.}} {{ end }}{{ end }}on {{ .Site.Title }}</description>
|
||||||
|
<generator>Hugo -- gohugo.io</generator>{{ with .Site.LanguageCode }}
|
||||||
|
<language>{{.}}</language>{{end}}{{ with .Site.Author.email }}
|
||||||
|
<managingEditor>{{.}}{{ with $.Site.Author.name }} ({{.}}){{end}}</managingEditor>{{end}}{{ with .Site.Author.email }}
|
||||||
|
<webMaster>{{.}}{{ with $.Site.Author.name }} ({{.}}){{end}}</webMaster>{{end}}{{ with .Site.Copyright }}
|
||||||
|
<copyright>{{.}}</copyright>{{end}}{{ if not .Date.IsZero }}
|
||||||
|
<lastBuildDate>{{ .Date.Format "Mon, 02 Jan 2006 15:04:05 -0700" | safeHTML }}</lastBuildDate>{{ end }}
|
||||||
|
{{ with .OutputFormats.Get "RSS" }}
|
||||||
|
{{ printf "<atom:link href=%q rel=\"self\" type=%q />" .Permalink .MediaType | safeHTML }}
|
||||||
|
{{ end }}
|
||||||
|
{{ if not $.Section }}
|
||||||
|
{{ $sections := .Site.Params.rss.sections | default (slice "blog") }}
|
||||||
|
{{ .Scratch.Set "rssPages" (first 50 (where $.Site.RegularPages "Type" "in" $sections )) }}
|
||||||
|
{{ else }}
|
||||||
|
{{ if $.Parent.IsHome }}
|
||||||
|
{{ .Scratch.Set "rssPages" (first 50 (where $.Site.RegularPages "Type" $.Section )) }}
|
||||||
|
{{ else }}
|
||||||
|
{{ .Scratch.Set "rssPages" (first 50 $.Pages) }}
|
||||||
|
{{ end }}
|
||||||
|
{{ end }}
|
||||||
|
{{ range (.Scratch.Get "rssPages") }}
|
||||||
|
<item>
|
||||||
|
<title>{{ .Title }}</title>
|
||||||
|
<link>{{ .Permalink }}</link>
|
||||||
|
<pubDate>{{ .Date.Format "Mon, 02 Jan 2006 15:04:05 -0700" | safeHTML }}</pubDate>
|
||||||
|
{{ with .Site.Author.email }}<author>{{.}}{{ with $.Site.Author.name }} ({{.}}){{end}}</author>{{end}}
|
||||||
|
<guid>{{ .Permalink }}</guid>
|
||||||
|
<description>
|
||||||
|
{{ $img := (.Resources.ByType "image").GetMatch "*featured*" }}
|
||||||
|
{{ with $img }}
|
||||||
|
{{ $img := .Resize "640x" }}
|
||||||
|
{{ printf "<![CDATA[<img src=\"%s\" width=\"%d\" height=\"%d\"/>]]>" $img.Permalink $img.Width $img.Height | safeHTML }}
|
||||||
|
{{ end }}
|
||||||
|
{{ .Content | html }}
|
||||||
|
</description>
|
||||||
|
</item>
|
||||||
|
{{ end }}
|
||||||
|
</channel>
|
||||||
|
</rss>
|
@ -1,5 +1,5 @@
|
|||||||
{{ define "main" }}
|
{{ define "main" }}
|
||||||
<div class="mx-auto flex max-w-screen-xl">
|
<div class='mx-auto flex {{ partial "utils/page-width" . }}'>
|
||||||
{{ partial "sidebar.html" (dict "context" . "disableSidebar" true "displayPlaceholder" true) }}
|
{{ partial "sidebar.html" (dict "context" . "disableSidebar" true "displayPlaceholder" true) }}
|
||||||
{{ partial "toc.html" . }}
|
{{ partial "toc.html" . }}
|
||||||
<article class="w-full break-words flex min-h-[calc(100vh-var(--navbar-height))] min-w-0 justify-center pb-8 pr-[calc(env(safe-area-inset-right)-1.5rem)]">
|
<article class="w-full break-words flex min-h-[calc(100vh-var(--navbar-height))] min-w-0 justify-center pb-8 pr-[calc(env(safe-area-inset-right)-1.5rem)]">
|
||||||
@ -10,6 +10,8 @@
|
|||||||
<div class="content">
|
<div class="content">
|
||||||
{{ .Content }}
|
{{ .Content }}
|
||||||
</div>
|
</div>
|
||||||
|
<div class="mt-16"></div>
|
||||||
|
{{ partial "components/comments.html" . }}
|
||||||
</main>
|
</main>
|
||||||
</article>
|
</article>
|
||||||
</div>
|
</div>
|
||||||
|
12
layouts/_default/wide.html
Normal file
12
layouts/_default/wide.html
Normal file
@ -0,0 +1,12 @@
|
|||||||
|
{{ define "main" }}
|
||||||
|
<div class="mx-auto flex max-w-[90rem]">
|
||||||
|
{{ partial "sidebar.html" (dict "context" . "disableSidebar" true "displayPlaceholder" false) }}
|
||||||
|
<article class="w-full break-words min-h-[calc(100vh-var(--navbar-height))] min-w-0 pt-4 pb-8 pl-[max(env(safe-area-inset-left),1.5rem)] pr-[max(env(safe-area-inset-left),1.5rem)]">
|
||||||
|
<br class="mt-1.5 text-sm" />
|
||||||
|
<h1 class="text-center mt-2 text-4xl font-bold tracking-tight text-slate-900 dark:text-slate-100">{{ .Title }}</h1>
|
||||||
|
<div class="content">
|
||||||
|
{{ .Content }}
|
||||||
|
</div>
|
||||||
|
</article>
|
||||||
|
</div>
|
||||||
|
{{ end }}
|
@ -1,12 +1,12 @@
|
|||||||
{{ define "main" }}
|
{{ define "main" }}
|
||||||
<div class="mx-auto flex max-w-screen-xl">
|
<div class='mx-auto flex {{ partial "utils/page-width" . }}'>
|
||||||
{{ partial "sidebar.html" (dict "context" . "disableSidebar" true "displayPlaceholder" true) }}
|
{{ partial "sidebar.html" (dict "context" . "disableSidebar" true "displayPlaceholder" true) }}
|
||||||
<article class="w-full break-words flex min-h-[calc(100vh-var(--navbar-height))] min-w-0 justify-center pb-8 pr-[calc(env(safe-area-inset-right)-1.5rem)]">
|
<article class="w-full break-words flex min-h-[calc(100vh-var(--navbar-height))] min-w-0 justify-center pb-8 pr-[calc(env(safe-area-inset-right)-1.5rem)]">
|
||||||
<main class="w-full min-w-0 max-w-6xl px-6 pt-4 md:px-12">
|
<main class="w-full min-w-0 max-w-6xl px-6 pt-4 md:px-12">
|
||||||
<br class="mt-1.5 text-sm" />
|
<br class="mt-1.5 text-sm" />
|
||||||
<h1 class="text-center mt-2 text-4xl font-bold tracking-tight text-slate-900 dark:text-slate-100">{{ .Title }}</h1>
|
<h1 class="text-center mt-2 text-4xl font-bold tracking-tight text-slate-900 dark:text-slate-100">{{ .Title }}</h1>
|
||||||
<div class="content">{{ .Content }}</div>
|
<div class="content">{{ .Content }}</div>
|
||||||
{{ range .Pages.ByDate }}
|
{{ range .Pages.ByDate.Reverse }}
|
||||||
<div class="mb-10">
|
<div class="mb-10">
|
||||||
<h3><a style="color: inherit; text-decoration: none;" class="block font-semibold mt-8 text-2xl " href="{{ .RelPermalink }}">{{ .Title }}</a></h3>
|
<h3><a style="color: inherit; text-decoration: none;" class="block font-semibold mt-8 text-2xl " href="{{ .RelPermalink }}">{{ .Title }}</a></h3>
|
||||||
<p class="opacity-80 mt-6 leading-7">
|
<p class="opacity-80 mt-6 leading-7">
|
||||||
|
@ -1,5 +1,5 @@
|
|||||||
{{ define "main" }}
|
{{ define "main" }}
|
||||||
<div class="mx-auto flex max-w-screen-xl">
|
<div class='mx-auto flex {{ partial "utils/page-width" . }}'>
|
||||||
{{ partial "sidebar.html" (dict "context" . "disableSidebar" true "displayPlaceholder" true) }}
|
{{ partial "sidebar.html" (dict "context" . "disableSidebar" true "displayPlaceholder" true) }}
|
||||||
{{ partial "toc.html" . }}
|
{{ partial "toc.html" . }}
|
||||||
<article class="w-full break-words flex min-h-[calc(100vh-var(--navbar-height))] min-w-0 justify-center pb-8 pr-[calc(env(safe-area-inset-right)-1.5rem)]">
|
<article class="w-full break-words flex min-h-[calc(100vh-var(--navbar-height))] min-w-0 justify-center pb-8 pr-[calc(env(safe-area-inset-right)-1.5rem)]">
|
||||||
@ -28,6 +28,7 @@
|
|||||||
{{ partial "components/last-updated.html" . }}
|
{{ partial "components/last-updated.html" . }}
|
||||||
{{ .Scratch.Set "reversePagination" true }}
|
{{ .Scratch.Set "reversePagination" true }}
|
||||||
{{ partial "components/pager.html" . }}
|
{{ partial "components/pager.html" . }}
|
||||||
|
{{ partial "components/comments.html" . }}
|
||||||
</main>
|
</main>
|
||||||
</article>
|
</article>
|
||||||
</div>
|
</div>
|
||||||
|
@ -1,5 +1,5 @@
|
|||||||
{{ define "main" }}
|
{{ define "main" }}
|
||||||
<div class="mx-auto flex max-w-screen-xl">
|
<div class='mx-auto flex {{ partial "utils/page-width" . }}'>
|
||||||
{{ partial "sidebar.html" (dict "context" .) }}
|
{{ partial "sidebar.html" (dict "context" .) }}
|
||||||
{{ partial "toc.html" . }}
|
{{ partial "toc.html" . }}
|
||||||
<article class="w-full break-words flex min-h-[calc(100vh-var(--navbar-height))] min-w-0 justify-center pb-8 pr-[calc(env(safe-area-inset-right)-1.5rem)]">
|
<article class="w-full break-words flex min-h-[calc(100vh-var(--navbar-height))] min-w-0 justify-center pb-8 pr-[calc(env(safe-area-inset-right)-1.5rem)]">
|
||||||
@ -9,9 +9,9 @@
|
|||||||
<h1>{{ .Title }}</h1>
|
<h1>{{ .Title }}</h1>
|
||||||
{{ .Content }}
|
{{ .Content }}
|
||||||
</div>
|
</div>
|
||||||
<div class="mt-16"></div>
|
|
||||||
{{ partial "components/last-updated.html" . }}
|
{{ partial "components/last-updated.html" . }}
|
||||||
{{ partial "components/pager.html" . }}
|
{{ partial "components/pager.html" . }}
|
||||||
|
{{ partial "components/comments.html" . }}
|
||||||
</main>
|
</main>
|
||||||
</article>
|
</article>
|
||||||
</div>
|
</div>
|
||||||
|
@ -1,5 +1,5 @@
|
|||||||
{{ define "main" }}
|
{{ define "main" }}
|
||||||
<div class="mx-auto flex max-w-screen-xl">
|
<div class='mx-auto flex {{ partial "utils/page-width" . }}'>
|
||||||
{{ partial "sidebar.html" (dict "context" .) }}
|
{{ partial "sidebar.html" (dict "context" .) }}
|
||||||
{{ partial "toc.html" . }}
|
{{ partial "toc.html" . }}
|
||||||
<article class="w-full break-words flex min-h-[calc(100vh-var(--navbar-height))] min-w-0 justify-center pb-8 pr-[calc(env(safe-area-inset-right)-1.5rem)]">
|
<article class="w-full break-words flex min-h-[calc(100vh-var(--navbar-height))] min-w-0 justify-center pb-8 pr-[calc(env(safe-area-inset-right)-1.5rem)]">
|
||||||
@ -11,6 +11,7 @@
|
|||||||
</div>
|
</div>
|
||||||
{{ partial "components/last-updated.html" . }}
|
{{ partial "components/last-updated.html" . }}
|
||||||
{{ partial "components/pager.html" . }}
|
{{ partial "components/pager.html" . }}
|
||||||
|
{{ partial "components/comments.html" . }}
|
||||||
</main>
|
</main>
|
||||||
</article>
|
</article>
|
||||||
</div>
|
</div>
|
||||||
|
@ -1,118 +1,10 @@
|
|||||||
{{ define "main" }}
|
{{ define "main" }}
|
||||||
<div class="mx-auto flex max-w-screen-xl">
|
<div class='mx-auto flex {{ partial "utils/page-width" . }}'>
|
||||||
{{ partial "sidebar.html" (dict "context" . "disableSidebar" true) }}
|
{{ partial "sidebar.html" (dict "context" . "disableSidebar" true) }}
|
||||||
<div class="w-full break-words flex flex-col min-h-[calc(100vh-var(--navbar-height))] min-w-0 justify-start pb-8 pr-[calc(env(safe-area-inset-right)-1.5rem)]">
|
<div class="w-full break-words min-h-[calc(100vh-var(--navbar-height))] min-w-0 pb-8 pt-8 md:pt-12 pl-[max(env(safe-area-inset-left),1.5rem)] pr-[max(env(safe-area-inset-left),1.5rem)]">
|
||||||
<section class="flex max-w-[90rem] flex-col items-start gap-2 px-6 sm:px-4 pt-8 md:pt-12">
|
<div class="flex flex-col items-start">
|
||||||
<a
|
{{ .Content }}
|
||||||
href="https://github.com/imfing/hextra"
|
</div>
|
||||||
target="_blank"
|
|
||||||
rel="noreferrer"
|
|
||||||
class="inline-flex items-center rounded-full gap-2 px-3 py-1 text-xs text-gray-600 dark:text-gray-400 bg-gray-100 dark:bg-neutral-800 dark:border-neutral-800 border hover:border-gray-400 dark:hover:text-gray-50 dark:hover:border-gray-600"
|
|
||||||
>
|
|
||||||
<div class="w-2 h-2 rounded-full bg-primary-400"></div>
|
|
||||||
<span>Free, open source</span>
|
|
||||||
{{- partial "utils/icon" (dict "name" "arrow-circle-right" "attributes" "height=14") -}}
|
|
||||||
</a>
|
|
||||||
<h1
|
|
||||||
class="text-4xl font-bold leading-tight tracking-tighter md:text-5xl mt-6 bg-clip-text text-transparent bg-gradient-to-r from-gray-900 to-gray-600 dark:from-gray-100 dark:to-gray-400"
|
|
||||||
>
|
|
||||||
Build modern websites <br class="sm:block hidden" />
|
|
||||||
with Markdown and Hugo
|
|
||||||
</h1>
|
|
||||||
<p class="mt-4 text-xl text-gray-600 dark:text-gray-400 sm:text-xl">
|
|
||||||
Fast, batteries-included Hugo theme <br class="sm:block hidden" />
|
|
||||||
for creating beautiful static websites.
|
|
||||||
</p>
|
|
||||||
<div class="mt-8">
|
|
||||||
{{- $docsURL := "docs" | relURL -}}
|
|
||||||
<a
|
|
||||||
href="{{ $docsURL }}"
|
|
||||||
class="font-medium cursor-pointer px-6 py-3 text-white bg-primary-600 hover:bg-primary-700 focus:outline-none focus:ring-4 focus:ring-primary-300 rounded-full text-center mr-2 mb-2 dark:bg-primary-600 dark:hover:bg-primary-700 dark:focus:ring-primary-800"
|
|
||||||
>
|
|
||||||
Get Started
|
|
||||||
</a>
|
|
||||||
</div>
|
|
||||||
</section>
|
|
||||||
<section class="flex max-w-[90rem] flex-col items-start gap-2 px-6 sm:px-4 pt-8 pb-8">
|
|
||||||
<h2 class="mt-12 text-2xl font-bold bg-clip-text text-transparent bg-gradient-to-b from-gray-800 to-gray-600 dark:from-gray-100 dark:to-gray-400">What's in Hextra?</h2>
|
|
||||||
<div class="mt-6 grid sm:grid-cols-2 lg:grid-cols-3 grid-cols-1 gap-4 w-full">
|
|
||||||
{{ template "hextra-feature-card" (dict
|
|
||||||
"title" "Fast and Full-featured"
|
|
||||||
"subtitle" "Simple and easy to use, yet powerful and feature-rich. "
|
|
||||||
"class" "aspect-auto md:aspect-[1.1/1] max-md:min-h-[340px]"
|
|
||||||
"image" "images/hextra-doc.webp"
|
|
||||||
"imageClass" "top-[40%] left-[24px] w-[180%] sm:w-[110%] dark:opacity-80"
|
|
||||||
"style" "background: radial-gradient(ellipse at 50% 80%,rgba(194,97,254,0.15),hsla(0,0%,100%,0));"
|
|
||||||
)
|
|
||||||
}}
|
|
||||||
{{ template "hextra-feature-card" (dict
|
|
||||||
"title" "Markdown is All You Need"
|
|
||||||
"subtitle" "Compose with just Markdown. Enrich with Shortcode components."
|
|
||||||
"class" "aspect-auto md:aspect-[1.1/1] max-lg:min-h-[340px]"
|
|
||||||
"image" "images/hextra-markdown.webp"
|
|
||||||
"imageClass" "top-[40%] left-[36px] w-[180%] sm:w-[110%] dark:opacity-80"
|
|
||||||
"style" "background: radial-gradient(ellipse at 50% 80%,rgba(142,53,74,0.15),hsla(0,0%,100%,0));"
|
|
||||||
)
|
|
||||||
}}
|
|
||||||
{{ template "hextra-feature-card" (dict
|
|
||||||
"title" "Full Text Search"
|
|
||||||
"subtitle" "Built-in full text search with FlexSearch, no extra setup required."
|
|
||||||
"class" "aspect-auto md:aspect-[1.1/1] max-md:min-h-[340px]"
|
|
||||||
"image" "images/hextra-search.webp"
|
|
||||||
"imageClass" "top-[40%] left-[36px] w-[110%] sm:w-[110%] dark:opacity-80"
|
|
||||||
"style" "background: radial-gradient(ellipse at 50% 80%,rgba(221,210,59,0.15),hsla(0,0%,100%,0));"
|
|
||||||
)
|
|
||||||
}}
|
|
||||||
{{ template "hextra-feature-card" (dict
|
|
||||||
"title" "Lightweight as a Feather"
|
|
||||||
"subtitle" "No dependency or Node.js is needed to use Hextra. Powered by Hugo, one of *the fastest* static site generators, building your site in just seconds with a single binary."
|
|
||||||
)
|
|
||||||
}}
|
|
||||||
{{ template "hextra-feature-card" (dict
|
|
||||||
"title" "Reponsive with Dark Mode Included"
|
|
||||||
"subtitle" "Looks great on different screen sizes. Built-in dark mode support, with auto-switching based on user's system preference."
|
|
||||||
)
|
|
||||||
}}
|
|
||||||
{{ template "hextra-feature-card" (dict
|
|
||||||
"title" "Build and Host for Free"
|
|
||||||
"subtitle" "Build with GitHub Actions, and host for free on GitHub Pages. Alternatively it can be hosted on any static hosting service."
|
|
||||||
)
|
|
||||||
}}
|
|
||||||
{{ template "hextra-feature-card" (dict
|
|
||||||
"title" "Multi-Language Made Easy"
|
|
||||||
"subtitle" "Create multi-language pages by just adding locales suffix to the Markdown file. Adding i18n support to your site is intuitive."
|
|
||||||
)
|
|
||||||
}}
|
|
||||||
{{ template "hextra-feature-card" (dict
|
|
||||||
"title" "And Much More..."
|
|
||||||
"subtitle" "Syntax highlighting / Table of contents / SEO / RSS / LaTeX / Mermaid / Customizable / and more..."
|
|
||||||
)
|
|
||||||
}}
|
|
||||||
</div>
|
|
||||||
</section>
|
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
{{ end }}
|
{{ end }}
|
||||||
|
|
||||||
{{- define "hextra-feature-card" -}}
|
|
||||||
{{- $title := .title -}}
|
|
||||||
{{- $subtitle := .subtitle -}}
|
|
||||||
{{- $class := .class -}}
|
|
||||||
{{- $image := .image -}}
|
|
||||||
{{- $imageClass := .imageClass -}}
|
|
||||||
{{- $style := .style -}}
|
|
||||||
|
|
||||||
|
|
||||||
<div
|
|
||||||
{{ with $style }}style="{{ . | safeCSS }}"{{ end }}
|
|
||||||
class="{{ $class }} relative overflow-hidden rounded-3xl border border-gray-200 hover:border-gray-300 dark:border-neutral-800 dark:hover:border-neutral-700 before:pointer-events-none before:absolute before:inset-0 before:bg-glass-gradient"
|
|
||||||
>
|
|
||||||
<div class="relative w-full p-6">
|
|
||||||
<h3 class="text-2xl font-medium leading-6 mb-2">{{ $title }}</h3>
|
|
||||||
<p class="text-gray-500 text-sm leading-6">{{ $subtitle | markdownify }}</p>
|
|
||||||
</div>
|
|
||||||
{{- with $image -}}
|
|
||||||
<img src="{{ . }}" class="absolute max-w-none {{ $imageClass }}" alt="{{ $title }}" />
|
|
||||||
{{- end -}}
|
|
||||||
</div>
|
|
||||||
{{- end -}}
|
|
||||||
|
@ -1,5 +1,5 @@
|
|||||||
{{ define "main" }}
|
{{ define "main" }}
|
||||||
<div class="mx-auto flex max-w-[90rem]">
|
<div class='mx-auto flex {{ partial "utils/page-width" . }}'>
|
||||||
{{ partial "sidebar.html" (dict "context" . "disableSidebar" true "displayPlaceholder" true) }}
|
{{ partial "sidebar.html" (dict "context" . "disableSidebar" true "displayPlaceholder" true) }}
|
||||||
{{ partial "toc.html" . }}
|
{{ partial "toc.html" . }}
|
||||||
<article class="w-full break-words flex min-h-[calc(100vh-var(--navbar-height))] min-w-0 justify-center pb-8 pr-[calc(env(safe-area-inset-right)-1.5rem)]">
|
<article class="w-full break-words flex min-h-[calc(100vh-var(--navbar-height))] min-w-0 justify-center pb-8 pr-[calc(env(safe-area-inset-right)-1.5rem)]">
|
||||||
|
11
layouts/partials/components/comments.html
Normal file
11
layouts/partials/components/comments.html
Normal file
@ -0,0 +1,11 @@
|
|||||||
|
{{- $enableComments := site.Params.comments.enable | default false -}}
|
||||||
|
|
||||||
|
{{ if not (eq .Params.comments nil) }}
|
||||||
|
{{ $enableComments = .Params.comments }}
|
||||||
|
{{ end }}
|
||||||
|
|
||||||
|
{{- if $enableComments -}}
|
||||||
|
{{- if eq site.Params.comments.type "giscus" -}}
|
||||||
|
{{ partial "components/giscus.html" . }}
|
||||||
|
{{- end -}}
|
||||||
|
{{- end -}}
|
62
layouts/partials/components/giscus.html
Normal file
62
layouts/partials/components/giscus.html
Normal file
@ -0,0 +1,62 @@
|
|||||||
|
{{- $lang := site.Language.LanguageCode | default `en` -}}
|
||||||
|
|
||||||
|
{{- with site.Params.comments.giscus -}}
|
||||||
|
<script>
|
||||||
|
/*
|
||||||
|
* "preferred color scheme" theme in giscus works using "prefers-color-scheme" in media query.
|
||||||
|
* but, hugo's theme switch function works by using "color-theme" in local storage.
|
||||||
|
* This solution was created with reference to:
|
||||||
|
* https://github.com/giscus/giscus/issues/336#issuecomment-1214366281
|
||||||
|
*/
|
||||||
|
function getGiscusTheme() {
|
||||||
|
return localStorage.getItem("color-theme");
|
||||||
|
}
|
||||||
|
|
||||||
|
function setGiscusTheme() {
|
||||||
|
function sendMessage(message) {
|
||||||
|
const iframe = document.querySelector('iframe.giscus-frame');
|
||||||
|
if (!iframe) return;
|
||||||
|
iframe.contentWindow.postMessage({ giscus: message }, 'https://giscus.app');
|
||||||
|
}
|
||||||
|
sendMessage({
|
||||||
|
setConfig: {
|
||||||
|
theme: getGiscusTheme(),
|
||||||
|
},
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
document.addEventListener('DOMContentLoaded', function () {
|
||||||
|
const giscusAttributes = {
|
||||||
|
"src": "https://giscus.app/client.js",
|
||||||
|
"data-repo": "{{ .repo }}",
|
||||||
|
"data-repo-id": "{{ .repoId }}",
|
||||||
|
"data-category": "{{ .category }}",
|
||||||
|
"data-category-id": "{{ .categoryId }}",
|
||||||
|
"data-mapping": "{{ .mapping | default `pathname` }}",
|
||||||
|
"data-strict": "{{ (string .strict) | default 0 }}",
|
||||||
|
"data-reactions-enabled": "{{ (string .reactionsEnabled) | default 1 }}",
|
||||||
|
"data-emit-metadata": "{{ (string .emitMetadata) | default 0 }}",
|
||||||
|
"data-input-position": "{{ .inputPosition | default `top` }}",
|
||||||
|
"data-theme": getGiscusTheme(),
|
||||||
|
"data-lang": "{{ .lang | default $lang }}",
|
||||||
|
"crossorigin": "anonymous",
|
||||||
|
"async": "",
|
||||||
|
};
|
||||||
|
|
||||||
|
// Dynamically create script tag
|
||||||
|
const giscusScript = document.createElement("script");
|
||||||
|
Object.entries(giscusAttributes).forEach(([key, value]) => giscusScript.setAttribute(key, value));
|
||||||
|
document.getElementById('giscus').appendChild(giscusScript);
|
||||||
|
|
||||||
|
// Update giscus theme when theme switcher is clicked
|
||||||
|
const toggles = document.querySelectorAll(".theme-toggle");
|
||||||
|
if (toggles) {
|
||||||
|
toggles.forEach(toggle => toggle.addEventListener('click', setGiscusTheme));
|
||||||
|
}
|
||||||
|
});
|
||||||
|
</script>
|
||||||
|
|
||||||
|
<div id="giscus"></div>
|
||||||
|
{{- else -}}
|
||||||
|
{{ warnf "giscus is not configured" }}
|
||||||
|
{{- end -}}
|
@ -2,21 +2,29 @@
|
|||||||
|
|
||||||
{{- $copyright := (T "copyright") | default "© 2023 Hextra." -}}
|
{{- $copyright := (T "copyright") | default "© 2023 Hextra." -}}
|
||||||
|
|
||||||
|
{{- $footerWidth := "max-w-screen-xl" -}}
|
||||||
|
{{- with .Site.Params.footer.width -}}
|
||||||
|
{{ if eq . "wide" -}}
|
||||||
|
{{ $footerWidth = "max-w-[90rem]" -}}
|
||||||
|
{{ else if eq . "full" -}}
|
||||||
|
{{ $footerWidth = "max-w-full" -}}
|
||||||
|
{{ end -}}
|
||||||
|
{{- end -}}
|
||||||
|
|
||||||
<footer class="hextra-footer bg-gray-100 pb-[env(safe-area-inset-bottom)] dark:bg-neutral-900 print:bg-transparent">
|
<footer class="hextra-footer bg-gray-100 pb-[env(safe-area-inset-bottom)] dark:bg-neutral-900 print:bg-transparent">
|
||||||
{{- if $enableFooterSwitches }}
|
{{- if $enableFooterSwitches }}
|
||||||
<div class="mx-auto flex max-w-screen-xl gap-2 py-2 px-4">
|
<div class='mx-auto flex gap-2 py-2 px-4 {{ $footerWidth }}'>
|
||||||
{{- partial "language-switch.html" (dict "context" .) -}}
|
{{- partial "language-switch.html" (dict "context" .) -}}
|
||||||
{{- partial "theme-toggle.html" -}}
|
{{- partial "theme-toggle.html" -}}
|
||||||
</div>
|
</div>
|
||||||
{{ end -}}
|
{{ end -}}
|
||||||
<hr class="dark:border-neutral-800" />
|
<hr class="dark:border-neutral-800" />
|
||||||
<div
|
<div
|
||||||
class="mx-auto flex max-w-screen-xl justify-center py-12 pl-[max(env(safe-area-inset-left),1.5rem)] pr-[max(env(safe-area-inset-right),1.5rem)] text-gray-600 dark:text-gray-400 md:justify-start"
|
class='{{ $footerWidth }} mx-auto flex justify-center py-12 pl-[max(env(safe-area-inset-left),1.5rem)] pr-[max(env(safe-area-inset-right),1.5rem)] text-gray-600 dark:text-gray-400 md:justify-start'
|
||||||
>
|
>
|
||||||
<div class="flex w-full flex-col items-center sm:items-start">
|
<div class="flex w-full flex-col items-center sm:items-start">
|
||||||
{{- if (.Site.Params.footer.displayPoweredBy | default true) }}<div class="font-semibold">{{ template "theme-credit" . }}</div>{{ end }}
|
{{- if (.Site.Params.footer.displayPoweredBy | default true) }}<div class="font-semibold">{{ template "theme-credit" . }}</div>{{ end }}
|
||||||
{{- if .Site.Params.footer.displayCopyright }}<p class="mt-6 text-xs">{{ $copyright | markdownify }}</p>{{ end }}
|
{{- if .Site.Params.footer.displayCopyright }}<div class="mt-6 text-xs">{{ $copyright | markdownify }}</div>{{ end }}
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
</footer>
|
</footer>
|
||||||
|
13
layouts/partials/google-analytics.html
Normal file
13
layouts/partials/google-analytics.html
Normal file
@ -0,0 +1,13 @@
|
|||||||
|
{{ with .Site.GoogleAnalytics }}
|
||||||
|
<!-- Global site tag (gtag.js) - Google Analytics -->
|
||||||
|
<script async src="https://www.googletagmanager.com/gtag/js?id={{ . }}"></script>
|
||||||
|
<script>
|
||||||
|
window.dataLayer = window.dataLayer || [];
|
||||||
|
function gtag() {
|
||||||
|
dataLayer.push(arguments);
|
||||||
|
}
|
||||||
|
gtag("js", new Date());
|
||||||
|
|
||||||
|
gtag("config", "{{ . }}");
|
||||||
|
</script>
|
||||||
|
{{ end }}
|
@ -7,7 +7,7 @@
|
|||||||
|
|
||||||
{{- if hugo.IsProduction }}
|
{{- if hugo.IsProduction }}
|
||||||
{{- $styles = $styles | minify | fingerprint }}
|
{{- $styles = $styles | minify | fingerprint }}
|
||||||
<link rel="preload" href="{{ $styles.RelPermalink }}" as="style" />
|
<link rel="preload" href="{{ $styles.RelPermalink }}" as="style" integrity="{{ $styles.Data.Integrity }}" />
|
||||||
<link href="{{ $styles.RelPermalink }}" rel="stylesheet" integrity="{{ $styles.Data.Integrity }}" />
|
<link href="{{ $styles.RelPermalink }}" rel="stylesheet" integrity="{{ $styles.Data.Integrity }}" />
|
||||||
{{- else }}
|
{{- else }}
|
||||||
<link href="{{ $styles.RelPermalink }}" rel="stylesheet" />
|
<link href="{{ $styles.RelPermalink }}" rel="stylesheet" />
|
||||||
|
@ -16,10 +16,21 @@
|
|||||||
{{ end -}}
|
{{ end -}}
|
||||||
</title>
|
</title>
|
||||||
<meta name="description" content="{{ partial "utils/page-description.html" . }}" />
|
<meta name="description" content="{{ partial "utils/page-description.html" . }}" />
|
||||||
|
|
||||||
{{ partial "opengraph.html" . }}
|
{{ partial "opengraph.html" . }}
|
||||||
{{ template "_internal/schema.html" . -}}
|
{{ template "_internal/schema.html" . -}}
|
||||||
{{ template "_internal/twitter_cards.html" . -}}
|
{{ template "_internal/twitter_cards.html" . -}}
|
||||||
|
|
||||||
{{ partialCached "head-css.html" . }}
|
{{ partialCached "head-css.html" . }}
|
||||||
|
|
||||||
|
|
||||||
|
<!-- Google Analytics -->
|
||||||
|
{{- if and .Site.GoogleAnalytics (eq hugo.Environment "production") }}
|
||||||
|
<link rel="preconnect" href="https://www.googletagmanager.com" crossorigin />
|
||||||
|
{{ partial "google-analytics.html" . }}
|
||||||
|
{{- end }}
|
||||||
|
|
||||||
|
|
||||||
<script>
|
<script>
|
||||||
/* Initialize light/dark mode */
|
/* Initialize light/dark mode */
|
||||||
if (localStorage.getItem("color-theme") === "dark" || (!("color-theme" in localStorage) && window.matchMedia("(prefers-color-scheme: dark)").matches)) {
|
if (localStorage.getItem("color-theme") === "dark" || (!("color-theme" in localStorage) && window.matchMedia("(prefers-color-scheme: dark)").matches)) {
|
||||||
|
@ -4,11 +4,19 @@
|
|||||||
{{- $logoHeight := .Site.Params.navbar.logo.height | default "20" -}}
|
{{- $logoHeight := .Site.Params.navbar.logo.height | default "20" -}}
|
||||||
{{- $logoDarkPath := .Site.Params.navbar.logo.dark | default $logoPath -}}
|
{{- $logoDarkPath := .Site.Params.navbar.logo.dark | default $logoPath -}}
|
||||||
|
|
||||||
|
{{- $navWidth := "max-w-[90rem]" -}}
|
||||||
|
{{- with .Site.Params.navbar.width -}}
|
||||||
|
{{ if eq . "normal" -}}
|
||||||
|
{{ $navWidth = "max-w-screen-xl" -}}
|
||||||
|
{{ else if eq . "full" -}}
|
||||||
|
{{ $navWidth = "max-w-full" -}}
|
||||||
|
{{ end -}}
|
||||||
|
{{- end -}}
|
||||||
|
|
||||||
<div class="nav-container sticky top-0 z-20 w-full bg-transparent print:hidden">
|
<div class="nav-container sticky top-0 z-20 w-full bg-transparent print:hidden">
|
||||||
<div class="nav-container-blur pointer-events-none absolute z-[-1] h-full w-full bg-white dark:bg-dark shadow-[0_2px_4px_rgba(0,0,0,.02),0_1px_0_rgba(0,0,0,.06)] contrast-more:shadow-[0_0_0_1px_#000] dark:shadow-[0_-1px_0_rgba(255,255,255,.1)_inset] contrast-more:dark:shadow-[0_0_0_1px_#fff]"></div>
|
<div class="nav-container-blur pointer-events-none absolute z-[-1] h-full w-full bg-white dark:bg-dark shadow-[0_2px_4px_rgba(0,0,0,.02),0_1px_0_rgba(0,0,0,.06)] contrast-more:shadow-[0_0_0_1px_#000] dark:shadow-[0_-1px_0_rgba(255,255,255,.1)_inset] contrast-more:dark:shadow-[0_0_0_1px_#fff]"></div>
|
||||||
|
|
||||||
<nav class="mx-auto flex items-center justify-end gap-2 h-16 px-6 max-w-[90rem]">
|
<nav class="mx-auto flex items-center justify-end gap-2 h-16 px-6 {{ $navWidth }}">
|
||||||
<a class="flex items-center hover:opacity-75 ltr:mr-auto rtl:ml-auto" href="{{ $logoLink }}">
|
<a class="flex items-center hover:opacity-75 ltr:mr-auto rtl:ml-auto" href="{{ $logoLink }}">
|
||||||
{{- if (.Site.Params.navbar.displayLogo | default true) }}
|
{{- if (.Site.Params.navbar.displayLogo | default true) }}
|
||||||
<img class="block dark:hidden" src="{{ $logoPath | relURL }}" alt="{{ .Site.Title }}" height="{{ $logoHeight }}" width="{{ $logoWidth }}" />
|
<img class="block dark:hidden" src="{{ $logoPath | relURL }}" alt="{{ .Site.Title }}" height="{{ $logoHeight }}" width="{{ $logoWidth }}" />
|
||||||
|
@ -13,7 +13,7 @@
|
|||||||
<kbd
|
<kbd
|
||||||
class="absolute my-1.5 select-none ltr:right-1.5 rtl:left-1.5 h-5 rounded bg-white px-1.5 font-mono text-[10px] font-medium text-gray-500 border dark:border-gray-100/20 dark:bg-dark/50 contrast-more:border-current contrast-more:text-current contrast-more:dark:border-current items-center gap-1 transition-opacity pointer-events-none hidden sm:flex"
|
class="absolute my-1.5 select-none ltr:right-1.5 rtl:left-1.5 h-5 rounded bg-white px-1.5 font-mono text-[10px] font-medium text-gray-500 border dark:border-gray-100/20 dark:bg-dark/50 contrast-more:border-current contrast-more:text-current contrast-more:dark:border-current items-center gap-1 transition-opacity pointer-events-none hidden sm:flex"
|
||||||
>
|
>
|
||||||
<span class="text-xs">⌘</span>K
|
CTRL K
|
||||||
</kbd>
|
</kbd>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
|
@ -1,11 +1,11 @@
|
|||||||
{{ with .Description | plainify -}}
|
{{ with .Description | plainify | htmlUnescape -}}
|
||||||
{{ . -}}
|
{{ . -}}
|
||||||
{{ else -}}
|
{{ else -}}
|
||||||
{{ if .IsHome -}}
|
{{ if .IsHome -}}
|
||||||
{{ with .Site.Params.description | plainify -}}
|
{{ with .Site.Params.description | plainify | htmlUnescape -}}
|
||||||
{{ . -}}
|
{{ . -}}
|
||||||
{{ end -}}
|
{{ end -}}
|
||||||
{{ else -}}
|
{{ else -}}
|
||||||
{{ .Summary | plainify | chomp -}}
|
{{ .Summary | plainify | htmlUnescape | chomp -}}
|
||||||
{{ end -}}
|
{{ end -}}
|
||||||
{{ end -}}
|
{{ end -}}
|
||||||
|
27
layouts/partials/utils/page-width.html
Normal file
27
layouts/partials/utils/page-width.html
Normal file
@ -0,0 +1,27 @@
|
|||||||
|
{{/* Get page width from site configuration */}}
|
||||||
|
|
||||||
|
{{/* Default page width */}}
|
||||||
|
{{- $pageWidth := "" -}}
|
||||||
|
|
||||||
|
{{/* Get page width setting from page front matter or site params */}}
|
||||||
|
{{ with .Params.width -}}
|
||||||
|
{{ $pageWidth = . -}}
|
||||||
|
{{ else -}}
|
||||||
|
{{ with .Site.Params.page.width -}}
|
||||||
|
{{ $pageWidth = . -}}
|
||||||
|
{{ end -}}
|
||||||
|
{{ end -}}
|
||||||
|
|
||||||
|
{{- with $pageWidth -}}
|
||||||
|
{{ if eq . "wide" -}}
|
||||||
|
{{ $pageWidth = "max-w-[90rem]" -}}
|
||||||
|
{{ else if eq . "full" -}}
|
||||||
|
{{ $pageWidth = "max-w-full" -}}
|
||||||
|
{{ else -}}
|
||||||
|
{{ $pageWidth = "max-w-screen-xl" -}}
|
||||||
|
{{ end -}}
|
||||||
|
{{ else -}}
|
||||||
|
{{ $pageWidth = "max-w-screen-xl" -}}
|
||||||
|
{{ end -}}
|
||||||
|
|
||||||
|
{{ return $pageWidth }}
|
@ -1,40 +0,0 @@
|
|||||||
{{- $pctx := . -}}
|
|
||||||
{{- if .IsHome -}}{{ $pctx = .Site }}{{- end -}}
|
|
||||||
{{- $pages := slice -}}
|
|
||||||
{{- if or $.IsHome $.IsSection -}}
|
|
||||||
{{- $pages = $pctx.RegularPages -}}
|
|
||||||
{{- else -}}
|
|
||||||
{{- $pages = $pctx.Pages -}}
|
|
||||||
{{- end -}}
|
|
||||||
{{- $limit := .Site.Config.Services.RSS.Limit -}}
|
|
||||||
{{- if ge $limit 1 -}}
|
|
||||||
{{- $pages = $pages | first $limit -}}
|
|
||||||
{{- end -}}
|
|
||||||
{{- printf "<?xml version=\"1.0\" encoding=\"utf-8\" standalone=\"yes\"?>" | safeHTML }}
|
|
||||||
<rss version="2.0" xmlns:atom="http://www.w3.org/2005/Atom">
|
|
||||||
<channel>
|
|
||||||
<title>{{ if eq .Title .Site.Title }}{{ .Site.Title }}{{ else }}{{ with .Title }}{{.}} of {{ end }}{{ .Site.Title }}{{ end }}</title>
|
|
||||||
<link>{{ .Permalink }}</link>
|
|
||||||
<description>Recent content {{ if ne .Title .Site.Title }}{{ with .Title }}in {{.}} {{ end }}{{ end }}on {{ .Site.Title }}</description>
|
|
||||||
<generator>Hugo -- gohugo.io</generator>
|
|
||||||
<language>{{ site.Language.LanguageCode }}</language>{{ with .Site.Author.email }}
|
|
||||||
<managingEditor>{{.}}{{ with $.Site.Author.name }} ({{.}}){{end}}</managingEditor>{{end}}{{ with .Site.Author.email }}
|
|
||||||
<webMaster>{{.}}{{ with $.Site.Author.name }} ({{.}}){{end}}</webMaster>{{end}}{{ with .Site.Copyright }}
|
|
||||||
<copyright>{{.}}</copyright>{{end}}{{ if not .Date.IsZero }}
|
|
||||||
<lastBuildDate>{{ .Date.Format "Mon, 02 Jan 2006 15:04:05 -0700" | safeHTML }}</lastBuildDate>{{ end }}
|
|
||||||
{{- with .OutputFormats.Get "RSS" -}}
|
|
||||||
{{ printf "<atom:link href=%q rel=\"self\" type=%q />" .Permalink .MediaType | safeHTML }}
|
|
||||||
{{- end -}}
|
|
||||||
{{ range $pages }}
|
|
||||||
<item>
|
|
||||||
<title>{{ .Title }}</title>
|
|
||||||
<link>{{ .Permalink }}</link>
|
|
||||||
<pubDate>{{ .Date.Format "Mon, 02 Jan 2006 15:04:05 -0700" | safeHTML }}</pubDate>
|
|
||||||
{{ with .Site.Author.email }}<author>{{.}}{{ with $.Site.Author.name }} ({{.}}){{end}}</author>{{end}}
|
|
||||||
<guid>{{ .Permalink }}</guid>
|
|
||||||
<description>{{ with .Description | html }}{{ . }}{{ else }}{{ .Summary | html }}{{ end -}}</description>
|
|
||||||
<content:encoded>{{ (printf "<![CDATA[%s]]>" .Content) | safeHTML }}</content:encoded>
|
|
||||||
</item>
|
|
||||||
{{ end }}
|
|
||||||
</channel>
|
|
||||||
</rss>
|
|
@ -1,26 +1,39 @@
|
|||||||
|
{{- $context := . -}}
|
||||||
{{- $link := .Get "link" -}}
|
{{- $link := .Get "link" -}}
|
||||||
{{- $title := .Get "title" -}}
|
{{- $title := .Get "title" -}}
|
||||||
{{- $icon := .Get "icon" -}}
|
{{- $icon := .Get "icon" -}}
|
||||||
{{- $subtitle := .Get "subtitle" -}}
|
{{- $subtitle := .Get "subtitle" -}}
|
||||||
{{- $image := .Get "image" -}}
|
{{- $image := .Get "image" -}}
|
||||||
|
{{- $width := 0 -}}
|
||||||
|
{{- $height := 0 -}}
|
||||||
|
{{- $imageStyle := .Get "imageStyle" -}}
|
||||||
|
|
||||||
|
{{/* Image processing options */}}
|
||||||
{{- $method := .Get "method" | default "Resize" | humanize -}}
|
{{- $method := .Get "method" | default "Resize" | humanize -}}
|
||||||
{{- $options := .Get "options" | default "800x webp q80" -}}
|
{{- $options := .Get "options" | default "800x webp q80" -}}
|
||||||
{{- $context := . -}}
|
|
||||||
|
|
||||||
|
{{- if and $image (not (urls.Parse $image).Scheme) -}}
|
||||||
{{/*- Adding asset support for images here, so that Hugo can do its image processing magic. -*/}}
|
{{/* Process images in assets */}}
|
||||||
{{/* Unfortunately we cannot pass .Resize/.Fit/.Fill as variables, so we're left with chaining IFs */}}
|
|
||||||
|
|
||||||
{{- if not (urls.Parse $image).Scheme -}}
|
|
||||||
{{- with resources.Get $image -}}
|
{{- with resources.Get $image -}}
|
||||||
|
{{- $processed := "" -}}
|
||||||
{{- if eq $method "Resize" -}}
|
{{- if eq $method "Resize" -}}
|
||||||
{{- $image = (.Resize $options).RelPermalink -}}
|
{{- $processed = (.Resize $options) -}}
|
||||||
{{- else if eq $method "Fit" -}}
|
{{- else if eq $method "Fit" -}}
|
||||||
{{- $image = (.Fit $options).RelPermalink -}}
|
{{- $processed = (.Fit $options) -}}
|
||||||
{{- else if eq $method "Fill" -}}
|
{{- else if eq $method "Fill" -}}
|
||||||
{{- $image = (.Fill $options).RelPermalink -}}
|
{{- $processed = (.Fill $options) -}}
|
||||||
{{- else if eq $method "Crop" -}}
|
{{- else if eq $method "Crop" -}}
|
||||||
{{- $image = (.Crop $options).RelPermalink -}}
|
{{- $processed = (.Crop $options) -}}
|
||||||
|
{{- else -}}
|
||||||
|
{{- errorf "Invalid image processing command: Must be one of Crop, Fit, Fill or Resize." -}}
|
||||||
|
{{- end -}}
|
||||||
|
{{- $width = $processed.Width -}}
|
||||||
|
{{- $height = $processed.Height -}}
|
||||||
|
{{- $image = $processed.RelPermalink -}}
|
||||||
|
{{- else -}}
|
||||||
|
{{/* Otherwise, use relative link of the image */}}
|
||||||
|
{{- if hasPrefix $image "/" -}}
|
||||||
|
{{- $image = relURL (strings.TrimPrefix "/" $image) -}}
|
||||||
{{- end -}}
|
{{- end -}}
|
||||||
{{- end -}}
|
{{- end -}}
|
||||||
{{- end -}}
|
{{- end -}}
|
||||||
@ -33,6 +46,7 @@
|
|||||||
{{- $external := strings.HasPrefix $link "http" -}}
|
{{- $external := strings.HasPrefix $link "http" -}}
|
||||||
{{- $href := cond (strings.HasPrefix $link "/") ($link | relURL) $link -}}
|
{{- $href := cond (strings.HasPrefix $link "/") ($link | relURL) $link -}}
|
||||||
|
|
||||||
|
|
||||||
<a
|
<a
|
||||||
class="hextra-card group flex flex-col justify-start overflow-hidden rounded-lg border border-gray-200 text-current no-underline dark:shadow-none hover:shadow-gray-100 dark:hover:shadow-none shadow-gray-100 active:shadow-sm active:shadow-gray-200 transition-all duration-200 {{ $linkClass }}"
|
class="hextra-card group flex flex-col justify-start overflow-hidden rounded-lg border border-gray-200 text-current no-underline dark:shadow-none hover:shadow-gray-100 dark:hover:shadow-none shadow-gray-100 active:shadow-sm active:shadow-gray-200 transition-all duration-200 {{ $linkClass }}"
|
||||||
href="{{ $href }}"
|
href="{{ $href }}"
|
||||||
@ -41,7 +55,15 @@
|
|||||||
{{- end -}}
|
{{- end -}}
|
||||||
>
|
>
|
||||||
{{- with $image -}}
|
{{- with $image -}}
|
||||||
<img alt="{{ $title }}" loading="lazy" decoding="async" style="color: transparent;" src="{{ $image | safeURL }}" />
|
<img
|
||||||
|
alt="{{ $title }}"
|
||||||
|
loading="lazy"
|
||||||
|
decoding="async"
|
||||||
|
src="{{ $image | safeURL }}"
|
||||||
|
{{ with $width }}width="{{ . }}"{{ end }}
|
||||||
|
{{ with $height }}height="{{ . }}"{{ end }}
|
||||||
|
{{ with $imageStyle }}style="{{ . | safeCSS }}"{{ end }}
|
||||||
|
/>
|
||||||
{{- end -}}
|
{{- end -}}
|
||||||
|
|
||||||
{{- $padding := "p-4" -}}
|
{{- $padding := "p-4" -}}
|
||||||
|
28
layouts/shortcodes/hextra/feature-card.html
Normal file
28
layouts/shortcodes/hextra/feature-card.html
Normal file
@ -0,0 +1,28 @@
|
|||||||
|
{{- $title := .Get "title" -}}
|
||||||
|
{{- $subtitle := .Get "subtitle" -}}
|
||||||
|
{{- $class := .Get "class" -}}
|
||||||
|
{{- $image := .Get "image" -}}
|
||||||
|
{{- $imageClass := .Get "imageClass" -}}
|
||||||
|
{{- $style := .Get "style" -}}
|
||||||
|
{{- $icon := .Get "icon" -}}
|
||||||
|
|
||||||
|
|
||||||
|
<div
|
||||||
|
{{ with $style }}style="{{ . | safeCSS }}"{{ end }}
|
||||||
|
class="{{ $class }} hextra-feature-card relative overflow-hidden rounded-3xl border border-gray-200 hover:border-gray-300 dark:border-neutral-800 dark:hover:border-neutral-700 before:pointer-events-none before:absolute before:inset-0 before:bg-glass-gradient"
|
||||||
|
>
|
||||||
|
<div class="relative w-full p-6">
|
||||||
|
<h3 class="text-2xl font-medium leading-6 mb-2 flex items-center">
|
||||||
|
{{ with $icon -}}
|
||||||
|
<span class="pr-2">
|
||||||
|
{{- partial "utils/icon.html" (dict "name" . "attributes" "height=1.5rem") -}}
|
||||||
|
</span>
|
||||||
|
{{ end -}}
|
||||||
|
<span>{{ $title }}</span>
|
||||||
|
</h3>
|
||||||
|
<p class="text-gray-500 dark:text-gray-400 text-sm leading-6">{{ $subtitle | markdownify }}</p>
|
||||||
|
</div>
|
||||||
|
{{- with $image -}}
|
||||||
|
<img src="{{ . }}" class="absolute max-w-none {{ $imageClass }}" alt="{{ $title }}" />
|
||||||
|
{{- end -}}
|
||||||
|
</div>
|
9
layouts/shortcodes/hextra/feature-grid.html
Normal file
9
layouts/shortcodes/hextra/feature-grid.html
Normal file
@ -0,0 +1,9 @@
|
|||||||
|
{{- $style := .Get "style" -}}
|
||||||
|
|
||||||
|
|
||||||
|
<div
|
||||||
|
class="grid sm:grid-cols-2 lg:grid-cols-3 grid-cols-1 gap-4 w-full"
|
||||||
|
{{ with $style }}style="{{ . | safeCSS }}"{{ end }}
|
||||||
|
>
|
||||||
|
{{ .Inner }}
|
||||||
|
</div>
|
15
layouts/shortcodes/hextra/hero-badge.html
Normal file
15
layouts/shortcodes/hextra/hero-badge.html
Normal file
@ -0,0 +1,15 @@
|
|||||||
|
{{- $link := .Get "link" -}}
|
||||||
|
{{- $external := hasPrefix $link "http" -}}
|
||||||
|
{{- $href := cond (hasPrefix $link "/") ($link | relURL) $link -}}
|
||||||
|
{{- $class := .Get "class" }}
|
||||||
|
{{- $style := .Get "style" -}}
|
||||||
|
|
||||||
|
|
||||||
|
<a
|
||||||
|
href="{{ $href }}"
|
||||||
|
class="{{ $class }} inline-flex items-center rounded-full gap-2 px-3 py-1 text-xs text-gray-600 dark:text-gray-400 bg-gray-100 dark:bg-neutral-800 dark:border-neutral-800 border hover:border-gray-400 dark:hover:text-gray-50 dark:hover:border-gray-600 transition-all ease-in duration-200"
|
||||||
|
{{ with $style }}style="{{ . | safeCSS }}"{{ end }}
|
||||||
|
{{ if $external }}target="_blank" rel="noreferrer"{{ end -}}
|
||||||
|
>
|
||||||
|
{{ .Inner | markdownify }}
|
||||||
|
</a>
|
15
layouts/shortcodes/hextra/hero-button.html
Normal file
15
layouts/shortcodes/hextra/hero-button.html
Normal file
@ -0,0 +1,15 @@
|
|||||||
|
{{- $link := .Get "link" -}}
|
||||||
|
{{- $text := .Get "text" -}}
|
||||||
|
{{- $style := .Get "style" -}}
|
||||||
|
|
||||||
|
{{- $external := hasPrefix $link "http" -}}
|
||||||
|
{{- $href := cond (hasPrefix $link "/") ($link | relURL) $link -}}
|
||||||
|
|
||||||
|
<a
|
||||||
|
href="{{ $href }}"
|
||||||
|
class="font-medium cursor-pointer px-6 py-3 rounded-full text-center text-white inline-block bg-primary-600 hover:bg-primary-700 focus:outline-none focus:ring-4 focus:ring-primary-300 dark:bg-primary-600 dark:hover:bg-primary-700 dark:focus:ring-primary-800 transition-all ease-in duration-200"
|
||||||
|
{{ with $style }}style="{{ . | safeCSS }}"{{ end }}
|
||||||
|
{{ if $external }}target="_blank" rel="noreferrer"{{ end -}}
|
||||||
|
>
|
||||||
|
{{- $text -}}
|
||||||
|
</a>
|
9
layouts/shortcodes/hextra/hero-headline.html
Normal file
9
layouts/shortcodes/hextra/hero-headline.html
Normal file
@ -0,0 +1,9 @@
|
|||||||
|
{{- $style := .Get "style" -}}
|
||||||
|
|
||||||
|
|
||||||
|
<h1
|
||||||
|
class="text-4xl font-bold leading-none tracking-tighter md:text-5xl py-2 bg-clip-text text-transparent bg-gradient-to-r from-gray-900 to-gray-600 dark:from-gray-100 dark:to-gray-400"
|
||||||
|
{{ with $style }}style="{{ . | safeCSS }}"{{ end }}
|
||||||
|
>
|
||||||
|
{{ .Inner | markdownify }}
|
||||||
|
</h1>
|
9
layouts/shortcodes/hextra/hero-subtitle.html
Normal file
9
layouts/shortcodes/hextra/hero-subtitle.html
Normal file
@ -0,0 +1,9 @@
|
|||||||
|
{{- $style := .Get "style" -}}
|
||||||
|
|
||||||
|
|
||||||
|
<p
|
||||||
|
class="text-xl text-gray-600 dark:text-gray-400 sm:text-xl"
|
||||||
|
{{ with $style }}style="{{ . | safeCSS }}"{{ end }}
|
||||||
|
>
|
||||||
|
{{ .Inner | markdownify }}
|
||||||
|
</p>
|
@ -1,9 +1,9 @@
|
|||||||
{{- $name := .Get "name" | default (.Get 0) -}}
|
{{- $name := .Get "name" | default (.Get 0) -}}
|
||||||
{{- $icon := index site.Data.icons (.Get 0) -}}
|
{{- $icon := index site.Data.icons $name -}}
|
||||||
{{- $attributes := "height=1em"}}
|
{{- $attributes := .Get "attributes" | default "height=1em"}}
|
||||||
|
|
||||||
{{- if not $icon -}}
|
{{- if not $icon -}}
|
||||||
{{ errorf "icon %q not found" (.Get 0) }}
|
{{ errorf "icon %q not found" $name }}
|
||||||
{{- end -}}
|
{{- end -}}
|
||||||
|
|
||||||
{{- $icon = replaceRE "<svg" (printf "<svg %s" $attributes) $icon -}}
|
{{- $icon = replaceRE "<svg" (printf "<svg %s" $attributes) $icon -}}
|
||||||
|
@ -14,6 +14,11 @@ module.exports = {
|
|||||||
content: [
|
content: [
|
||||||
'./**/hugo_stats.json',
|
'./**/hugo_stats.json',
|
||||||
],
|
],
|
||||||
|
safelist: [
|
||||||
|
'max-w-screen-xl',
|
||||||
|
'max-w-[90rem]',
|
||||||
|
'max-w-full'
|
||||||
|
],
|
||||||
theme: {
|
theme: {
|
||||||
screens: {
|
screens: {
|
||||||
sm: '640px',
|
sm: '640px',
|
||||||
|
Reference in New Issue
Block a user