Skip to content

Commit 4ea94c4

Browse files
jmooringbep
authored andcommitted
tpl/images: Add images.QR function
Closes #13205
1 parent 5d2cbee commit 4ea94c4

File tree

10 files changed

+495
-28
lines changed

10 files changed

+495
-28
lines changed

‎README.md

+17-14
Original file line numberDiff line numberDiff line change
@@ -101,6 +101,7 @@ Build the extended edition:
101101
```text
102102
CGO_ENABLED=1 go install -tags extended github.com/gohugoio/hugo@latest
103103
```
104+
104105
## Star History
105106

106107
[![Star History Chart](https://api.star-history.com/svg?repos=gohugoio/hugo&type=Timeline)](https://star-history.com/#gohugoio/hugo&Timeline)
@@ -154,7 +155,7 @@ github.com/bep/clocks="v0.5.0"
154155
github.com/bep/debounce="v1.2.0"
155156
github.com/bep/gitmap="v1.6.0"
156157
github.com/bep/goat="v0.5.0"
157-
github.com/bep/godartsass/v2="v2.3.0"
158+
github.com/bep/godartsass/v2="v2.3.2"
158159
github.com/bep/golibsass="v1.2.0"
159160
github.com/bep/gowebp="v0.3.0"
160161
github.com/bep/imagemeta="v0.8.3"
@@ -189,7 +190,7 @@ github.com/gohugoio/locales="v0.14.0"
189190
github.com/gohugoio/localescompressed="v1.0.1"
190191
github.com/google/go-cmp="v0.6.0"
191192
github.com/gorilla/websocket="v1.5.3"
192-
github.com/hairyhenderson/go-codeowners="v0.6.1"
193+
github.com/hairyhenderson/go-codeowners="v0.7.0"
193194
github.com/hashicorp/golang-lru/v2="v2.0.7"
194195
github.com/invopop/yaml="v0.2.0"
195196
github.com/jdkato/prose="v1.2.1"
@@ -218,30 +219,32 @@ github.com/russross/blackfriday/v2="v2.1.0"
218219
github.com/sass/dart-sass/compiler="1.81.0"
219220
github.com/sass/dart-sass/implementation="1.81.0"
220221
github.com/sass/dart-sass/protocol="3.1.0"
222+
github.com/sass/libsass="3.6.6"
221223
github.com/spf13/afero="v1.11.0"
222-
github.com/spf13/cast="v1.7.0"
224+
github.com/spf13/cast="v1.7.1"
223225
github.com/spf13/cobra="v1.8.1"
224226
github.com/spf13/fsync="v0.10.1"
225227
github.com/spf13/pflag="v1.0.5"
226-
github.com/tdewolff/minify/v2="v2.21.1"
227-
github.com/tdewolff/parse/v2="v2.7.18"
228-
github.com/tetratelabs/wazero="v1.8.1"
228+
github.com/tdewolff/minify/v2="v2.20.37"
229+
github.com/tdewolff/parse/v2="v2.7.15"
230+
github.com/tetratelabs/wazero="v1.8.2"
231+
github.com/webmproject/libwebp="v1.3.2"
229232
github.com/yuin/goldmark-emoji="v1.0.4"
230233
github.com/yuin/goldmark="v1.7.8"
231234
go.uber.org/automaxprocs="v1.5.3"
232-
golang.org/x/crypto="v0.29.0"
235+
golang.org/x/crypto="v0.31.0"
233236
golang.org/x/exp="v0.0.0-20221031165847-c99f073a8326"
234237
golang.org/x/image="v0.22.0"
235238
golang.org/x/mod="v0.22.0"
236-
golang.org/x/net="v0.31.0"
237-
golang.org/x/sync="v0.9.0"
238-
golang.org/x/sys="v0.27.0"
239-
golang.org/x/text="v0.20.0"
240-
golang.org/x/tools="v0.27.0"
241-
google.golang.org/protobuf="v1.35.1"
239+
golang.org/x/net="v0.33.0"
240+
golang.org/x/sync="v0.10.0"
241+
golang.org/x/sys="v0.28.0"
242+
golang.org/x/text="v0.21.0"
243+
golang.org/x/tools="v0.28.0"
244+
google.golang.org/protobuf="v1.35.2"
242245
gopkg.in/yaml.v2="v2.4.0"
243246
gopkg.in/yaml.v3="v3.0.1"
244-
howett.net/plist="v1.0.0"
247+
rsc.io/qr="v0.2.0"
245248
software.sslmate.com/src/go-pkcs12="v0.2.0"
246249
```
247250
</details>

‎docs/content/en/content-management/shortcodes.md

+101-1
Original file line numberDiff line numberDiff line change
@@ -121,7 +121,7 @@ Hugo renders this to:
121121
</details>
122122
```
123123

124-
The details shortcode accepts these named arguments:
124+
The `details` shortcode accepts these named arguments:
125125

126126
summary
127127
: (`string`) The content of the child `summary` element rendered from Markdown to HTML. Default is `Details`.
@@ -333,6 +333,106 @@ Access nested values by [chaining] the [identifiers]:
333333
{{</* param my.nested.param */>}}
334334
```
335335

336+
### qr
337+
338+
{{% note %}}
339+
To override Hugo's embedded `qr` shortcode, copy the [source code] to a file with the same name in the layouts/shortcodes directory.
340+
341+
[source code]: {{% eturl qr %}}
342+
{{% /note %}}
343+
344+
The `qr` shortcode encodes the given text into a [QR code] using the specified options and renders the resulting image.
345+
346+
[QR code]: https://en.wikipedia.org/wiki/QR_code
347+
348+
Use the self-closing syntax to pass the text as an argument:
349+
350+
```text
351+
{{</* qr text="https://gohugo.io" /*/>}}
352+
```
353+
354+
Or insert the text between the opening and closing tags:
355+
356+
```text
357+
{{</* qr */>}}
358+
https://gohugo.io
359+
{{</* /qr */>}}
360+
```
361+
362+
Both of the above produce this image:
363+
364+
{{< qr text="https://gohugo.io" class="qrcode" />}}
365+
366+
To create a QR code for a phone number:
367+
368+
```text
369+
{{</* qr text="tel:+12065550101" /*/>}}
370+
```
371+
372+
{{< qr text="tel:+12065550101" class="qrcode" />}}
373+
374+
To create a QR code containing contact information in the [vCard] format:
375+
376+
[vCard]: https://en.wikipedia.org/wiki/VCard
377+
378+
```text
379+
{{</* qr level="low" scale=2 alt="QR code of vCard for John Smith" */>}}
380+
BEGIN:VCARD
381+
VERSION:2.1
382+
N;CHARSET=UTF-8:Smith;John;R.;Dr.;PhD
383+
FN;CHARSET=UTF-8:Dr. John R. Smith, PhD.
384+
ORG;CHARSET=UTF-8:ABC Widgets
385+
TITLE;CHARSET=UTF-8:Vice President Engineering
386+
TEL;TYPE=WORK:+12065550101
387+
EMAIL;TYPE=WORK:jsmith@example.org
388+
END:VCARD
389+
{{</* /qr */>}}
390+
```
391+
392+
{{< qr level="low" scale=2 alt="QR code of vCard for John Smith" class="qrcode" >}}
393+
BEGIN:VCARD
394+
VERSION:2.1
395+
N;CHARSET=UTF-8:Smith;John;R.;Dr.;PhD
396+
FN;CHARSET=UTF-8:Dr. John R. Smith, PhD.
397+
ORG;CHARSET=UTF-8:ABC Widgets
398+
TITLE;CHARSET=UTF-8:Vice President Engineering
399+
TEL;TYPE=WORK:+12065550101
400+
EMAIL;TYPE=WORK:jsmith@example.org
401+
END:VCARD
402+
{{< /qr >}}
403+
404+
Internally this shortcode calls the `images.QR` function. Please read the [related documentation] for implementation details and guidance.
405+
406+
[related documentation]: /functions/images/qr/
407+
408+
The `qr` shortcode accepts these named arguments:
409+
410+
text
411+
: (`string`) The text to encode, falling back to the text between the opening and closing shortcode tags.
412+
413+
level
414+
: (`string`) The error correction level to use when encoding the text, one of `low`, `medium`, `quartile`, or `high`. Default is `medium`.
415+
416+
scale
417+
: (`int`) The number of image pixels per QR code module. Must be greater than or equal to 2. Default is `4`.
418+
419+
targetDir
420+
: (`string`) The subdirectory within the [`publishDir`] where Hugo will place the generated image.
421+
422+
[`publishDir`]: /getting-started/configuration/#publishdir
423+
424+
alt
425+
: (`string`) The `alt` attribute of the `img` element.
426+
427+
class
428+
: (`string`) The `class` attribute of the `img` element.
429+
430+
id
431+
: (`string`) The `id` attribute of the `img` element.
432+
433+
title
434+
: (`string`) The `title` attribute of the `img` element.
435+
336436
### ref
337437

338438
{{% note %}}
+115
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,115 @@
1+
---
2+
title: images.QR
3+
description: Encodes the given text into a QR code using the specified options, returning an image resource.
4+
keywords: []
5+
action:
6+
aliases: []
7+
related: []
8+
returnType: images.ImageResource
9+
signatures: ['images.QR OPTIONS']
10+
toc: true
11+
math: true
12+
---
13+
14+
{{< new-in 0.141.0 >}}
15+
16+
The `images.QR` function encodes the given text into a [QR code] using the specified options, returning an image resource. The size of the generated image depends on three factors:
17+
18+
- Data length: Longer text necessitates a larger image to accommodate the increased information density.
19+
- Error correction level: Higher error correction levels enhance the QR code's resistance to damage, but this typically results in a slightly larger image size to maintain readability.
20+
- Pixels per module: The number of image pixels assigned to each individual module (the smallest unit of the QR code) directly impacts the overall image size. A higher pixel count per module leads to a larger, higher-resolution image.
21+
22+
Although the default option values are sufficient for most applications, you should test the rendered QR code both on-screen and in print.
23+
24+
[QR code]: https://en.wikipedia.org/wiki/QR_code
25+
26+
## Options
27+
28+
text
29+
: (`string`) The text to encode.
30+
31+
level
32+
: (`string`) The error correction level to use when encoding the text, one of `low`, `medium`, `quartile`, or `high`. Default is `medium`.
33+
34+
Error correction level|Redundancy
35+
:--|:--|:--
36+
low|20%
37+
medium|38%
38+
quartile|55%
39+
high|65%
40+
41+
scale
42+
: (`int`) The number of image pixels per QR code module. Must be greater than or equal to `2`. Default is `4`.
43+
44+
targetDir
45+
: (`string`) The subdirectory within the [`publishDir`] where Hugo will place the generated image. Use Unix-style slashes (`/`) to separarate path segments. If empty or not provided, the image is placed directly in the `publishDir` root. Hugo automatically creates the necessary subdirectories if they don't exist.
46+
47+
[`publishDir`]: /getting-started/configuration/#publishdir
48+
49+
## Examples
50+
51+
To create a QR code using the default values for `level` and `scale`:
52+
53+
```go-html-template
54+
{{ $opts := dict "text" "https://gohugo.io" }}
55+
{{ with images.QR $opts }}
56+
<img src="{{ .RelPermalink }}" width="{{ .Width }}" height="{{ .Height }}" alt="">
57+
{{ end }}
58+
```
59+
60+
{{< qr text="https://gohugo.io" class="qrcode" />}}
61+
62+
Specify `level`, `scale`, and `targetDir` as needed to achieve the desired result:
63+
64+
```go-html-template
65+
{{ $opts := dict
66+
"text" "https://gohugo.io"
67+
"level" "high"
68+
"scale" 3
69+
"targetDir" "codes"
70+
}}
71+
{{ with images.QR $opts }}
72+
<img src="{{ .RelPermalink }}" width="{{ .Width }}" height="{{ .Height }}" alt="">
73+
{{ end }}
74+
```
75+
76+
{{< qr text="https://gohugo.io" level="high" scale=3 targetDir="codes" class="qrcode" />}}
77+
78+
## Scale
79+
80+
As you decrease the size of a QR code, the maximum distance at which it can be reliably scanned by a device also decreases.
81+
82+
In the example above, we set the `scale` to `2`, resulting in a QR code where each module consists of 2x2 pixels. While this might be sufficient for on-screen display, it's likely to be problematic when printed at 600 dpi.
83+
84+
\[ \frac{2\:px}{module} \times \frac{1\:inch}{600\:px} \times \frac{25.4\:mm}{1\:inch} = \frac{0.085\:mm}{module} \]
85+
86+
This module size is half of the commonly recommended minimum of 0.170 mm.\
87+
If the QR code will be printed, use the default `scale` value of `4` pixels per module.
88+
89+
Avoid using Hugo's image processing methods to resize QR codes. Resizing can introduce blurring due to anti-aliasing when a QR code module occupies a fractional number of pixels.
90+
91+
{{% note %}}
92+
Always test the rendered QR code both on-screen and in print.
93+
{{% /note %}}
94+
95+
## Shortcode
96+
97+
Call the `qr` shortcode to insert a QR code into your content.
98+
99+
Use the self-closing syntax to pass the text as an argument:
100+
101+
```text
102+
{{</* qr text="https://gohugo.io" /*/>}}
103+
```
104+
105+
Or insert the text between the opening and closing tags:
106+
107+
```text
108+
{{</* qr */>}}
109+
https://gohugo.io
110+
{{</* /qr */>}}
111+
```
112+
113+
The `qr` shortcode accepts several arguments including `level` and `scale`. See the [related documentation] for details.
114+
115+
[related documentation]: /content-management/shortcodes/#qr

‎docs/data/embedded_template_urls.toml

+5-4
Original file line numberDiff line numberDiff line change
@@ -10,18 +10,18 @@
1010
'google_analytics' = 'google_analytics.html'
1111
'opengraph' = 'opengraph.html'
1212
'pagination' = 'pagination.html'
13-
'schema' = 'schema.html'
14-
'twitter_cards' = 'twitter_cards.html'
15-
1613
'robots' = '_default/robots.txt'
1714
'rss' = '_default/rss.xml'
15+
'schema' = 'schema.html'
1816
'sitemap' = '_default/sitemap.xml'
1917
'sitemapindex' = '_default/sitemapindex.xml'
18+
'twitter_cards' = 'twitter_cards.html'
2019

2120
# Render hooks
21+
'render-codeblock-goat' = '_default/_markup/render-codeblock-goat.html'
2222
'render-image' = '_default/_markup/render-image.html'
2323
'render-link' = '_default/_markup/render-link.html'
24-
'render-codeblock-goat' = '_default/_markup/render-codeblock-goat.html'
24+
'render-table' = '_default/_markup/render-table.html'
2525

2626
# Shortcodes
2727
'comment' = 'shortcodes/comment.html'
@@ -31,6 +31,7 @@
3131
'highlight' = 'shortcodes/highlight.html'
3232
'instagram' = 'shortcodes/instagram.html'
3333
'param' = 'shortcodes/param.html'
34+
'qr' = 'shortcodes/qr.html'
3435
'ref' = 'shortcodes/ref.html'
3536
'relref' = 'shortcodes/relref.html'
3637
'twitter' = 'shortcodes/twitter.html'

‎go.mod

+1
Original file line numberDiff line numberDiff line change
@@ -165,6 +165,7 @@ require (
165165
google.golang.org/protobuf v1.35.2 // indirect
166166
gopkg.in/yaml.v3 v3.0.1 // indirect
167167
howett.net/plist v1.0.0 // indirect
168+
rsc.io/qr v0.2.0 // indirect
168169
software.sslmate.com/src/go-pkcs12 v0.2.0 // indirect
169170
)
170171

‎go.sum

+2
Original file line numberDiff line numberDiff line change
@@ -878,6 +878,8 @@ honnef.co/go/tools v0.0.1-2020.1.4/go.mod h1:X/FiERA/W4tHapMX5mGpAtMSVEeEUOyHaw9
878878
howett.net/plist v1.0.0 h1:7CrbWYbPPO/PyNy38b2EB/+gYbjCe2DXBxgtOOZbSQM=
879879
howett.net/plist v1.0.0/go.mod h1:lqaXoTrLY4hg8tnEzNru53gicrbv7rrk+2xJA/7hw9g=
880880
rsc.io/binaryregexp v0.2.0/go.mod h1:qTv7/COck+e2FymRvadv62gMdZztPaShugOCi3I+8D8=
881+
rsc.io/qr v0.2.0 h1:6vBLea5/NRMVTz8V66gipeLycZMl/+UlFmk8DvqQ6WY=
882+
rsc.io/qr v0.2.0/go.mod h1:IF+uZjkb9fqyeF/4tlBoynqmQxUoPfWEKh921coOuXs=
881883
rsc.io/quote/v3 v3.1.0/go.mod h1:yEA65RcK8LyAZtP9Kv3t0HmxON59tX3rD+tICJqUlj0=
882884
rsc.io/sampler v1.3.0/go.mod h1:T1hPZKmBbMNahiBKFy5HrXp6adAjACjK9JXDnKaTXpA=
883885
software.sslmate.com/src/go-pkcs12 v0.2.0 h1:nlFkj7bTysH6VkC4fGphtjXRbezREPgrHuJG20hBGPE=

0 commit comments

Comments
 (0)