From 95274983cde6d9bb28a401e7741609e695ff0b22 Mon Sep 17 00:00:00 2001 From: Joe Mooring Date: Sat, 30 Sep 2023 22:11:09 -0700 Subject: [PATCH] Namespace functions and methods Closes #53 Closes #447 Closes #546 Closes #1163 Closes #1400 Closes #1406 Closes #1547 Closes #1622 Closes #1637 Closes #1639 Closes #1667 Closes #1745 Closes #2001 Closes #2018 Closes #2051 Closes #2256 Closes #2264 Closes #2275 Closes #2276 --- .cspell.json | 5 +- LICENSE.md | 393 ++-- README.md | 7 +- archetypes/functions.md | 9 +- archetypes/methods.md | 10 + archetypes/showcase/bio.md | 5 +- .../examples/zion-national-park-grayscale.jpg | Bin 0 -> 42537 bytes assets/images/examples/zion-national-park.jpg | Bin 0 -> 44553 bytes assets/images/logos/logo-128x128.png | Bin 0 -> 4759 bytes assets/images/logos/logo-256x256.png | Bin 0 -> 8822 bytes assets/images/logos/logo-512x512.png | Bin 0 -> 73651 bytes assets/images/logos/logo-64x64.png | Bin 0 -> 2533 bytes assets/images/logos/logo-96x96.png | Bin 0 -> 3644 bytes config/_default/menus/menus.en.toml | 8 +- config/_default/params.toml | 3 + content/en/about/hugo-and-gdpr.md | 8 +- content/en/about/license.md | 184 +- .../en/content-management/_common/_index.md | 13 + .../content-management/_common/page-kinds.md | 10 + content/en/content-management/archetypes.md | 27 +- .../en/content-management/build-options.md | 8 +- content/en/content-management/comments.md | 5 +- .../en/content-management/cross-references.md | 4 +- content/en/content-management/diagrams.md | 3 - .../en/content-management/emoji-shortcodes.md | 1621 +++++++++++++++++ content/en/content-management/formats.md | 5 +- content/en/content-management/front-matter.md | 6 +- .../image-processing/index.md | 39 +- content/en/content-management/menus.md | 15 +- content/en/content-management/multilingual.md | 41 +- .../content-management/organization/index.md | 31 +- content/en/content-management/page-bundles.md | 22 +- .../en/content-management/page-resources.md | 6 +- content/en/content-management/related.md | 16 +- content/en/content-management/sections.md | 64 +- content/en/content-management/shortcodes.md | 22 +- content/en/content-management/static-files.md | 4 +- content/en/content-management/summaries.md | 4 +- .../content-management/syntax-highlighting.md | 3 +- content/en/content-management/taxonomies.md | 15 +- content/en/content-management/toc.md | 5 +- content/en/content-management/urls.md | 34 +- content/en/contribute/development.md | 2 +- content/en/contribute/documentation.md | 8 +- content/en/documentation.md | 10 +- content/en/functions/Get.md | 26 - content/en/functions/GetPage.md | 84 - content/en/functions/Param.md | 50 - content/en/functions/Render.md | 28 - content/en/functions/RenderString.md | 35 - content/en/functions/Scratch.md | 151 -- content/en/functions/Store.md | 111 -- content/en/functions/Unix.md | 36 - content/en/functions/_common/_index.md | 13 + content/en/functions/_common/glob-patterns.md | 23 + .../_common/go-template-functions.md | 3 - content/en/functions/_common/index.md | 3 - content/en/functions/_common/locales.md | 7 + .../functions/_common/regular-expressions.md | 4 + .../functions/_common/time-layout-string.md | 10 +- content/en/functions/_index.md | 8 +- content/en/functions/cast/ToFloat.md | 17 +- content/en/functions/cast/ToInt.md | 24 +- content/en/functions/cast/ToString.md | 17 +- content/en/functions/cast/_index.md | 12 + content/en/functions/collections/After.md | 51 +- content/en/functions/collections/Append.md | 19 +- content/en/functions/collections/Apply.md | 9 +- .../en/functions/collections/Complement.md | 26 +- content/en/functions/collections/Delimit.md | 33 +- .../en/functions/collections/Dictionary.md | 36 +- content/en/functions/collections/EchoParam.md | 40 - content/en/functions/collections/First.md | 25 +- content/en/functions/collections/Group.md | 23 +- content/en/functions/collections/In.md | 18 +- .../en/functions/collections/IndexFunction.md | 57 +- content/en/functions/collections/Intersect.md | 18 +- content/en/functions/collections/IsSet.md | 23 +- content/en/functions/collections/KeyVals.md | 17 +- content/en/functions/collections/Last.md | 15 +- content/en/functions/collections/Merge.md | 13 +- .../en/functions/collections/NewScratch.md | 111 +- content/en/functions/collections/Querify.md | 12 +- content/en/functions/collections/Reverse.md | 10 +- content/en/functions/collections/Seq.md | 10 +- content/en/functions/collections/Shuffle.md | 11 +- content/en/functions/collections/Slice.md | 12 +- content/en/functions/collections/Sort.md | 35 +- content/en/functions/collections/SymDiff.md | 12 +- content/en/functions/collections/Union.md | 10 +- content/en/functions/collections/Uniq.md | 11 +- content/en/functions/collections/Where.md | 26 +- content/en/functions/collections/_index.md | 12 + .../compare/{Cond.md => Conditional.md} | 20 +- content/en/functions/compare/Default.md | 94 +- content/en/functions/compare/Eq.md | 21 +- content/en/functions/compare/Ge.md | 21 +- content/en/functions/compare/Gt.md | 21 +- content/en/functions/compare/Le.md | 21 +- content/en/functions/compare/Lt.md | 21 +- content/en/functions/compare/Ne.md | 21 +- content/en/functions/compare/_index.md | 12 + content/en/functions/crypto/FNV32a.md | 18 +- content/en/functions/crypto/HMAC.md | 19 +- content/en/functions/crypto/MD5.md | 22 +- content/en/functions/crypto/SHA1.md | 19 +- content/en/functions/crypto/SHA256.md | 19 +- content/en/functions/crypto/_index.md | 12 + content/en/functions/data/GetCSV.md | 53 +- content/en/functions/data/GetJSON.md | 55 +- content/en/functions/data/_index.md | 12 + content/en/functions/debug/Dump.md | 11 +- content/en/functions/debug/Timer.md | 11 +- content/en/functions/debug/_index.md | 12 + content/en/functions/encoding/Base64Decode.md | 15 +- content/en/functions/encoding/Base64Encode.md | 15 +- content/en/functions/encoding/Jsonify.md | 32 +- content/en/functions/encoding/_index.md | 12 + content/en/functions/fmt/Errorf.md | 15 +- content/en/functions/fmt/Erroridf.md | 17 +- content/en/functions/fmt/Print.md | 19 +- content/en/functions/fmt/Printf.md | 15 +- content/en/functions/fmt/Println.md | 19 +- content/en/functions/fmt/Warnf.md | 15 +- content/en/functions/fmt/_index.md | 12 + content/en/functions/global/_index.md | 11 + .../{hugo/index.md => global/hugo.md} | 50 +- .../{page/index.md => global/page.md} | 20 +- .../{site/index.md => global/site.md} | 16 +- .../functions/go-template/_common/_index.md | 13 + .../go-template/_common/text-template.md | 7 + .../go-template/_common/truthy-falsy.md | 5 + content/en/functions/go-template/_index.md | 14 + content/en/functions/go-template/and.md | 26 + content/en/functions/go-template/block.md | 55 + content/en/functions/go-template/break.md | 33 + content/en/functions/go-template/continue.md | 34 + content/en/functions/go-template/define.md | 55 + content/en/functions/go-template/else.md | 69 + content/en/functions/go-template/end.md | 65 + content/en/functions/go-template/if.md | 54 + content/en/functions/go-template/len.md | 24 +- content/en/functions/go-template/not.md | 35 + content/en/functions/go-template/or.md | 26 + content/en/functions/go-template/range.md | 154 +- content/en/functions/go-template/template.md | 49 + content/en/functions/go-template/urlquery.md | 18 +- content/en/functions/go-template/with.md | 88 +- content/en/functions/images/Brightness.md | 36 + content/en/functions/images/ColorBalance.md | 36 + content/en/functions/images/Colorize.md | 40 + content/en/functions/images/Config.md | 31 + content/en/functions/images/Contrast.md | 36 + content/en/functions/images/Filter.md | 67 + content/en/functions/images/Gamma.md | 36 + content/en/functions/images/GaussianBlur.md | 36 + content/en/functions/images/Grayscale.md | 34 + content/en/functions/images/Hue.md | 36 + content/en/functions/images/Invert.md | 34 + content/en/functions/images/Opacity.md | 50 + content/en/functions/images/Overlay.md | 54 + content/en/functions/images/Padding.md | 73 + content/en/functions/images/Pixelate.md | 34 + content/en/functions/images/Process.md | 110 ++ content/en/functions/images/Saturation.md | 36 + content/en/functions/images/Sepia.md | 36 + content/en/functions/images/Sigmoid.md | 40 + content/en/functions/images/Text.md | 95 + content/en/functions/images/UnsharpMask.md | 40 + content/en/functions/images/_common/_index.md | 13 + .../images/_common/apply-image-filter.md | 27 + content/en/functions/images/_index.md | 12 + content/en/functions/images/index.md | 302 --- content/en/functions/inflect/Humanize.md | 29 +- content/en/functions/inflect/Pluralize.md | 19 +- content/en/functions/inflect/Singularize.md | 19 +- content/en/functions/inflect/_index.md | 12 + content/en/functions/js/Build.md | 182 ++ content/en/functions/js/_index.md | 12 + content/en/functions/lang/FormatAccounting.md | 24 +- content/en/functions/lang/FormatCurrency.md | 24 +- content/en/functions/lang/FormatNumber.md | 24 +- .../en/functions/lang/FormatNumberCustom.md | 25 +- content/en/functions/lang/FormatPercent.md | 24 +- content/en/functions/lang/Merge.md | 14 +- content/en/functions/lang/Translate.md | 26 +- content/en/functions/lang/_index.md | 12 + content/en/functions/math/Abs.md | 15 + content/en/functions/math/Add.md | 22 + content/en/functions/math/Ceil.md | 17 + content/en/functions/math/Counter.md | 35 + content/en/functions/math/Div.md | 22 + content/en/functions/math/Floor.md | 17 + content/en/functions/math/Log.md | 15 + content/en/functions/math/Max.md | 16 + content/en/functions/math/Min.md | 16 + content/en/functions/math/Mod.md | 16 + content/en/functions/math/ModBool.md | 16 + content/en/functions/math/Mul.md | 22 + content/en/functions/math/Pow.md | 16 + content/en/functions/math/Product.md | 20 + content/en/functions/math/Round.md | 17 + content/en/functions/math/Sqrt.md | 16 + content/en/functions/math/Sub.md | 21 + content/en/functions/math/Sum.md | 19 + content/en/functions/math/_index.md | 11 + content/en/functions/math/index.md | 39 - content/en/functions/os/FileExists.md | 33 +- content/en/functions/os/Getenv.md | 52 +- content/en/functions/os/ReadDir.md | 23 +- content/en/functions/os/ReadFile.md | 21 +- content/en/functions/os/Stat.md | 20 +- content/en/functions/os/_index.md | 12 + content/en/functions/partials/Include.md | 20 +- .../en/functions/partials/IncludeCached.md | 21 +- content/en/functions/partials/_index.md | 12 + content/en/functions/path/Base.md | 39 +- content/en/functions/path/BaseName.md | 37 +- content/en/functions/path/Clean.md | 42 +- content/en/functions/path/Dir.md | 41 +- content/en/functions/path/Ext.md | 33 +- content/en/functions/path/Join.md | 44 +- content/en/functions/path/Split.md | 45 +- content/en/functions/path/_index.md | 12 + content/en/functions/reflect/IsMap.md | 13 +- content/en/functions/reflect/IsSlice.md | 13 +- content/en/functions/reflect/_index.md | 12 + content/en/functions/resources/Babel.md | 88 + content/en/functions/resources/ByType.md | 34 + content/en/functions/resources/Concat.md | 21 + content/en/functions/resources/Copy.md | 30 + .../functions/resources/ExecuteAsTemplate.md | 56 + content/en/functions/resources/Fingerprint.md | 42 + content/en/functions/resources/FromString.md | 71 + content/en/functions/resources/Get.md | 30 + content/en/functions/resources/GetMatch.md | 36 + content/en/functions/resources/GetRemote.md | 176 ++ content/en/functions/resources/Match.md | 36 + content/en/functions/resources/Minify.md | 23 + content/en/functions/resources/PostCSS.md | 129 ++ content/en/functions/resources/PostProcess.md | 160 ++ content/en/functions/resources/ToCSS.md | 224 +++ .../en/functions/resources/_common/_index.md | 12 + .../_common/postcss-windows-warning.md | 8 + content/en/functions/resources/_index.md | 12 + content/en/functions/safe/CSS.md | 23 +- content/en/functions/safe/HTML.md | 25 +- content/en/functions/safe/HTMLAttr.md | 25 +- content/en/functions/safe/JS.md | 23 +- content/en/functions/safe/JSStr.md | 22 +- content/en/functions/safe/URL.md | 27 +- content/en/functions/safe/_index.md | 14 + content/en/functions/strings/Chomp.md | 36 +- content/en/functions/strings/Contains.md | 22 +- content/en/functions/strings/ContainsAny.md | 21 +- .../en/functions/strings/ContainsNonSpace.md | 23 +- content/en/functions/strings/Count.md | 18 +- content/en/functions/strings/CountRunes.md | 19 +- content/en/functions/strings/CountWords.md | 25 +- .../en/functions/strings/FindRESubmatch.md | 19 +- content/en/functions/strings/FindRe.md | 19 +- content/en/functions/strings/FirstUpper.md | 18 +- content/en/functions/strings/HasPrefix.md | 21 +- content/en/functions/strings/HasSuffix.md | 21 +- content/en/functions/strings/Repeat.md | 12 +- content/en/functions/strings/Replace.md | 32 +- content/en/functions/strings/ReplaceRE.md | 28 +- content/en/functions/strings/RuneCount.md | 18 +- content/en/functions/strings/SliceString.md | 16 +- content/en/functions/strings/Split.md | 18 +- content/en/functions/strings/Substr.md | 34 +- content/en/functions/strings/Title.md | 32 +- content/en/functions/strings/ToLower.md | 25 +- content/en/functions/strings/ToUpper.md | 24 +- content/en/functions/strings/Trim.md | 64 +- content/en/functions/strings/TrimLeft.md | 31 +- content/en/functions/strings/TrimPrefix.md | 30 +- content/en/functions/strings/TrimRight.md | 31 +- content/en/functions/strings/TrimSuffix.md | 30 +- content/en/functions/strings/Truncate.md | 10 +- content/en/functions/strings/_index.md | 12 + content/en/functions/templates/Exists.md | 10 +- content/en/functions/templates/_index.md | 13 + content/en/functions/time/AsTime.md | 77 +- content/en/functions/time/Duration.md | 44 +- content/en/functions/time/Format.md | 49 +- content/en/functions/time/Now.md | 61 +- content/en/functions/time/ParseDuration.md | 39 +- content/en/functions/time/_common/_index.md | 13 + .../_common/parsable-date-time-strings.md | 14 + content/en/functions/time/_index.md | 12 + .../en/functions/transform/CanHighlight.md | 14 +- content/en/functions/transform/Emojify.md | 16 +- content/en/functions/transform/HTMLEscape.md | 30 +- .../en/functions/transform/HTMLUnescape.md | 30 +- content/en/functions/transform/Highlight.md | 22 +- .../functions/transform/HighlightCodeBlock.md | 19 +- content/en/functions/transform/Markdownify.md | 18 +- content/en/functions/transform/Plainify.md | 12 +- content/en/functions/transform/Remarshal.md | 16 +- content/en/functions/transform/Unmarshal.md | 292 ++- content/en/functions/transform/_index.md | 12 + content/en/functions/urls/AbsLangURL.md | 19 +- content/en/functions/urls/AbsURL.md | 19 +- content/en/functions/urls/Anchorize.md | 27 +- content/en/functions/urls/JoinPath.md | 26 +- content/en/functions/urls/Parse.md | 12 +- content/en/functions/urls/Ref.md | 46 +- content/en/functions/urls/RelLangURL.md | 23 +- content/en/functions/urls/RelRef.md | 52 +- content/en/functions/urls/RelURL.md | 19 +- content/en/functions/urls/URLize.md | 15 +- content/en/functions/urls/_index.md | 12 + .../getting-started/configuration-markup.md | 3 +- content/en/getting-started/configuration.md | 59 +- .../en/getting-started/directory-structure.md | 57 +- content/en/getting-started/glossary.md | 139 +- content/en/getting-started/quick-start.md | 2 +- content/en/getting-started/usage.md | 33 +- .../deployment-with-rsync.md | 4 +- .../hosting-on-firebase.md | 13 +- .../hosting-on-github/index.md | 5 +- .../hosting-on-gitlab.md | 4 +- .../hosting-on-keycdn.md | 4 +- .../en/hosting-and-deployment/hugo-deploy.md | 4 +- content/en/hugo-modules/configuration.md | 10 +- content/en/hugo-modules/theme-components.md | 7 +- content/en/hugo-modules/use-modules.md | 17 +- content/en/hugo-pipes/babel.md | 8 +- content/en/hugo-pipes/bundling.md | 5 +- content/en/hugo-pipes/fingerprint.md | 5 +- content/en/hugo-pipes/introduction.md | 146 +- content/en/hugo-pipes/js.md | 8 +- content/en/hugo-pipes/minification.md | 5 +- content/en/hugo-pipes/postcss.md | 9 +- content/en/hugo-pipes/postprocess.md | 10 +- content/en/hugo-pipes/resource-from-string.md | 5 +- .../en/hugo-pipes/resource-from-template.md | 5 +- .../en/hugo-pipes/transpile-sass-to-css.md | 18 +- .../en/installation/_common/01-editions.md | 4 + .../installation/_common/02-prerequisites.md | 4 + .../_common/03-prebuilt-binaries.md | 8 +- .../_common/04-build-from-source.md | 4 + content/en/installation/_common/_index.md | 13 + content/en/installation/_common/homebrew.md | 4 + content/en/installation/_common/index.md | 3 - content/en/installation/_index.md | 2 +- content/en/installation/bsd.md | 8 +- content/en/installation/linux.md | 12 +- content/en/installation/macos.md | 10 +- content/en/installation/windows.md | 8 +- content/en/methods/_common/_index.md | 13 + ...next-prev-on-page-vs-next-prev-on-pages.md | 37 + content/en/methods/_index.md | 16 + content/en/methods/duration/Abs.md | 17 + content/en/methods/duration/Hours.md | 17 + content/en/methods/duration/Microseconds.md | 17 + content/en/methods/duration/Milliseconds.md | 17 + content/en/methods/duration/Minutes.md | 17 + content/en/methods/duration/Nanoseconds.md | 17 + content/en/methods/duration/Round.md | 20 + content/en/methods/duration/Seconds.md | 17 + content/en/methods/duration/Truncate.md | 21 + content/en/methods/duration/_index.md | 12 + content/en/methods/page/Aliases.md | 29 + content/en/methods/page/AllTranslations.md | 90 + .../methods/page/AlternativeOutputFormats.md | 43 + content/en/methods/page/Ancestors.md | 85 + content/en/methods/page/BundleType.md | 37 + content/en/methods/page/CodeOwners.md | 66 + content/en/methods/page/Content.md | 22 + content/en/methods/page/CurrentSection.md | 60 + content/en/methods/page/Data.md | 111 ++ content/en/methods/page/Date.md | 39 + content/en/methods/page/Description.md | 28 + content/en/methods/page/Draft.md | 21 + content/en/methods/page/Eq.md | 21 + content/en/methods/page/ExpiryDate.md | 35 + content/en/methods/page/File.md | 180 ++ content/en/methods/page/FirstSection.md | 56 + content/en/methods/page/Fragments.md | 104 ++ content/en/methods/page/FuzzyWordCount.md | 20 + content/en/methods/page/GetPage.md | 65 + content/en/methods/page/GetTerms.md | 40 + content/en/methods/page/GitInfo.md | 130 ++ .../page}/HasMenuCurrent.md | 16 +- content/en/methods/page/HasShortcode.md | 50 + content/en/methods/page/HeadingsFiltered.md | 18 + content/en/methods/page/InSection.md | 102 ++ content/en/methods/page/IsAncestor.md | 100 + content/en/methods/page/IsDescendant.md | 99 + content/en/methods/page/IsHome.md | 31 + .../page}/IsMenuCurrent.md | 16 +- content/en/methods/page/IsNode.md | 36 + content/en/methods/page/IsPage.md | 31 + content/en/methods/page/IsSection.md | 31 + content/en/methods/page/IsTranslated.md | 59 + content/en/methods/page/Keywords.md | 46 + content/en/methods/page/Kind.md | 35 + content/en/methods/page/Language.md | 65 + content/en/methods/page/Lastmod.md | 40 + content/en/methods/page/Layout.md | 40 + content/en/methods/page/LinkTitle.md | 30 + content/en/methods/page/Next.md | 53 + content/en/methods/page/NextInSection.md | 71 + content/en/methods/page/OutputFormats.md | 40 + content/en/methods/page/Page.md | 40 + content/en/methods/page/Pages.md | 90 + content/en/methods/page/Paginate.md | 50 + content/en/methods/page/Paginator.md | 42 + content/en/methods/page/Param.md | 47 + content/en/methods/page/Params.md | 43 + content/en/methods/page/Parent.md | 60 + content/en/methods/page/Permalink.md | 25 + content/en/methods/page/Plain.md | 28 + content/en/methods/page/PlainWords.md | 34 + content/en/methods/page/Prev.md | 53 + content/en/methods/page/PrevInSection.md | 72 + content/en/methods/page/PublishDate.md | 35 + content/en/methods/page/RawContent.md | 31 + content/en/methods/page/ReadingTime.md | 49 + content/en/methods/page/Ref.md | 44 + content/en/methods/page/RegularPages.md | 87 + .../en/methods/page/RegularPagesRecursive.md | 90 + content/en/methods/page/RelPermalink.md | 25 + content/en/methods/page/RelRef.md | 44 + content/en/methods/page/Render.md | 75 + content/en/methods/page/RenderShortcodes.md | 76 + content/en/methods/page/RenderString.md | 51 + content/en/methods/page/Resources.md | 81 + content/en/methods/page/Scratch.md | 23 + content/en/methods/page/Section.md | 54 + content/en/methods/page/Sections.md | 69 + content/en/methods/page/Site.md | 19 + content/en/methods/page/Sitemap.md | 70 + content/en/methods/page/Sites.md | 69 + content/en/methods/page/Slug.md | 25 + content/en/methods/page/Store.md | 97 + content/en/methods/page/Summary.md | 29 + content/en/methods/page/TableOfContents.md | 47 + content/en/methods/page/Title.md | 23 + content/en/methods/page/TranslationKey.md | 74 + content/en/methods/page/Translations.md | 88 + content/en/methods/page/Truncated.md | 35 + content/en/methods/page/Type.md | 56 + content/en/methods/page/Weight.md | 27 + content/en/methods/page/WordCount.md | 20 + content/en/methods/page/_common/_index.md | 13 + .../page/_common/definition-of-section.md | 5 + .../page/_common/output-format-definition.md | 11 + .../page/_common/output-format-methods.md | 27 + .../methods/page/_common/scratch-methods.md | 79 + content/en/methods/page/_index.md | 12 + content/en/methods/pages/ByDate.md | 31 + content/en/methods/pages/ByExpiryDate.md | 31 + content/en/methods/pages/ByLanguage.md | 24 + content/en/methods/pages/ByLastmod.md | 31 + content/en/methods/pages/ByLength.md | 24 + content/en/methods/pages/ByLinkTitle.md | 26 + content/en/methods/pages/ByParam.md | 36 + content/en/methods/pages/ByPublishDate.md | 31 + content/en/methods/pages/ByTitle.md | 26 + content/en/methods/pages/ByWeight.md | 28 + content/en/methods/pages/GroupBy.md | 36 + content/en/methods/pages/GroupByDate.md | 68 + content/en/methods/pages/GroupByExpiryDate.md | 68 + content/en/methods/pages/GroupByLastmod.md | 68 + content/en/methods/pages/GroupByParam.md | 36 + content/en/methods/pages/GroupByParamDate.md | 65 + .../en/methods/pages/GroupByPublishDate.md | 68 + content/en/methods/pages/Len.md | 14 + content/en/methods/pages/Limit.md | 16 + content/en/methods/pages/Next.md | 55 + content/en/methods/pages/Prev.md | 55 + content/en/methods/pages/Related.md | 77 + content/en/methods/pages/Reverse.md | 16 + content/en/methods/pages/_common/_index.md | 13 + .../methods/pages/_common/group-sort-order.md | 5 + content/en/methods/pages/_index.md | 13 + content/en/methods/resource/Colors.md | 20 + content/en/methods/resource/Content.md | 61 + content/en/methods/resource/Crop.md | 48 + content/en/methods/resource/Data.md | 53 + content/en/methods/resource/Err.md | 56 + content/en/methods/resource/Exif.md | 78 + content/en/methods/resource/Fill.md | 48 + content/en/methods/resource/Filter.md | 68 + content/en/methods/resource/Fit.md | 48 + content/en/methods/resource/Height.md | 27 + content/en/methods/resource/Key.md | 45 + content/en/methods/resource/MediaType.md | 52 + content/en/methods/resource/Name.md | 82 + content/en/methods/resource/Params.md | 65 + content/en/methods/resource/Permalink.md | 25 + content/en/methods/resource/Process.md | 66 + content/en/methods/resource/Publish.md | 35 + content/en/methods/resource/RelPermalink.md | 25 + content/en/methods/resource/Resize.md | 49 + content/en/methods/resource/ResourceType.md | 43 + content/en/methods/resource/Title.md | 95 + content/en/methods/resource/Width.md | 27 + content/en/methods/resource/_common/_index.md | 13 + .../_common/global-page-remote-resources.md | 13 + .../resource/_common/processing-spec.md | 34 + content/en/methods/resource/_index.md | 12 + content/en/methods/shortcode/Get.md | 50 + content/en/methods/shortcode/Inner.md | 153 ++ content/en/methods/shortcode/InnerDeindent.md | 99 + content/en/methods/shortcode/IsNamedParams.md | 30 + content/en/methods/shortcode/Name.md | 29 + content/en/methods/shortcode/Ordinal.md | 50 + content/en/methods/shortcode/Page.md | 36 + content/en/methods/shortcode/Params.md | 32 + content/en/methods/shortcode/Parent.md | 50 + content/en/methods/shortcode/Position.md | 33 + content/en/methods/shortcode/Scratch.md | 24 + content/en/methods/shortcode/_index.md | 12 + content/en/methods/site/AllPages.md | 26 + content/en/methods/site/BaseURL.md | 37 + content/en/methods/site/BuildDrafts.md | 28 + content/en/methods/site/Config.md | 57 + content/en/methods/site/Copyright.md | 22 + content/en/methods/site/Data.md | 108 ++ content/en/methods/site/GetPage.md | 109 ++ content/en/methods/site/Home.md | 25 + content/en/methods/site/IsMultiLingual.md | 34 + content/en/methods/site/Language.md | 83 + content/en/methods/site/LanguagePrefix.md | 53 + content/en/methods/site/Languages.md | 59 + content/en/methods/site/LastChange.md | 21 + content/en/methods/site/MainSections.md | 55 + content/en/methods/site/Menus.md | 94 + content/en/methods/site/Pages.md | 26 + content/en/methods/site/Params.md | 47 + content/en/methods/site/RegularPages.md | 38 + content/en/methods/site/Sections.md | 41 + content/en/methods/site/Sites.md | 66 + content/en/methods/site/Taxonomies.md | 99 + content/en/methods/site/Title.md | 22 + content/en/methods/site/_index.md | 12 + content/en/methods/taxonomy/Alphabetical.md | 78 + content/en/methods/taxonomy/ByCount.md | 78 + content/en/methods/taxonomy/Count.md | 26 + content/en/methods/taxonomy/Get.md | 72 + content/en/methods/taxonomy/_common/_index.md | 13 + .../taxonomy/_common/get-a-taxonomy-object.md | 68 + .../ordered-taxonomy-element-methods.md | 25 + content/en/methods/taxonomy/_index.md | 12 + content/en/methods/time/Add.md | 23 + .../en/{functions => methods/time}/AddDate.md | 14 +- content/en/methods/time/After.md | 20 + content/en/methods/time/Before.md | 19 + content/en/methods/time/Day.md | 21 + content/en/methods/time/Equal.md | 20 + .../en/{functions => methods/time}/Format.md | 21 +- content/en/methods/time/Hour.md | 21 + content/en/methods/time/IsDST.md | 19 + content/en/methods/time/IsZero.md | 19 + content/en/methods/time/Local.md | 17 + content/en/methods/time/Minute.md | 21 + content/en/methods/time/Month.md | 30 + content/en/methods/time/Nanosecond.md | 16 + content/en/methods/time/Second.md | 21 + content/en/methods/time/Sub.md | 18 + content/en/methods/time/UTC.md | 16 + content/en/methods/time/Unix.md | 21 + content/en/methods/time/UnixMicro.md | 21 + content/en/methods/time/UnixMilli.md | 21 + content/en/methods/time/UnixNano.md | 21 + content/en/methods/time/Weekday.md | 24 + content/en/methods/time/Year.md | 21 + content/en/methods/time/YearDay.md | 15 + content/en/methods/time/_index.md | 13 + content/en/myshowcase/bio.md | 5 +- content/en/showcase/1password-support/bio.md | 1 - .../en/showcase/1password-support/index.md | 2 +- content/en/showcase/alora-labs/bio.md | 2 +- content/en/showcase/alora-labs/index.md | 6 +- content/en/showcase/ampio-help/index.md | 2 +- content/en/showcase/bypasscensorship/bio.md | 2 +- content/en/showcase/bypasscensorship/index.md | 2 +- content/en/showcase/digitalgov/index.md | 30 +- content/en/showcase/forestry/index.md | 4 +- content/en/showcase/godot-tutorials/index.md | 3 +- content/en/showcase/letsencrypt/index.md | 1 - content/en/showcase/pharmaseal/index.md | 4 +- .../quiply-employee-communications-app/bio.md | 2 +- content/en/showcase/template/bio.md | 5 +- content/en/templates/data-templates.md | 118 +- content/en/templates/internal.md | 13 +- content/en/templates/introduction.md | 19 +- content/en/templates/lists/index.md | 16 +- content/en/templates/lookup-order.md | 10 +- content/en/templates/menu-templates.md | 4 +- content/en/templates/output-formats.md | 10 +- content/en/templates/partials.md | 2 +- content/en/templates/render-hooks.md | 1 - content/en/templates/robots.md | 2 +- content/en/templates/rss.md | 8 +- content/en/templates/section-templates.md | 12 +- content/en/templates/shortcode-templates.md | 27 +- content/en/templates/single-page-templates.md | 6 +- content/en/templates/sitemap-template.md | 2 +- content/en/templates/taxonomy-templates.md | 15 +- content/en/templates/views.md | 3 +- content/en/tools/editors.md | 70 +- content/en/tools/frontends.md | 20 +- content/en/tools/migrations.md | 75 +- content/en/tools/other.md | 2 +- content/en/tools/search.md | 51 +- content/en/troubleshooting/faq.md | 2 +- content/en/variables/_common/_index.md | 13 + .../_common/consistent-terminology.md | 16 + content/en/variables/_index.md | 6 +- content/en/variables/file.md | 18 + content/en/variables/files.md | 102 -- content/en/variables/git.md | 69 +- .../en/variables/{menus.md => menu-entry.md} | 48 +- content/en/variables/page.md | 356 +--- content/en/variables/pages.md | 38 +- content/en/variables/shortcode.md | 16 + content/en/variables/shortcodes.md | 45 - content/en/variables/site.md | 113 +- content/en/variables/sitemap.md | 24 - content/en/variables/taxonomy.md | 137 +- data/page_filters.yaml | 89 + data/titles.toml | 2 - hugo.toml | 4 - layouts/_default/_markup/render-link.html | 250 +++ layouts/_default/page.html | 18 +- layouts/_default/single.html | 29 + layouts/partials/boxes-section-summaries.html | 18 +- layouts/partials/docs/functions-aliases.html | 2 +- .../partials/docs/functions-return-type.html | 6 + .../partials/docs/functions-signatures.html | 2 +- layouts/partials/pagelayout.html | 36 + layouts/partials/related.html | 13 +- layouts/partials/right-sidebar.html | 29 + layouts/shortcodes/code-toggle.html | 2 +- layouts/shortcodes/code.html | 2 +- layouts/shortcodes/funcsig.html | 2 +- layouts/shortcodes/getcontent.html | 21 - layouts/shortcodes/img.html | 374 ++++ layouts/shortcodes/imgproc.html | 63 +- layouts/shortcodes/include.html | 20 + layouts/shortcodes/list-pages-in-section.html | 96 + layouts/shortcodes/readfile.html | 37 +- 647 files changed, 19180 insertions(+), 5223 deletions(-) create mode 100644 archetypes/methods.md create mode 100644 assets/images/examples/zion-national-park-grayscale.jpg create mode 100644 assets/images/examples/zion-national-park.jpg create mode 100644 assets/images/logos/logo-128x128.png create mode 100644 assets/images/logos/logo-256x256.png create mode 100644 assets/images/logos/logo-512x512.png create mode 100644 assets/images/logos/logo-64x64.png create mode 100644 assets/images/logos/logo-96x96.png create mode 100644 content/en/content-management/_common/_index.md rename layouts/shortcodes/page-kinds.html => content/en/content-management/_common/page-kinds.md (65%) create mode 100644 content/en/content-management/emoji-shortcodes.md delete mode 100644 content/en/functions/Get.md delete mode 100644 content/en/functions/GetPage.md delete mode 100644 content/en/functions/Param.md delete mode 100644 content/en/functions/Render.md delete mode 100644 content/en/functions/RenderString.md delete mode 100644 content/en/functions/Scratch.md delete mode 100644 content/en/functions/Store.md delete mode 100644 content/en/functions/Unix.md create mode 100644 content/en/functions/_common/_index.md create mode 100644 content/en/functions/_common/glob-patterns.md delete mode 100644 content/en/functions/_common/go-template-functions.md delete mode 100644 content/en/functions/_common/index.md create mode 100644 content/en/functions/cast/_index.md delete mode 100644 content/en/functions/collections/EchoParam.md create mode 100644 content/en/functions/collections/_index.md rename content/en/functions/compare/{Cond.md => Conditional.md} (79%) create mode 100644 content/en/functions/compare/_index.md create mode 100644 content/en/functions/crypto/_index.md create mode 100644 content/en/functions/data/_index.md create mode 100644 content/en/functions/debug/_index.md create mode 100644 content/en/functions/encoding/_index.md create mode 100644 content/en/functions/fmt/_index.md create mode 100644 content/en/functions/global/_index.md rename content/en/functions/{hugo/index.md => global/hugo.md} (78%) rename content/en/functions/{page/index.md => global/page.md} (91%) rename content/en/functions/{site/index.md => global/site.md} (75%) create mode 100644 content/en/functions/go-template/_common/_index.md create mode 100644 content/en/functions/go-template/_common/text-template.md create mode 100644 content/en/functions/go-template/_common/truthy-falsy.md create mode 100644 content/en/functions/go-template/_index.md create mode 100644 content/en/functions/go-template/and.md create mode 100644 content/en/functions/go-template/block.md create mode 100644 content/en/functions/go-template/break.md create mode 100644 content/en/functions/go-template/continue.md create mode 100644 content/en/functions/go-template/define.md create mode 100644 content/en/functions/go-template/else.md create mode 100644 content/en/functions/go-template/end.md create mode 100644 content/en/functions/go-template/if.md create mode 100644 content/en/functions/go-template/not.md create mode 100644 content/en/functions/go-template/or.md create mode 100644 content/en/functions/go-template/template.md create mode 100644 content/en/functions/images/Brightness.md create mode 100644 content/en/functions/images/ColorBalance.md create mode 100644 content/en/functions/images/Colorize.md create mode 100644 content/en/functions/images/Config.md create mode 100644 content/en/functions/images/Contrast.md create mode 100644 content/en/functions/images/Filter.md create mode 100644 content/en/functions/images/Gamma.md create mode 100644 content/en/functions/images/GaussianBlur.md create mode 100644 content/en/functions/images/Grayscale.md create mode 100644 content/en/functions/images/Hue.md create mode 100644 content/en/functions/images/Invert.md create mode 100644 content/en/functions/images/Opacity.md create mode 100644 content/en/functions/images/Overlay.md create mode 100644 content/en/functions/images/Padding.md create mode 100644 content/en/functions/images/Pixelate.md create mode 100644 content/en/functions/images/Process.md create mode 100644 content/en/functions/images/Saturation.md create mode 100644 content/en/functions/images/Sepia.md create mode 100644 content/en/functions/images/Sigmoid.md create mode 100644 content/en/functions/images/Text.md create mode 100644 content/en/functions/images/UnsharpMask.md create mode 100644 content/en/functions/images/_common/_index.md create mode 100644 content/en/functions/images/_common/apply-image-filter.md create mode 100644 content/en/functions/images/_index.md delete mode 100644 content/en/functions/images/index.md create mode 100644 content/en/functions/inflect/_index.md create mode 100644 content/en/functions/js/Build.md create mode 100644 content/en/functions/js/_index.md create mode 100644 content/en/functions/lang/_index.md create mode 100644 content/en/functions/math/Abs.md create mode 100644 content/en/functions/math/Add.md create mode 100644 content/en/functions/math/Ceil.md create mode 100644 content/en/functions/math/Counter.md create mode 100644 content/en/functions/math/Div.md create mode 100644 content/en/functions/math/Floor.md create mode 100644 content/en/functions/math/Log.md create mode 100644 content/en/functions/math/Max.md create mode 100644 content/en/functions/math/Min.md create mode 100644 content/en/functions/math/Mod.md create mode 100644 content/en/functions/math/ModBool.md create mode 100644 content/en/functions/math/Mul.md create mode 100644 content/en/functions/math/Pow.md create mode 100644 content/en/functions/math/Product.md create mode 100644 content/en/functions/math/Round.md create mode 100644 content/en/functions/math/Sqrt.md create mode 100644 content/en/functions/math/Sub.md create mode 100644 content/en/functions/math/Sum.md create mode 100644 content/en/functions/math/_index.md delete mode 100644 content/en/functions/math/index.md create mode 100644 content/en/functions/os/_index.md create mode 100644 content/en/functions/partials/_index.md create mode 100644 content/en/functions/path/_index.md create mode 100644 content/en/functions/reflect/_index.md create mode 100644 content/en/functions/resources/Babel.md create mode 100644 content/en/functions/resources/ByType.md create mode 100644 content/en/functions/resources/Concat.md create mode 100644 content/en/functions/resources/Copy.md create mode 100644 content/en/functions/resources/ExecuteAsTemplate.md create mode 100644 content/en/functions/resources/Fingerprint.md create mode 100644 content/en/functions/resources/FromString.md create mode 100644 content/en/functions/resources/Get.md create mode 100644 content/en/functions/resources/GetMatch.md create mode 100644 content/en/functions/resources/GetRemote.md create mode 100644 content/en/functions/resources/Match.md create mode 100644 content/en/functions/resources/Minify.md create mode 100644 content/en/functions/resources/PostCSS.md create mode 100644 content/en/functions/resources/PostProcess.md create mode 100644 content/en/functions/resources/ToCSS.md create mode 100644 content/en/functions/resources/_common/_index.md create mode 100644 content/en/functions/resources/_common/postcss-windows-warning.md create mode 100644 content/en/functions/resources/_index.md create mode 100644 content/en/functions/safe/_index.md create mode 100644 content/en/functions/strings/_index.md create mode 100644 content/en/functions/templates/_index.md create mode 100644 content/en/functions/time/_common/_index.md create mode 100644 content/en/functions/time/_common/parsable-date-time-strings.md create mode 100644 content/en/functions/time/_index.md create mode 100644 content/en/functions/transform/_index.md create mode 100644 content/en/functions/urls/_index.md create mode 100644 content/en/installation/_common/_index.md delete mode 100644 content/en/installation/_common/index.md create mode 100644 content/en/methods/_common/_index.md create mode 100644 content/en/methods/_common/next-prev-on-page-vs-next-prev-on-pages.md create mode 100644 content/en/methods/_index.md create mode 100644 content/en/methods/duration/Abs.md create mode 100644 content/en/methods/duration/Hours.md create mode 100644 content/en/methods/duration/Microseconds.md create mode 100644 content/en/methods/duration/Milliseconds.md create mode 100644 content/en/methods/duration/Minutes.md create mode 100644 content/en/methods/duration/Nanoseconds.md create mode 100644 content/en/methods/duration/Round.md create mode 100644 content/en/methods/duration/Seconds.md create mode 100644 content/en/methods/duration/Truncate.md create mode 100644 content/en/methods/duration/_index.md create mode 100644 content/en/methods/page/Aliases.md create mode 100644 content/en/methods/page/AllTranslations.md create mode 100644 content/en/methods/page/AlternativeOutputFormats.md create mode 100644 content/en/methods/page/Ancestors.md create mode 100644 content/en/methods/page/BundleType.md create mode 100644 content/en/methods/page/CodeOwners.md create mode 100644 content/en/methods/page/Content.md create mode 100644 content/en/methods/page/CurrentSection.md create mode 100644 content/en/methods/page/Data.md create mode 100644 content/en/methods/page/Date.md create mode 100644 content/en/methods/page/Description.md create mode 100644 content/en/methods/page/Draft.md create mode 100644 content/en/methods/page/Eq.md create mode 100644 content/en/methods/page/ExpiryDate.md create mode 100644 content/en/methods/page/File.md create mode 100644 content/en/methods/page/FirstSection.md create mode 100644 content/en/methods/page/Fragments.md create mode 100644 content/en/methods/page/FuzzyWordCount.md create mode 100644 content/en/methods/page/GetPage.md create mode 100644 content/en/methods/page/GetTerms.md create mode 100644 content/en/methods/page/GitInfo.md rename content/en/{functions => methods/page}/HasMenuCurrent.md (84%) create mode 100644 content/en/methods/page/HasShortcode.md create mode 100644 content/en/methods/page/HeadingsFiltered.md create mode 100644 content/en/methods/page/InSection.md create mode 100644 content/en/methods/page/IsAncestor.md create mode 100644 content/en/methods/page/IsDescendant.md create mode 100644 content/en/methods/page/IsHome.md rename content/en/{functions => methods/page}/IsMenuCurrent.md (81%) create mode 100644 content/en/methods/page/IsNode.md create mode 100644 content/en/methods/page/IsPage.md create mode 100644 content/en/methods/page/IsSection.md create mode 100644 content/en/methods/page/IsTranslated.md create mode 100644 content/en/methods/page/Keywords.md create mode 100644 content/en/methods/page/Kind.md create mode 100644 content/en/methods/page/Language.md create mode 100644 content/en/methods/page/Lastmod.md create mode 100644 content/en/methods/page/Layout.md create mode 100644 content/en/methods/page/LinkTitle.md create mode 100644 content/en/methods/page/Next.md create mode 100644 content/en/methods/page/NextInSection.md create mode 100644 content/en/methods/page/OutputFormats.md create mode 100644 content/en/methods/page/Page.md create mode 100644 content/en/methods/page/Pages.md create mode 100644 content/en/methods/page/Paginate.md create mode 100644 content/en/methods/page/Paginator.md create mode 100644 content/en/methods/page/Param.md create mode 100644 content/en/methods/page/Params.md create mode 100644 content/en/methods/page/Parent.md create mode 100644 content/en/methods/page/Permalink.md create mode 100644 content/en/methods/page/Plain.md create mode 100644 content/en/methods/page/PlainWords.md create mode 100644 content/en/methods/page/Prev.md create mode 100644 content/en/methods/page/PrevInSection.md create mode 100644 content/en/methods/page/PublishDate.md create mode 100644 content/en/methods/page/RawContent.md create mode 100644 content/en/methods/page/ReadingTime.md create mode 100644 content/en/methods/page/Ref.md create mode 100644 content/en/methods/page/RegularPages.md create mode 100644 content/en/methods/page/RegularPagesRecursive.md create mode 100644 content/en/methods/page/RelPermalink.md create mode 100644 content/en/methods/page/RelRef.md create mode 100644 content/en/methods/page/Render.md create mode 100644 content/en/methods/page/RenderShortcodes.md create mode 100644 content/en/methods/page/RenderString.md create mode 100644 content/en/methods/page/Resources.md create mode 100644 content/en/methods/page/Scratch.md create mode 100644 content/en/methods/page/Section.md create mode 100644 content/en/methods/page/Sections.md create mode 100644 content/en/methods/page/Site.md create mode 100644 content/en/methods/page/Sitemap.md create mode 100644 content/en/methods/page/Sites.md create mode 100644 content/en/methods/page/Slug.md create mode 100644 content/en/methods/page/Store.md create mode 100644 content/en/methods/page/Summary.md create mode 100644 content/en/methods/page/TableOfContents.md create mode 100644 content/en/methods/page/Title.md create mode 100644 content/en/methods/page/TranslationKey.md create mode 100644 content/en/methods/page/Translations.md create mode 100644 content/en/methods/page/Truncated.md create mode 100644 content/en/methods/page/Type.md create mode 100644 content/en/methods/page/Weight.md create mode 100644 content/en/methods/page/WordCount.md create mode 100644 content/en/methods/page/_common/_index.md create mode 100644 content/en/methods/page/_common/definition-of-section.md create mode 100644 content/en/methods/page/_common/output-format-definition.md create mode 100644 content/en/methods/page/_common/output-format-methods.md create mode 100644 content/en/methods/page/_common/scratch-methods.md create mode 100644 content/en/methods/page/_index.md create mode 100644 content/en/methods/pages/ByDate.md create mode 100644 content/en/methods/pages/ByExpiryDate.md create mode 100644 content/en/methods/pages/ByLanguage.md create mode 100644 content/en/methods/pages/ByLastmod.md create mode 100644 content/en/methods/pages/ByLength.md create mode 100644 content/en/methods/pages/ByLinkTitle.md create mode 100644 content/en/methods/pages/ByParam.md create mode 100644 content/en/methods/pages/ByPublishDate.md create mode 100644 content/en/methods/pages/ByTitle.md create mode 100644 content/en/methods/pages/ByWeight.md create mode 100644 content/en/methods/pages/GroupBy.md create mode 100644 content/en/methods/pages/GroupByDate.md create mode 100644 content/en/methods/pages/GroupByExpiryDate.md create mode 100644 content/en/methods/pages/GroupByLastmod.md create mode 100644 content/en/methods/pages/GroupByParam.md create mode 100644 content/en/methods/pages/GroupByParamDate.md create mode 100644 content/en/methods/pages/GroupByPublishDate.md create mode 100644 content/en/methods/pages/Len.md create mode 100644 content/en/methods/pages/Limit.md create mode 100644 content/en/methods/pages/Next.md create mode 100644 content/en/methods/pages/Prev.md create mode 100644 content/en/methods/pages/Related.md create mode 100644 content/en/methods/pages/Reverse.md create mode 100644 content/en/methods/pages/_common/_index.md create mode 100644 content/en/methods/pages/_common/group-sort-order.md create mode 100644 content/en/methods/pages/_index.md create mode 100644 content/en/methods/resource/Colors.md create mode 100644 content/en/methods/resource/Content.md create mode 100644 content/en/methods/resource/Crop.md create mode 100644 content/en/methods/resource/Data.md create mode 100644 content/en/methods/resource/Err.md create mode 100644 content/en/methods/resource/Exif.md create mode 100644 content/en/methods/resource/Fill.md create mode 100644 content/en/methods/resource/Filter.md create mode 100644 content/en/methods/resource/Fit.md create mode 100644 content/en/methods/resource/Height.md create mode 100644 content/en/methods/resource/Key.md create mode 100644 content/en/methods/resource/MediaType.md create mode 100644 content/en/methods/resource/Name.md create mode 100644 content/en/methods/resource/Params.md create mode 100644 content/en/methods/resource/Permalink.md create mode 100644 content/en/methods/resource/Process.md create mode 100644 content/en/methods/resource/Publish.md create mode 100644 content/en/methods/resource/RelPermalink.md create mode 100644 content/en/methods/resource/Resize.md create mode 100644 content/en/methods/resource/ResourceType.md create mode 100644 content/en/methods/resource/Title.md create mode 100644 content/en/methods/resource/Width.md create mode 100644 content/en/methods/resource/_common/_index.md create mode 100644 content/en/methods/resource/_common/global-page-remote-resources.md create mode 100644 content/en/methods/resource/_common/processing-spec.md create mode 100644 content/en/methods/resource/_index.md create mode 100644 content/en/methods/shortcode/Get.md create mode 100644 content/en/methods/shortcode/Inner.md create mode 100644 content/en/methods/shortcode/InnerDeindent.md create mode 100644 content/en/methods/shortcode/IsNamedParams.md create mode 100644 content/en/methods/shortcode/Name.md create mode 100644 content/en/methods/shortcode/Ordinal.md create mode 100644 content/en/methods/shortcode/Page.md create mode 100644 content/en/methods/shortcode/Params.md create mode 100644 content/en/methods/shortcode/Parent.md create mode 100644 content/en/methods/shortcode/Position.md create mode 100644 content/en/methods/shortcode/Scratch.md create mode 100644 content/en/methods/shortcode/_index.md create mode 100644 content/en/methods/site/AllPages.md create mode 100644 content/en/methods/site/BaseURL.md create mode 100644 content/en/methods/site/BuildDrafts.md create mode 100644 content/en/methods/site/Config.md create mode 100644 content/en/methods/site/Copyright.md create mode 100644 content/en/methods/site/Data.md create mode 100644 content/en/methods/site/GetPage.md create mode 100644 content/en/methods/site/Home.md create mode 100644 content/en/methods/site/IsMultiLingual.md create mode 100644 content/en/methods/site/Language.md create mode 100644 content/en/methods/site/LanguagePrefix.md create mode 100644 content/en/methods/site/Languages.md create mode 100644 content/en/methods/site/LastChange.md create mode 100644 content/en/methods/site/MainSections.md create mode 100644 content/en/methods/site/Menus.md create mode 100644 content/en/methods/site/Pages.md create mode 100644 content/en/methods/site/Params.md create mode 100644 content/en/methods/site/RegularPages.md create mode 100644 content/en/methods/site/Sections.md create mode 100644 content/en/methods/site/Sites.md create mode 100644 content/en/methods/site/Taxonomies.md create mode 100644 content/en/methods/site/Title.md create mode 100644 content/en/methods/site/_index.md create mode 100644 content/en/methods/taxonomy/Alphabetical.md create mode 100644 content/en/methods/taxonomy/ByCount.md create mode 100644 content/en/methods/taxonomy/Count.md create mode 100644 content/en/methods/taxonomy/Get.md create mode 100644 content/en/methods/taxonomy/_common/_index.md create mode 100644 content/en/methods/taxonomy/_common/get-a-taxonomy-object.md create mode 100644 content/en/methods/taxonomy/_common/ordered-taxonomy-element-methods.md create mode 100644 content/en/methods/taxonomy/_index.md create mode 100644 content/en/methods/time/Add.md rename content/en/{functions => methods/time}/AddDate.md (88%) create mode 100644 content/en/methods/time/After.md create mode 100644 content/en/methods/time/Before.md create mode 100644 content/en/methods/time/Day.md create mode 100644 content/en/methods/time/Equal.md rename content/en/{functions => methods/time}/Format.md (87%) create mode 100644 content/en/methods/time/Hour.md create mode 100644 content/en/methods/time/IsDST.md create mode 100644 content/en/methods/time/IsZero.md create mode 100644 content/en/methods/time/Local.md create mode 100644 content/en/methods/time/Minute.md create mode 100644 content/en/methods/time/Month.md create mode 100644 content/en/methods/time/Nanosecond.md create mode 100644 content/en/methods/time/Second.md create mode 100644 content/en/methods/time/Sub.md create mode 100644 content/en/methods/time/UTC.md create mode 100644 content/en/methods/time/Unix.md create mode 100644 content/en/methods/time/UnixMicro.md create mode 100644 content/en/methods/time/UnixMilli.md create mode 100644 content/en/methods/time/UnixNano.md create mode 100644 content/en/methods/time/Weekday.md create mode 100644 content/en/methods/time/Year.md create mode 100644 content/en/methods/time/YearDay.md create mode 100644 content/en/methods/time/_index.md create mode 100644 content/en/variables/_common/_index.md create mode 100644 content/en/variables/_common/consistent-terminology.md create mode 100644 content/en/variables/file.md delete mode 100644 content/en/variables/files.md rename content/en/variables/{menus.md => menu-entry.md} (81%) create mode 100644 content/en/variables/shortcode.md delete mode 100644 content/en/variables/shortcodes.md delete mode 100644 content/en/variables/sitemap.md create mode 100644 data/page_filters.yaml delete mode 100644 data/titles.toml create mode 100644 layouts/_default/_markup/render-link.html create mode 100644 layouts/_default/single.html create mode 100644 layouts/partials/docs/functions-return-type.html create mode 100644 layouts/partials/pagelayout.html create mode 100644 layouts/partials/right-sidebar.html delete mode 100644 layouts/shortcodes/getcontent.html create mode 100644 layouts/shortcodes/img.html create mode 100644 layouts/shortcodes/include.html create mode 100644 layouts/shortcodes/list-pages-in-section.html diff --git a/.cspell.json b/.cspell.json index 0958b76450a..13865a83c7c 100644 --- a/.cspell.json +++ b/.cspell.json @@ -335,7 +335,7 @@ "مدونتي" ], "language": "en,en-US,de,fr", - "allowCompoundWords": true, + "allowCompoundWords": false, "files": [ "**/*.md" ], @@ -354,7 +354,8 @@ "**/node_modules/**", "*.min.*", "**/news/*", - "**/showcase/*" + "**/showcase/*", + "**/content-management/emoji-shortcodes.md" ], "useGitignore": true, "enabled": true diff --git a/LICENSE.md b/LICENSE.md index b62a9b5ff78..b09cd7856d5 100644 --- a/LICENSE.md +++ b/LICENSE.md @@ -1,194 +1,201 @@ Apache License -============== - -_Version 2.0, January 2004_ -_<>_ - -### Terms and Conditions for use, reproduction, and distribution - -#### 1. Definitions - -“License” shall mean the terms and conditions for use, reproduction, and -distribution as defined by Sections 1 through 9 of this document. - -“Licensor” shall mean the copyright owner or entity authorized by the copyright -owner that is granting the License. - -“Legal Entity” shall mean the union of the acting entity and all other entities -that control, are controlled by, or are under common control with that entity. -For the purposes of this definition, “control” means **(i)** the power, direct or -indirect, to cause the direction or management of such entity, whether by -contract or otherwise, or **(ii)** ownership of fifty percent (50%) or more of the -outstanding shares, or **(iii)** beneficial ownership of such entity. - -“You” (or “Your”) shall mean an individual or Legal Entity exercising -permissions granted by this License. - -“Source” form shall mean the preferred form for making modifications, including -but not limited to software source code, documentation source, and configuration -files. - -“Object” form shall mean any form resulting from mechanical transformation or -translation of a Source form, including but not limited to compiled object code, -generated documentation, and conversions to other media types. - -“Work” shall mean the work of authorship, whether in Source or Object form, made -available under the License, as indicated by a copyright notice that is included -in or attached to the work (an example is provided in the Appendix below). - -“Derivative Works” shall mean any work, whether in Source or Object form, that -is based on (or derived from) the Work and for which the editorial revisions, -annotations, elaborations, or other modifications represent, as a whole, an -original work of authorship. For the purposes of this License, Derivative Works -shall not include works that remain separable from, or merely link (or bind by -name) to the interfaces of, the Work and Derivative Works thereof. - -“Contribution” shall mean any work of authorship, including the original version -of the Work and any modifications or additions to that Work or Derivative Works -thereof, that is intentionally submitted to Licensor for inclusion in the Work -by the copyright owner or by an individual or Legal Entity authorized to submit -on behalf of the copyright owner. For the purposes of this definition, -“submitted” means any form of electronic, verbal, or written communication sent -to the Licensor or its representatives, including but not limited to -communication on electronic mailing lists, source code control systems, and -issue tracking systems that are managed by, or on behalf of, the Licensor for -the purpose of discussing and improving the Work, but excluding communication -that is conspicuously marked or otherwise designated in writing by the copyright -owner as “Not a Contribution.” - -“Contributor” shall mean Licensor and any individual or Legal Entity on behalf -of whom a Contribution has been received by Licensor and subsequently -incorporated within the Work. - -#### 2. Grant of Copyright License - -Subject to the terms and conditions of this License, each Contributor hereby -grants to You a perpetual, worldwide, non-exclusive, no-charge, royalty-free, -irrevocable copyright license to reproduce, prepare Derivative Works of, -publicly display, publicly perform, sublicense, and distribute the Work and such -Derivative Works in Source or Object form. - -#### 3. Grant of Patent License - -Subject to the terms and conditions of this License, each Contributor hereby -grants to You a perpetual, worldwide, non-exclusive, no-charge, royalty-free, -irrevocable (except as stated in this section) patent license to make, have -made, use, offer to sell, sell, import, and otherwise transfer the Work, where -such license applies only to those patent claims licensable by such Contributor -that are necessarily infringed by their Contribution(s) alone or by combination -of their Contribution(s) with the Work to which such Contribution(s) was -submitted. If You institute patent litigation against any entity (including a -cross-claim or counterclaim in a lawsuit) alleging that the Work or a -Contribution incorporated within the Work constitutes direct or contributory -patent infringement, then any patent licenses granted to You under this License -for that Work shall terminate as of the date such litigation is filed. - -#### 4. Redistribution - -You may reproduce and distribute copies of the Work or Derivative Works thereof -in any medium, with or without modifications, and in Source or Object form, -provided that You meet the following conditions: - -* **(a)** You must give any other recipients of the Work or Derivative Works a copy of -this License; and -* **(b)** You must cause any modified files to carry prominent notices stating that You -changed the files; and -* **(c)** You must retain, in the Source form of any Derivative Works that You distribute, -all copyright, patent, trademark, and attribution notices from the Source form -of the Work, excluding those notices that do not pertain to any part of the -Derivative Works; and -* **(d)** If the Work includes a “NOTICE” text file as part of its distribution, then any -Derivative Works that You distribute must include a readable copy of the -attribution notices contained within such NOTICE file, excluding those notices -that do not pertain to any part of the Derivative Works, in at least one of the -following places: within a NOTICE text file distributed as part of the -Derivative Works; within the Source form or documentation, if provided along -with the Derivative Works; or, within a display generated by the Derivative -Works, if and wherever such third-party notices normally appear. The contents of -the NOTICE file are for informational purposes only and do not modify the -License. You may add Your own attribution notices within Derivative Works that -You distribute, alongside or as an addendum to the NOTICE text from the Work, -provided that such additional attribution notices cannot be construed as -modifying the License. - -You may add Your own copyright statement to Your modifications and may provide -additional or different license terms and conditions for use, reproduction, or -distribution of Your modifications, or for any such Derivative Works as a whole, -provided Your use, reproduction, and distribution of the Work otherwise complies -with the conditions stated in this License. - -#### 5. Submission of Contributions - -Unless You explicitly state otherwise, any Contribution intentionally submitted -for inclusion in the Work by You to the Licensor shall be under the terms and -conditions of this License, without any additional terms or conditions. -Notwithstanding the above, nothing herein shall supersede or modify the terms of -any separate license agreement you may have executed with Licensor regarding -such Contributions. - -#### 6. Trademarks - -This License does not grant permission to use the trade names, trademarks, -service marks, or product names of the Licensor, except as required for -reasonable and customary use in describing the origin of the Work and -reproducing the content of the NOTICE file. - -#### 7. Disclaimer of Warranty - -Unless required by applicable law or agreed to in writing, Licensor provides the -Work (and each Contributor provides its Contributions) on an “AS IS” BASIS, -WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied, -including, without limitation, any warranties or conditions of TITLE, -NON-INFRINGEMENT, MERCHANTABILITY, or FITNESS FOR A PARTICULAR PURPOSE. You are -solely responsible for determining the appropriateness of using or -redistributing the Work and assume any risks associated with Your exercise of -permissions under this License. - -#### 8. Limitation of Liability - -In no event and under no legal theory, whether in tort (including negligence), -contract, or otherwise, unless required by applicable law (such as deliberate -and grossly negligent acts) or agreed to in writing, shall any Contributor be -liable to You for damages, including any direct, indirect, special, incidental, -or consequential damages of any character arising as a result of this License or -out of the use or inability to use the Work (including but not limited to -damages for loss of goodwill, work stoppage, computer failure or malfunction, or -any and all other commercial damages or losses), even if such Contributor has -been advised of the possibility of such damages. - -#### 9. Accepting Warranty or Additional Liability - -While redistributing the Work or Derivative Works thereof, You may choose to -offer, and charge a fee for, acceptance of support, warranty, indemnity, or -other liability obligations and/or rights consistent with this License. However, -in accepting such obligations, You may act only on Your own behalf and on Your -sole responsibility, not on behalf of any other Contributor, and only if You -agree to indemnify, defend, and hold each Contributor harmless for any liability -incurred by, or claims asserted against, such Contributor by reason of your -accepting any such warranty or additional liability. - -_END OF TERMS AND CONDITIONS_ - -### APPENDIX: How to apply the Apache License to your work - -To apply the Apache License to your work, attach the following boilerplate -notice, with the fields enclosed by brackets `[]` replaced with your own -identifying information. (Don't include the brackets!) The text should be -enclosed in the appropriate comment syntax for the file format. We also -recommend that a file or class name and description of purpose be included on -the same “printed page” as the copyright notice for easier identification within -third-party archives. - - Copyright [yyyy] [name of copyright owner] - - Licensed under the Apache License, Version 2.0 (the "License"); - you may not use this file except in compliance with the License. - You may obtain a copy of the License at - - http://www.apache.org/licenses/LICENSE-2.0 - - Unless required by applicable law or agreed to in writing, software - distributed under the License is distributed on an "AS IS" BASIS, - WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. - See the License for the specific language governing permissions and - limitations under the License. + Version 2.0, January 2004 + http://www.apache.org/licenses/ + + TERMS AND CONDITIONS FOR USE, REPRODUCTION, AND DISTRIBUTION + + 1. Definitions. + + "License" shall mean the terms and conditions for use, reproduction, + and distribution as defined by Sections 1 through 9 of this document. + + "Licensor" shall mean the copyright owner or entity authorized by + the copyright owner that is granting the License. + + "Legal Entity" shall mean the union of the acting entity and all + other entities that control, are controlled by, or are under common + control with that entity. For the purposes of this definition, + "control" means (i) the power, direct or indirect, to cause the + direction or management of such entity, whether by contract or + otherwise, or (ii) ownership of fifty percent (50%) or more of the + outstanding shares, or (iii) beneficial ownership of such entity. + + "You" (or "Your") shall mean an individual or Legal Entity + exercising permissions granted by this License. + + "Source" form shall mean the preferred form for making modifications, + including but not limited to software source code, documentation + source, and configuration files. + + "Object" form shall mean any form resulting from mechanical + transformation or translation of a Source form, including but + not limited to compiled object code, generated documentation, + and conversions to other media types. + + "Work" shall mean the work of authorship, whether in Source or + Object form, made available under the License, as indicated by a + copyright notice that is included in or attached to the work + (an example is provided in the Appendix below). + + "Derivative Works" shall mean any work, whether in Source or Object + form, that is based on (or derived from) the Work and for which the + editorial revisions, annotations, elaborations, or other modifications + represent, as a whole, an original work of authorship. For the purposes + of this License, Derivative Works shall not include works that remain + separable from, or merely link (or bind by name) to the interfaces of, + the Work and Derivative Works thereof. + + "Contribution" shall mean any work of authorship, including + the original version of the Work and any modifications or additions + to that Work or Derivative Works thereof, that is intentionally + submitted to Licensor for inclusion in the Work by the copyright owner + or by an individual or Legal Entity authorized to submit on behalf of + the copyright owner. For the purposes of this definition, "submitted" + means any form of electronic, verbal, or written communication sent + to the Licensor or its representatives, including but not limited to + communication on electronic mailing lists, source code control systems, + and issue tracking systems that are managed by, or on behalf of, the + Licensor for the purpose of discussing and improving the Work, but + excluding communication that is conspicuously marked or otherwise + designated in writing by the copyright owner as "Not a Contribution." + + "Contributor" shall mean Licensor and any individual or Legal Entity + on behalf of whom a Contribution has been received by Licensor and + subsequently incorporated within the Work. + + 2. Grant of Copyright License. Subject to the terms and conditions of + this License, each Contributor hereby grants to You a perpetual, + worldwide, non-exclusive, no-charge, royalty-free, irrevocable + copyright license to reproduce, prepare Derivative Works of, + publicly display, publicly perform, sublicense, and distribute the + Work and such Derivative Works in Source or Object form. + + 3. Grant of Patent License. Subject to the terms and conditions of + this License, each Contributor hereby grants to You a perpetual, + worldwide, non-exclusive, no-charge, royalty-free, irrevocable + (except as stated in this section) patent license to make, have made, + use, offer to sell, sell, import, and otherwise transfer the Work, + where such license applies only to those patent claims licensable + by such Contributor that are necessarily infringed by their + Contribution(s) alone or by combination of their Contribution(s) + with the Work to which such Contribution(s) was submitted. If You + institute patent litigation against any entity (including a + cross-claim or counterclaim in a lawsuit) alleging that the Work + or a Contribution incorporated within the Work constitutes direct + or contributory patent infringement, then any patent licenses + granted to You under this License for that Work shall terminate + as of the date such litigation is filed. + + 4. Redistribution. You may reproduce and distribute copies of the + Work or Derivative Works thereof in any medium, with or without + modifications, and in Source or Object form, provided that You + meet the following conditions: + + (a) You must give any other recipients of the Work or + Derivative Works a copy of this License; and + + (b) You must cause any modified files to carry prominent notices + stating that You changed the files; and + + (c) You must retain, in the Source form of any Derivative Works + that You distribute, all copyright, patent, trademark, and + attribution notices from the Source form of the Work, + excluding those notices that do not pertain to any part of + the Derivative Works; and + + (d) If the Work includes a "NOTICE" text file as part of its + distribution, then any Derivative Works that You distribute must + include a readable copy of the attribution notices contained + within such NOTICE file, excluding those notices that do not + pertain to any part of the Derivative Works, in at least one + of the following places: within a NOTICE text file distributed + as part of the Derivative Works; within the Source form or + documentation, if provided along with the Derivative Works; or, + within a display generated by the Derivative Works, if and + wherever such third-party notices normally appear. The contents + of the NOTICE file are for informational purposes only and + do not modify the License. You may add Your own attribution + notices within Derivative Works that You distribute, alongside + or as an addendum to the NOTICE text from the Work, provided + that such additional attribution notices cannot be construed + as modifying the License. + + You may add Your own copyright statement to Your modifications and + may provide additional or different license terms and conditions + for use, reproduction, or distribution of Your modifications, or + for any such Derivative Works as a whole, provided Your use, + reproduction, and distribution of the Work otherwise complies with + the conditions stated in this License. + + 5. Submission of Contributions. Unless You explicitly state otherwise, + any Contribution intentionally submitted for inclusion in the Work + by You to the Licensor shall be under the terms and conditions of + this License, without any additional terms or conditions. + Notwithstanding the above, nothing herein shall supersede or modify + the terms of any separate license agreement you may have executed + with Licensor regarding such Contributions. + + 6. Trademarks. This License does not grant permission to use the trade + names, trademarks, service marks, or product names of the Licensor, + except as required for reasonable and customary use in describing the + origin of the Work and reproducing the content of the NOTICE file. + + 7. Disclaimer of Warranty. Unless required by applicable law or + agreed to in writing, Licensor provides the Work (and each + Contributor provides its Contributions) on an "AS IS" BASIS, + WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or + implied, including, without limitation, any warranties or conditions + of TITLE, NON-INFRINGEMENT, MERCHANTABILITY, or FITNESS FOR A + PARTICULAR PURPOSE. You are solely responsible for determining the + appropriateness of using or redistributing the Work and assume any + risks associated with Your exercise of permissions under this License. + + 8. Limitation of Liability. In no event and under no legal theory, + whether in tort (including negligence), contract, or otherwise, + unless required by applicable law (such as deliberate and grossly + negligent acts) or agreed to in writing, shall any Contributor be + liable to You for damages, including any direct, indirect, special, + incidental, or consequential damages of any character arising as a + result of this License or out of the use or inability to use the + Work (including but not limited to damages for loss of goodwill, + work stoppage, computer failure or malfunction, or any and all + other commercial damages or losses), even if such Contributor + has been advised of the possibility of such damages. + + 9. Accepting Warranty or Additional Liability. While redistributing + the Work or Derivative Works thereof, You may choose to offer, + and charge a fee for, acceptance of support, warranty, indemnity, + or other liability obligations and/or rights consistent with this + License. However, in accepting such obligations, You may act only + on Your own behalf and on Your sole responsibility, not on behalf + of any other Contributor, and only if You agree to indemnify, + defend, and hold each Contributor harmless for any liability + incurred by, or claims asserted against, such Contributor by reason + of your accepting any such warranty or additional liability. + + END OF TERMS AND CONDITIONS + + APPENDIX: How to apply the Apache License to your work. + + To apply the Apache License to your work, attach the following + boilerplate notice, with the fields enclosed by brackets "[]" + replaced with your own identifying information. (Don't include + the brackets!) The text should be enclosed in the appropriate + comment syntax for the file format. We also recommend that a + file or class name and description of purpose be included on the + same "printed page" as the copyright notice for easier + identification within third-party archives. + + Copyright [yyyy] [name of copyright owner] + + Licensed under the Apache License, Version 2.0 (the "License"); + you may not use this file except in compliance with the License. + You may obtain a copy of the License at + + http://www.apache.org/licenses/LICENSE-2.0 + + Unless required by applicable law or agreed to in writing, software + distributed under the License is distributed on an "AS IS" BASIS, + WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + See the License for the specific language governing permissions and + limitations under the License. diff --git a/README.md b/README.md index 730ad5fc81b..bf5d9402c19 100644 --- a/README.md +++ b/README.md @@ -19,12 +19,11 @@ Spelling fixes are most welcomed, and if you want to contribute longer sections * For example, try to find short snippets that teaches people about the concept. If the example is also useful as-is (copy and paste), then great. Don't list long and similar examples just so people can use them on their sites. * Hugo has users from all over the world, so easy to understand and [simple English](https://simple.wikipedia.org/wiki/Basic_English) is good. - ## Edit the theme If you want to do docs-related theme changes, the simplest way is to have both `hugoDocs` and `gohugoioTheme` cloned as sibling directories, and then run: -``` +```sh HUGO_MODULE_WORKSPACE=hugo.work hugo server --ignoreVendorPaths "**" ``` @@ -37,7 +36,7 @@ HUGO_MODULE_WORKSPACE=hugo.work hugo server --ignoreVendorPaths "**" To view the documentation site locally, you need to clone this repository: -```bash +```sh git clone https://github.com/gohugoio/hugoDocs.git ``` @@ -45,7 +44,7 @@ Also note that the documentation version for a given version of Hugo can also be Then to view the docs in your browser, run Hugo and open up the link: -```bash +```sh ▶ hugo server Started building sites ... diff --git a/archetypes/functions.md b/archetypes/functions.md index cb0e7c930ef..44a2a5635c2 100644 --- a/archetypes/functions.md +++ b/archetypes/functions.md @@ -1,14 +1,11 @@ --- title: {{ replace .File.ContentBaseName "-" " " | title }} description: -categories: [functions] +categories: [] keywords: [] -menu: - docs: - parent: functions -function: +action: aliases: [] + related: [] returnType: signatures: [] -relatedFunctions: [] --- diff --git a/archetypes/methods.md b/archetypes/methods.md new file mode 100644 index 00000000000..2c415547d2d --- /dev/null +++ b/archetypes/methods.md @@ -0,0 +1,10 @@ +--- +title: {{ replace .File.ContentBaseName "-" " " | title }} +description: +categories: [] +keywords: [] +action: + related: [] + returnType: + signatures: [] +--- diff --git a/archetypes/showcase/bio.md b/archetypes/showcase/bio.md index 2443c2f35dc..5d17089786f 100644 --- a/archetypes/showcase/bio.md +++ b/archetypes/showcase/bio.md @@ -3,6 +3,5 @@ Add some **general info** about {{ replace .Name "-" " " | title }} here. The site is built by: -* [Person 1](https://example.com) -* [Person 1](https://example.com) - +* [Person 1](https://example.org) +* [Person 1](https://example.org) diff --git a/assets/images/examples/zion-national-park-grayscale.jpg b/assets/images/examples/zion-national-park-grayscale.jpg new file mode 100644 index 0000000000000000000000000000000000000000..908bd88c6b95706cb4d77cf14628821e034536d2 GIT binary patch literal 42537 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 F{U6E1vTpzY literal 0 HcmV?d00001 diff --git a/assets/images/examples/zion-national-park.jpg b/assets/images/examples/zion-national-park.jpg new file mode 100644 index 0000000000000000000000000000000000000000..7980abccb8c8538320c66afbe57f17225ea038d8 GIT binary patch literal 44553 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 BSGNEF literal 0 HcmV?d00001 diff --git a/assets/images/logos/logo-512x512.png b/assets/images/logos/logo-512x512.png new file mode 100644 index 0000000000000000000000000000000000000000..76d463600e7b79b94799f775417836a59ff977dc GIT binary patch literal 73651 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

j*cBgxACBE(u}mHT(xPj4mGo7#XMy3b;( z{0B6DdjIqjpbBetyPAJ^u$HE}B9vM!>_@PGHjoY0CUqWVQoNkhSJ$K$CPkfsgaZf` z1e{3_%wZio)L^m`AcMtdeqn<2K^Yl5&yVIJmVQ z0x))ZPW6m=N>g-Fr-Joa8NEdX+hRhlh?{+&!!A%gpogf%!UPl-_$F|k3~RmpQMI^fN5#*JzJuS8^V5#DoSg5pqB$%% zVz45!EY-HBmq-er%}hIbEEE213H&EW@RNCWsmzUWT-n7FuSZ_^>6yq`a8}$AnqstZ zIE-oYr<{kq<(YvVczcMm^8SAyD6+Xy1gs09_nRl|+O@CZFYCgp_MaO1PiHymA|pC} z;*#LCJH|(^AD2HF{Vnd$$7>eG9&u>_+js%vPHTkpM4(#c@amgydk=oy6-TG}N<=>T z*KgqUMi{`JAfRGp0qMFkmN6m8vtzW)*;Y>^0)k}!q>MK@v8=Ry3br!B>(O)M%_MeB zU1iL$=NVX$k(E%0UKn$9WcUUf54geTq!kR-6WhpQ5*NE+^si*`3ItL9oJ#Ma1TNl; zWm`0kE*bZvCHQk(_zxh%C6CZIdNbH6+B6uONyBLDm;*{Rp9JHyR3C$JciA0>QZQ^c zaguBADR7nO`0hnX^7;Zql}JS_FQ6|h7{Dq7<9#c{DAo%o^%=cJ?MS^{T@IiN>tmfj z0o$AwKoDFGTCj)ZD;>j(S1Pi4?ssB?cP%p#GyPiod~ed^Fdr@L*7S@al+R5;RO^o@ z|Jvgd?{+>cLfn5j3j!8?b8O#oS-ao&)7@orAx;c=YwKxwVY>U09VSRIxwOE2{8(`0 zh0fK~e;La_LkSc%!KfIM3v7F;5Uhz)bQ#-K*IJF8qQF<(%Ns6y=p`rPjLjr+V_gzu zJA;WH-HHv349cYVHU3g=NMr($?Crz0Y*N`HbnEw+o2OX;D-*5a3$8128R$6rNpPEev)7XB({I!Duo)t+b>*iqtkCj-yYn@RHt5@YlB0O8B$eKOxa;3+Cxv zDI^PaI)>KD_FuO#0o&+9bHGZh0P@CS#gKdda6AjDYt-SeX!0r%c>(&iKG$z?D{#p) zc>24Hj3vAEyx^bjEeD(RG^VZ|(#aM?esRPBDO(BMD!bVjCD0%SYh|`NIXMlkGG|8k zm-nMy;U=3%N(T08+pZ8Na1Zed;u<(TyDa}0kT`|$KOY)88q+5^3Im~Xjghz7@e@us z-}S0_71T4?(fFv|- z`<<{Hz{-DNEh)cof>GjdzE`T--+Crtv`s(|z&#)mLc?ZiAg5H&Jc_*}n3wH~ z?#`N^-eL|!rLS{EMKXY{EtAm4BS4?uJWFOtypx)Hzg7y(u#N^)aa^d2dKCW zI!M|)WSNhXX?-5h1`Ljv`YN9w5O36lU5hEsYzB`|PM^ zFr_^g9LTa|aQ5}!buhj0b8Ngho)R2L`4CcbdK7h>@!GK_SegBUS;xv?4vG)8o#;pO zZ+wH1gleWPFb|DvFEno%Ibnc}5QTwQ?y8S4jEoy2K$nXJf*MUVtdC!7&x^CC%ggR$ zgjcQyUzhe;FhV9W1rYNl>$?g@(tAUGhm3V%9XkjDj8vmUSo{Um71Tr_0_}~+O>y>f ze=44wpodgO4zs7A1x3Hti0MUPnUB{0F=NH4Hh$&EG=)t~>K%XYf>y%%(nujWsP{E) zpL7Fw`ei&U7AjGhwU~bNJ`ESJ(N-Bh?%xB(E*4{dY*Man0^zZ=2NMaKO5Q3-U~JhI z2)U|zpt~om@YBejw1xune{54$W}t@IF+t2C+^#XK9*D_Md?q}>_V$ep6sT}HH2EMj-y zCJO{D*=0O)org?9ojr9pXAG=EVC#7b-4>h^|M|0Pkl4|P@) z^ORNsXotzdzjqIgW!rPPeQ_3U^T4Se)k2j05YMzip~I?dbq-B{{n;Kd$s2WK+(!+^ z9-ka+KMT2S`S?zlKP{J?ZC^vN6g1Z~Kf4mDqd>cZUNIsuFNT6gpKi#O1AUarb6}KD zOftnP46TaxY%^mgjlsJ(+Tadb3&4auiFuTd`-~&vN8U_!V%-lJM`@7P-J9+MHo(*k z_G?yLSg)`0{K0YVp!9wq&OPGdSf%;o20<*;Kn5s$!ptO6s46(>7~f}h^J8`PL?^N% zh&@FmSpKWX|BFuiyJkFJul(!Q;p8ykJ+ka`BfS?c8`;jmkqeV1U7OP)SRa%7z~>5I zRoGm~S!@mwR~vIdh!A;X82~khKTE4N4TSsDk_aYSEs?XlDDP7CiXFg zwco8rz+MpY{$%F|kJ43V*re}cc(rwcMqPFnDRl8D+Ntc4B87V|3)6Z*rf-_VQ3Zo4+Au%af=~|!7 zsVk(LMb;Sa?_smcHg!@Vu|ZMw*+s~@!iq?nvn^W$%S`Q_omlhMZ{_NpD*%gbFT z_W3%H5lF~E(`MbiG=8@~Ojd?@#D1-{8I_I>Mjf^W+%7Jyt$-I-&}J}ydf6a6gS}32 z$_4AN#{U0aR1^nnAoSUReMbpk*%j;!S>XrsELdnL-Q1U6rmzImhJ6N~{#MHZ4C%;% z6-W6cS9c@0m34Y zzL$>xo0OSet{mo_-tfOABfD@fRM~3Q>_#DlS-AiHXEcp(uV7tvxALd>+1TgO#$-?I z`8CZM#+QTaS>!x90_85<^+>sW?sUR~7>&YRS^iGQ)RtMor{=YKlOVe9n2dwt7ZW&YibbMU}b zku33VO@d zft-23!5B;+a%u% zl%G*TRyHKk=j^w3$d5J~KajSE-?wqpgq}UjmKnBLuqp%mzP~|?4hnHbP`xR6cxfR2 zK7r_Tu$S)B=(-&P+|ghO<wc$FSc`rAlc^4l?L1b)-D(Vhy9JJfKG7QRCX!qVJA*GI|iX*wEenN?A6jaeEazWq0$HGCTsN z5|Rvv)-vC2XPg1>c$b6loqMdhb>h1voo@Ej)bL%ha>dngZ%O=8-S~7)l5vPfi)F6s z7-yWFF_tZEge8zl-=Y#Dpv(#DO|UTe*1$KJPLab72W}Id68xL~RH+TvDl!kWns0+u zBh-K*tIQ*wNmx2tlV7q5JGbGPAhAk3%1Q%m3;74Rvv1a# zmvbhr8u2SI88e5S4A7BeM_L5P$cVSC8``vQdbT%|DPz-TYBEOgxB;`eYk`D;Y{1Sr z*KV%5+k}oyB9EVXPgVxegtw;+#y5gGzjd3?tU}%(ICxR83Ig*jAlK^OzAZ~uABLh5 z#wYZ{{ZlH#&&NOnd2e9m5saz zp zD8Bdm;Dc4G5}z|KtdB1FeQ*v6^LJxT%wsuHM6Ke83VUPAGZq%|Pelz?R9GXfV#6w* z>FK!zW+b1RM+2)NSS9ubwQ3$41Jn6GDg!Q*&Hfl2`?7OGo8x`A_-o?c`~6#!@gFJw zV~yy$9MV73z9rS1aqQ*36MIj*KlXC3)=MLf`+fJ#ueSqaj_Xv(JR0HBZ+?BOgX4GK z!=vwYyqtD8y7kh^bM(*};wK%5s3H_0Qo`(ZJ{J~-*~`LVLzmWmzefE=grHNOGI-R# z`9svDgWHjQV;Nxd3w4;54-T#N@Ha5oNe8_CWV<2c$uSi@ZZLr;_#qwHGc~3%QZ-@D za!vQmLBl38Xg<)!uo(!>RsqT;4sb3TkmVR=BQS}auCX*_ukw@y3+D}qh9Z5JA=~Na zPQyC!puQ1QKI1o>k9nAscaxaBKFuA4dI zeB(ucfhoImzPE~?Gus_~2Ct6(VX>$(M&8}MgFk`g=c0A@WNgG)c%I(^MGVoLu);;> zHe}7pHH|Zn&+Z*#gn88rNrpCP)ezd=GIQQ~zO>AlerSMuf14)G63jb2AbE93MR-cf z<3S>Yx1VNmAcWh_a?745L5u2r{~jt{o};$Z6&z7QO#aFUZyTUJKGo3tnP#;km{*=` z8|Zo7pZ%2;K1Y6QmfsS>ZN&ZxZAsCo`;2!8 za&VH;`WpX6D*6wp^E|9G*DD(IH|h2CnO4V$?|JgL%lwspyN#W%R>BQH6eYWI< z?^XPI5dEdb_x-m30}lFE4xwufef9nEI*7dSO->uXKbI%#{%J#!#EUnT8j`$AGrRA= z`#a;c$%N#nT2vnAF(zJWa!7I%n9%zg-20^l0~WW0s|9e-{m{RMg1ITdxV%;0rJN%d zDGzX_4b43nR#%mJIC#~2hI_L!Id~OH*P}U9)Rk_!H-v|POe{@2XY5e}iFy%B_Q(gH zf_vBPt5QF4x@BK>H1Q%Q$xlt}ldQ(%`PXpn9piM)Cqv&C(LJ=F!GXp8=yKGXYS@1# zsZ}sFL}-Qs!s@WSISmqy0Fu7EzoCfx?gi6LQ26p;XzzL>m?WwmvDBy{m=Mf?fhCj| z1ydRJ#f^s=!WlBfHJo7dDNM}s5iKWwh}ytWZ4RpauVo!hV)jcrU(*XBJLmWNOASJ3 z-ue%j(;qy&Ib@VJ$Bx7P7i!?`;6)+zDY}3|Wy>!96B7ZPy^iQ#+9jIre^LG41?Fg% ze*FUm`ny_h{cTye^oWc5Y*XLYS@;bd6}1a->D+&jX+mzXGq;(W?@cBlxJjA%xr1y5 z96z@7f0O@xLjc*~vx`)>-<$jR!>(QXF@Leg$MC-a&T{7L}qew^TYEJg1iy`NM^uH`p()S z4h@He&S_JA;^usF3U)bCDb2v`7212YS}4F~Ee__I{|&4fN+T{}(aMcb23dzttm(Li z#4x%x#3;{9%rxIY?$+pM7m*=5x9R?g-xDw!SXzeNU1Gt)9=oFr_Bq`}P8;q>GAj6{>+n36rhI-$O#5;bh+{IJB2#l9@Nn`HH z{&Z4;Z$U9^dXHEE6L#}6fL+zx&rhn>iRO3yV_j)K_NY>sid7#3MTx;-9c zu$^U=qP$t9tgsPZWA6xiGK6bjntCeO*tRK=bERGD2n#|_$X2j7gx~bD&?#2zss`SB zigmJ4t_&R9r;)T`F!!Ai~igcW{&W4*PM+-z0>~YNo}8xDyg5K8ly+Q~miE49-9 zU=!_2l;kPcZ=z>2sQQ({C8h0)(6))p7Kg=OIrq^|yu(8tTe4Tj#NZ>|pJQ8o^cF2w z&$$zh=eJkpJDuzONtCs;4X$F^(8`VU7e_#%_PGi^ep}3tiZ#=Fi^EJdKh81-EDzk% zD;!AJj_j4E%kba=<6eiRb6c+z41~9f)j7#H!$|Y#wrxSqllBfvmh?fQ=j`A)(1C1$ zrm~jpAUmJH1CzcIb=WthmeV!7X*g%&|swl%MJ=u4ptKpv1mXfJ23=;``Zoonh(6m{_sqgq@Fod zfs8qlNxsXMOGb*Vdd?jE6o3f5^HjWuG1)u_{{&NjkF3a-RrTr|5ieiwxyH44xU5`9 zJ5Y$RHrk1z%*m^~HNVp<5LCKmNl*B5_o~yqj@(`H_Z-#E1QdP3#XA`}rNn>;H`qn? z|0wa{=!4XdZ|c}!i#%+KOjU9#xo~dfD9Np zJw@M1VH{B_x{di1PBkmoyzS_Zk>!uIIPK z*;|(v*FOn+rI4gyQ-~ z&g}q=p4%-R)W?cJ53n=a=E{g5L`K;w%aq|A=^!;w<}pbi(T1b}*UTyvUr*?}ar~aQ2o$(OPB@nGjNco7SQjt7xC$eNCDZ z3C!@nRI5Dl!%1`&qXymRsGi$xev1a<{Bv2D$Eo>emVDDj7x(AlsAnOJ*S;R1R?xdn zrvvRB)<_A8*EU$MD^WDD;@1Ac>ixugQia0&djZ|%ERfF6ig+i#RdsOX_1W3YAr zwgl{?v!jUfzz#HUk9!7OsHR+PJ4qj0mk5-g}|UsZ`7Bo_(TGMVMZS z4TOVVZDw9|ha!ChE~fU6wXh!6gj*gb+^+su)o3#>LukZ|Hu3vA!pv z=F3K+4MBZE!KX%EW4&qeR~)v6t+^@;+c?i0y;I7Ix`Hf>SV~z6eu8I(O>Bi3&kAl0 z9ypUTdCq<`s}yOESVwGmkAvI~f`UR#wuw~#wtPqCy9ry?XP=Ju85g-K;3!#8Oxjk6 z)oL|;>cRknu_r?ZG*z`o43rvQ6ZK-~cwW|{r+w=9yuy3eb|{9WAakW8ffsa+kkXkS z`Wb1oc|+Q~`byU#x0M|&Nh~ntf^u03q2i0M7Q_(VSMp;{sDX8l<&9!57M<*O4|xi+i?c(gAtta6GDZxAX*QK169Q!jse>HF-z+2__NJok{4$M0ByK zs*6L?I?i`b!voG?djsfis!|fPd53swb5STKOkrih9kzMaJfCmR`m{^z)M_o^)Prc7 zFi1Dz$;wDk)wiuug^cu(EmBQSfZZ6N`7e`9t^2YjVTYb{l^NT({=qMWR-!_cdhuZBT5dB=;%@olY(AW9rHXKd(sSlK@wN~7Nm{M zxrM(sFUFPW2hY|mM;+5WozFzBs+r^miC+0pEukV#yO@+M<%5!U234@nsf6~%*vHv* z->xz1%edejV^p&@IwwG{wI*aSaaKV0+RvTs?daa-du+H*scM7A*tDA`Fd99YiwZ4q zuk=i-+NzZ(!4rFI~eW>MIJj`&d!N2jW0Be7f8tyJ)bzVX3uc*NX zjCJzCikR^3H9hd4i9dQ#TfdUM4K>uR>ga{!dq^pS^t1hdR?s?P8@W2^^QA0};{7|J zv8T*VPY>vR(hrqTXrMCDi$_CE>tFB8&Fq{f19cae>^~+d{b>KODpCEPkdy`AMN1q9rYZv3IbKoek%M(d4cVq&7Uozm6N+fz0 zwc-IgMz%O8c14wsPV*4yv;1kCt9%1R*kN-cF#`p-(zDPyT}jjN@cCR zx;Vv?|Pi#_V4}|oOF-Y;vVWL0* zxuFa-fOS6b(9@J0^h(dq^G7Sz+1C5ry2*AZ6xd7HH6YB;%lT>BgDT-MPwnOIx!v}n2Gg$GR6BxC=HamaZ;duPd$*I7A zflShr_IR@1c+h*!tY>Wrzdh;lbuFObFvq{@`_~pXfph&||8Tar=?(8W{?*r%13h~V z+~pjGxV`7V)lt6l=P!~Dkoljp?8=0MP#r@?%fG%|zI9$?&yhXNmv4#eiFjRUQ{%SQ zOCq5QwJj-`+z@*^i2m@ApHE$*{t+FbqIEb1YZx2eycjYM>-S1sYjRv|uWRy~QLzJT z`Pijh$l{5n?eGl}&C?mp7f{XH^E3gL|V@06FBvYu=%nc|#vstyEo6v$R z7F>_2IhDg9T>Z;0&QOHSLpBVJV=uZDx2o?F7)LfWzU3ch0L3cbg~ln`qvC^mHfmxW zXFNMgBK&n@+L6LR8l(?fp*5%H&y4#SOQ=~%0^TY zLFNaIKsJnU;?9TOo8-e*RGv$OACdK=G=U>}me}2KWN!IKbl0pMdC0CtW4%#%`&#A_@hjEXG zwXq;_6>=4(rgfvM@h$saY3kj8*goua`a2c}2z??^LO(&T>+pJ9Z>bLxJretrgDa2{ z({3Y1(u||^)P%f}(dmEW0=XKSwya9V9|*2Lx&I2MMb!1V3*S_}qKxMTDff6l8XsdU zIdmOPtgMg?N%!~i{;9}!{MX(p`0!4zfb8TX3LW5~3$v`dUdP+FogqmgNYW`nEJh?7 zdIf0Af?@lRp37Lxzlu%fHLW5fRf!lTgyVqyr)cTVHUD+xvuhVe{Y;qOkkUWYG^_Qp z?y|^9&gqKBd*2>2kZsNm+^v=huHD`%)66+pA@X*{K;-J4;{y8nBDQ<9f3!7;NM}SN zGC`15!X8;myPe?g?rAJ?;dA<&N{5>=n||O3_&I%uhE35xHXp*0YIKC zoX(QkL|e_Q=c2(@!fPS7$vHRKHMMLBf3ku))7P)DSaJm`smC8m)yyKWAxvK}U5d^e zJioiGuK^YXD~fSyC9^GB!RDIqM#J%{Y%+5#KtonzTS175lMh_Fv$~DGvsr^EZvh*@ zG{`3i%$x~egN9Mqv>_B21dK=kOPg7)L94y*%AaBo%JO%A;>1+x$bawkP1+JE_6cL zee)v-z~rY>&h=2QNrvF&u|bS?BYJ?@xOqL97BCPtR{=i7hH=%7h(7rh)Cc9PpFw}E zpI98hScB@I>t;Mw@`ReoC-^*WuB7|Qi3+Sj&L@i-@_&9< z7z%~LUR!g$@fsUZDwAx=IJg42V+o(~&|KW!_66v4x^GywU#^Vhh>~R^EGS58l;bTx z6p{c2gQ0T6uOJwl4LCE>`B^{>^@`hvTVYy$#l>em$4U%IRf( zW=96V->h!wbzQjO`|Z7z^F08^jAudoOg0T=fp_TYw<4?O6T5Tsql^ znR!pTTeChn-kt*0OK={@tSwc-Nl};CaOWjpT?d+JdV^ZLet+XM>TcI9iW=+gv>bwj z&69QliZ*Y^%8t<7_8*gRTwt9cfZmT#!LNUR_FTcyJYH;+JR+~Hr;!>$H6^0OXnLtE zFe;8@nRKdKC!cUI_G=+CR%OebT#;$UIZ!u$J*Hxe@^t(5NCY!Dq+?5W@JY{##>$fv z51P(0V!R8;DZSmuo#f6438AQOoMbgI_Bi){4_rC~IO(>ZB%b(f-*^hN%0&EarZVp& zkK=wnCpfCmMn0__OD=u{mHgf4N~a7#D#$?<-~z3o)K0hG!O;T%)Ljm_ZG>f-xE?P^ zBW=5$?bcc~V$NTvLw=fG4o<=-7T!jl#=HG?zPE&Y&nlf{$U#PQiRt7PwVK{c_mdQR zEhnMm5Utb20cEG_C$zg$*53KW?mQhyrPo@#rf82em@2qip-4yKtn9O8a(@M%qFhgI zz~`vKv+>Dpp4n{l1Ify#;6~SrWUKl~l+v+S!taMj+pI>sDG)FVs0n zR4%ZzagW-)<{sn9)k89f-z+`BTzI)H3d__lIKn9Ybh)Ck*K`#84$ zzTu=P0*7IZF`84oT9Q^19}Pv+d}CFzx@-hvmpB>ImTs7CnlW}-E!wIC6x2}zex#=%sNLrJQR7G z&-(zMw}`m6$e&C}eYil?_v65Oeq8vKG;_wXf_t6TfEymA1Vihnna{|VwB zBjs-~C@*Fw4fej|eW`H3aIeAMo&&lqQq7{v&mPYybAnNp_LlA6yH9zK{!_VX5!<~l z1+vOdL~@C$rPXr+=6Ttop|L+#@u?%H5f83{DKX=9Hf8?7^UnT*SoeZ$*FvEN5ytQl zYB5W90@eve0iLpi`i{)-F>SS5O-^xao~I8B)xW$zKV*dvsw^lk70_Hfp}ZUL(9`uk zLT6{xo&nRbQ`zL;ZR-(1T{c$m(B_eP7-6FKQ4)|2YlIO7r&t9?s-a_l0MRg&6-BR= zvWHyo+cX&LQ>m~b4I}R?TMg^~1I~cKPKSo5F@t74!!jCmFP$yn!(XgSs8hbXgoA2i z?iw$PS8B&xd-x{1b+?yY8;|QQ?CY@B?Z?tH;rm3mD)&kDWu9yfm`WHcdDXo2JJgrC zb;Wa6V%n6x!mzhmWuu$9*rA!gjLWlUSI%!fszE4#Df|b&Ox{)e28rfIXogq&+Gb2l zqd(glxLuHEUOBLxI3O&nFyuOM7Jn_s?8K#=q$jto^gy%@cMRy6^PWHLmVdE2_PVNYIe-3l7@N5))a+> zy|j^6&Q^Dbc&DK_La%OXYJTayKT~>-pZOOmAy3Q8mO9ANRgn}{0oOCDsySgT2Qw3H zwxscR$ybUU6C?jhSCSf*QT1v%BmBnC#QWjy#`52gs+sYnCfQ+?>J)|s^P;qF*sj+C zK&()4ZfRoz88>i)t2CQO*3n% zcMsPZ6=vdgpUd?Nw`>bl&5Im6rPEJ15T2GVU2S)5_7UczJX+wVh2beZtN6&z+woJpH^g%->&qKT@x&ODgYRw_$f1ab8|5X-HRCaYcjdd$Db{2{xjh7CiHK z6a4M#4cW~ZvaO`<5p$`Vr_ANfYA$HZXy+aV@n`&8WerC0S?gz0S|W^vD)aZonSSxr zNbBXk8F7ELgJ$4xKjSp&^Nj}4=X)IMl)K0WnR#e_HDw7G$FrZ%;2~XbhSG(VlU^C- zT%cVnSm2?f2R{e^<1BtzJKY=4Te7O|Ii;mO485+XyHK47ow>HX|F(-=pXVu?TaWr# zTyHg4688^UBY*48ivI+QVbEbJ59%32*PDzpchlZg0iNgzGKxr8L8;ICmnI-$ma%WormSpMH^_DWw<2{usG9IWrm=U(35= z`L_F#WnsRF)cz~2{^c%@*Mv<4mKSkX>UR@f1R;2`1aLCZI^W#IT?B3^5AhsJrp^{vOI$OacmY%o#`_KGN;|@YDVF1R6ixA6-FSmKakG+%fj}z=-zrf!}C+fmScrYMud% zXQcdoyxUq`dy^b1@N_2ku;2y;^JrZnG4h?w1Mwr;db+5amv3ZiZSQ97E|@zx=xry) zm9+~BGJbtn&~;BATdloIqIMoZTPdx5j=Y@!g1zKYN1x>3@|Bgs@^eGfL87G_)>0?~3ppj%6W&I*l=>VGvQOZi7B=r~nLRb~ zLkXRrCZoC9x6spZY3Qa_%8_vN`y=laB@31pAm==Ej9)g>+YJvJw3kFmk2ts?!>GOZ z5}Z(XhD}6_iti)1Q}uN*al^f;i;BVA0~f}{pNC6oe{?l%AH?-G^Je|tkm^X;E4-K; zWhB2OH577Cv&(a{yUpWC9`)OD)#hws$Yr0MTc7nr+Fe@$KDc5U8nXoCs)b$ei$4vD zKR9BDqNH}{|M+dtqSfMeQ>1ue`r<864P?)t+sC+eC#%?^(Kl0OiHC;ek{})_Z;S}v zytust6d$~nIS5$#rD=2|P+ISOPa7<*UOaDT60dnKw2CERhX~1FYN}teT^XvZqX*v} z75fr=eD)=ZkKa$s_=O%G*#7S4m0CJZ-nl!LUFWHGOa3jj+-AmpSYv)w)kBCzMjo~!^qF*Z&l-=~XqMiKSE|R3+^Q9!djl9v zJ~4DmnE-2eKU2}nx6!5US_;UZ;yv@BK;A}pS>lykB6*KD^VRT=&Pis-oA76D+)oOQl+Ct~JOk&QmG(rD~m{7h+0x%-su&^jT9PRipu^ zcICOFjoqStg1?^fjA~0??#j0?PEqD2juGA$YqV_Dm$YicxWp}Tk+qzoC($mlOED|)&$)&u$*b!R_D0nhZgsU6aSeETZ?2DOdP#Mg>Wc7+Y=oQ~9R;LB75+WYClTs${ zp8X1?*K}1)PNpSWupV+o2zQ;`BHdZ&C;U6=N^_TzMYM;A<2>S3St)v3*4H<&32%z- z(d{Ty5v`b!w2%)qeGa)$w?b^z_1l3O?z`F%?AEe%@{sZ1f&B^uv%S&~G1YaXFP{U~ z1VV}EV;p}^`OA^AoVTQ-K=05?;{tlO56O|8>bEUr>c%uyW$&0K4>=P7`Kp zi!P4a(^rz^iCmL+ve@l@to3ApI#Ti2C&P$9=2`GhZP7}nCd;*o*W%aeFJ`8!HXY+% zo;kdCy8>I&YIyNOIPdYKGH_ zO4zHK*dq}w$Jwzc2B17|Kq#6S?eEbCaT}h7#Vd|6V*Q(O3U>Zd3vFZTQp}x$js!$+f>u0-}y(!mEmiHYv2K&F}qf8 zgF9dB!6iIoxSthQcXMg_seCL7hCbF*H5 zA2=X@aW{+z56)Cz{*)%%lq_kC(#tbNx}rc)K0bmMqq%A8tGUcWDo!lnRKq5pgY2fT zTagmqJq)lmUy5vk$Ti1e_X~t+t`8vsyQcyg z`M6DiVVyW>&Aihm<8kajy2CiNweLJ!AUr11V~YH4d*k#0`nX@)Wp&>x%ZIKEm$;{I z$zC#+h?sYLx<#mIe=hj3SNGP92LLIi_;N%nA&%BD2`o!z)g8W=VG>#tEBA1}I8^)* z-1UHk*&OiYOmf(D%i}GsL@TZr;iezb9$j% zQ=NJa5=_G$owZv;OdmIJ@n1gM|CET@kodGAz?D4CMh z_4BO{Gj~_hLd(SrV+kSYOFj!@L%b&I(Jwj!Yd7?zNZMVhLgNP)3sOQ4SNR>MjpJne-c}zC&?C%nPnZTt}Dvxs8l@{7%f`gf6__b~VG8u>UPCl!;0tG;?ekvPyX4>u@y+ep3zd?AIdfs+yO zF3CAGo(e}diWXHlTnl-!oh&(p<(95(@RX*!y&TYX{@rcgTZs9ZgQ*a)p77pd;Js== z2CBc}LP8q>9`&-^>Nn7Cy}%IfvyYIa6a6nAWr2BK#PqRmC!8YY_P8|{kS~&6enYmj`Z_qwaJE;3g4`2WM+djLiCbo-)v$dVBRlrRJX3KAsesE7zC zsAP}~l5=K;tRhhm0R@qy0+I)iFdzs>PJ%GxoI%n6liue4|IYWFbL!T8`D z#jvM)_3G~5TD`h^_3rMySY!Ik8g1XPeS}5p#1Z1X5j?KR@0p3Nmrpum(9G!0=A?xQ zAEcAB!zk_E%~$d3ge@+S=jl^9q}ldyGw8P){$#>vJs{yOcZWZ1&wb*1blsWso4mS8 zLuHr6N-VOZ>4~n~_zuZzjhR7coGZH7E@kljNW6##h(;GjW#CK3Ggw{m=y= z-e)v^rIP~_vTZkXK|DXjKn~{4kId>w_HDkm-+uYy^~1P5@Ktw5l6{)G^t|~3zJpUN3Zs&lA~q##u8*x+3$-gG~mT%uby0bdOU0?vBjby1;LrTV}e!yZdC{A z-!}?vaEt&Cxrdn80Q~);mv$#I2D;DKAVcyvt7SY@u$ z1<^b`EYHukYj?LAo*E>qMJ~TBG@ccAKFmI%rT&7v56JU)2OlQ3x%Iui<`-*u_~Y4^ zt@+e>UI#7yiesfZBkqJV`0BpTAx~xo4U;E}&aozDR}Nm}R1|-n^3IveYt8$qDOdER zIHfrBWTXtOn7tG#yykUwotO6pi)fSV31x4hPOgZukVg!Xq%n<@lC6!QPxqR56WFD7 zZ8%I^e!@#)$azEN%S9s!K8s@OC8cSFD*=?5Ok@$Gcj=2tDvD4w^A%5qI|tovGUqch{Rxr^`_Ot3 zU$0}%`FB+IO5{WZMFXNcHfNy{c%jlLq83UecI~QeJbM=w@NiU2x(S^*)iB84vYmoL*OCG|$xkw%Twt zT-j1}@L=^o=a}0g02yKNK380?tjA>}HK*b!1MgQRIh>fNfBb|5&!=5y*>v<_vfZILo4by=>*)}tzqmDW)qYu72x%0(8yiL5*Q)M4dM_o6W zEJgoCK*_7L>ae5bN7!9dpc_Ty`4lYlnb6N`U3Yn6nXJwIsiE7jEJp#cEh_cySGWO7 zH3fm@K0#N-4G#)2c5C$w!&0FQv2NDD1Rj85Z_%1k8n(2e&&&q*Rz4?bntI?=;8p8s%PpeDWyN zwIEHa%YsYVgO!2hBEY!y;|mk9(e--!JxBXpC?!-;_2tFtGZ?d1zr0%G(MV})zC}(vd(8aw29%JU5ZQ!1lDwG+m|Ge<;xNkFf6vg>?Z8oomb8=XPG{&dQEi%=BcJu zWi)iF77L<4lzGotBNK`#Zu$sk;_AdmBj0;Qfj8<}w#P4OT zORrfw$y|9wjS;!c(=2ye8$op0l&V#{?=W5hA>Gc+tg$H~%aib(^JvS^Pq}YY^LncP z0n9nb9tKUPwq~b!&ttmNH%dUQ*HS#J6|`_Hw6nDIy&+ZpE!Tb8Pu*Vu z9qU)uM@UARWwf6&ov~IgNbSng8pG%PT0BB$Yn*v>q%L2fe+1)(cu1^f_zv6_-790Q zW5x`+@%-W#Ma8gCLu#|ba(L||uKDFjLw7z$Y&!k$dVab%#T{Xv$QT=y;5xo7$`9C8a)cblTxaWcLagGyZu;cu?rHf>}*q7N3e&DK4-#8LM;*7aN~$Sf|WpUk5~?O^v8o{qmuJ7~kO#}JzqCxLsy zmg+$)G9Yr!QS9i%;flfc^H|$(6PXE{&r|hiu8EM^d(!EPgCd*LvGEm@htR0Yl5tY^ zvn||g4%&ZhN39$_73Y$6^~~2A_fI`@ph!FGAf`jnxV--`5{W}|wv6F7g&z>?6+b>sy z`mte6v>fyG6-wNA(@VAC69K%V@~wArJ=-6Pro!~OYy#wB?`f#-(y6`-`-~C(ptK^k z*^WU{5&xLt#E91riQ84B_z|q)HXk6>_a`t9&+qe==nk`~s^? zHN=Cpa~3BG*9zGT)PveJNxQ5h>KeS_Pz^RdE{tKxvM1N+^-|4972(wvT_L}gj**dV zxNOcEtP+C+XisG zPxWPSZaTL9#xqI^C>l00%NyR}cI9H_t%NS568B)Ba3;&62!?z^{f|Gty4n;Ae0+Z8 zQ_SrvUy$g>%N{t9-nsFWbK$({J)nN->b<%!NzZ8bC4L%~WTe8;Z{Hg9%CEfo_MQ@r z1j;bLK_BWA;d`I4nVNxpHaN`RosPwGrg)=*`*C3fG{m3x{dFDINva=-pAM_DHQpPf zpc9-&x)_*X^?34MAG-bfeuWMkWCd+wuL-*W$WWcovbLX8MNPFBCa(0@UbE2O9yE`> ze1-Ckqp(Zfv-c%xHGV7c7#LNuiO}tr6W!m{sAs*2d)?`bUYU%)Q0HGvsWPB~Z5ZV} zPzbofAbEU0I@FyZBce90vG)6mO(I^+IbFeBA=Q_4q>KzqrCN^6+#)~u-d|_F*|Attp}rS zI&rh61LoIq|3!NBT;^WJp`GE$H*e!n?U4QyR?ZhCJNBYiLYqE+JxZ>AN!A`%&&~uX zOXuX2l^qGW_3lt}OH{JoF`emEx3?UH+hLV;XWZh-Q-0MCu=DA;6lv74y0dhNXsv+S zw)kbFB->RNU=U96af#$`n9J!VF|9fWn?HE#R} zEYp>Pm21|QHkmVfD)A!oI5y!X@lPoZ8n9=8p+a4EHA_QwJMeWKUcahiZB5RhRe;1~W%jr(9sB=Mb9F zUN~e0ru1bdpI@#_9c_#nlkwxBUMtQU2&b2gYSmq4+KfxCWuXrJY_*UPIgoZx_G{uC z4vS;C>Ex{bwZK9?o{GhuVwd^r2NiYmCGW_JQ9~D_jl$wvhMrve4_B*r&IM?(UG9?U znpAhGf%JKA3LdB#Ji8G7UEIKhi9NkJ`;aI@b}AS#TzJVqN3kK~#^vM5pX^&TS+d*zj9~W0?V~Befl#D*%ge>;2#I+H4+9HUZY7~N(*6ZDyI-{YPxfft`I_Ss0@7*n$W-ZT0<_u@$DwBccDrc8M7w0N~w&F8^+FG8ti6^ zNopNboFa%lt|BQa@_jqXX&yKxna$3Aww9h}e^R*6aFguUXQh|6!LRcW-wDg#?%jM+ z;@mOrBeQ+KgOlSal4N^u%0s)c#Mrlc|82m`wZpRK;_MVLR*LVRK(h#otjR^mwr?e? zs!|m-%?B>rdpmGNfXmZ}&-DRKP~uw4OOxxD)>e{}F!m8i8~)+VN8k23P0mH;7$Y0$&`F2Mw7Z~JyDPDs!ns=BVsZXCrD4q;ot4Sm^dn6usO(Wj=M@6j*%k{oWXD_;qjG)r&xu&A$(4;x4qe9bXo_VY)F{ zs~CMb<&CfMtnSmw##=3|jq8X;hKDxAVTI(EaBr>qx}1pEDjbeNW5)6B_*5;=%gYjJ z1{uE0*Dv>SG8=9-MvC&~6{o37Cl&=-jfQrQ6t#o>!P2B`&imq3cAu%86Szw#Piij( zWYe;WqxLFEgIxG%W6#8|XdCk#5yUwT>>Ib074&j2x==)9fASV@;YnR!x$;y<%In?TH_gV&Pm7~YfY(0~y8@dEOgik3T!E>tm*oRYC zC}fvqcv?(a`ZX)^+6I{m zekk9FpxmBJ-{RCuI2(IiyVAvvTgu4J&}GDMPPs`iik5<7B`Q$xFu zly7JA9{FKY`zvT}2Z%2>(iwlEEI0qy*Id2AR(ap+O|$beAMGwx^X^}d-$nba3}EL% zXtXRTXgmStQKox0{pzCHe@4sBHKDEan^|65b1@-$J))RKh#y7zgv;nq9F&PF*)~GM zzYgm5U2d~M?H=rLoZD`9CVge>JoZMFJR*8c+?It8ORY7sFfyj?K#YttH0by1|G=2H z^h3TtjMBgLHmgpe+(M>sPG67kzdmf?a-Yh zzwP;Y@lFjMK5Uo&w1b*7WZJ=4H{ePXYkfL-oBFv--V&e0)ZlT2JVpHwH)x4x-#f)j zf@8&$GI~_A#p7F5>$hGTI5e>nU4%?&86fn1#tq5Qsx_J&)m-5+Y^b3oZpJj z$2U1SPlvyL`P^1fEE88>C)2Ro9ZL7oJZCuBhYJ(Wfzg_gbluNSLuu#OW6ej7 zXFZ?%jC1!jy?84+Czze^{#>1ZA-{IF~7;mH25umI)3hjzl3 zy19^7ZCWv$;jI`-!xY`hXL(|C68zlp$-zn?-qBA=t-Ri<4xh{%+!Op}$cOVIjK;;k zYZwivQj|O^+@^1EYa*x7m5ulXTSIk8jA#IDOz3mCFa@lX`7?d9mDC5_oHg~b_jb?> z;h955YNGrHsSnpL9PLIl)RDxz^)2QR604XRi3g{uja#aX_a7UR2I^G2B>7_1;ZkCn zLX6-~2}A`JFV|1~=%vzT;0_f{36=Xi)&6KRY{TDgRe)QSaw*NST=3l*g_G^b=wkEH z721fwW)rT_tlqxO`E@&Q0s7wh>Y8-l>q5-?)1s)n<1fE9wM#eW4EQpvXu{)$=3Ec7 z7}RzWJv-08x2WdD@(Kr9(KL57?OdwzR?}BifiUJ7!If8gj3nv=e#S3j{BTr!Mwg_;Pc1J(Fd=;hM5 z=i0G)LiKh8;+FWwN7Bi~f@b8VS@A7dX>F>T-TVq^8=~9c{3)b3@wRo-Tr&5!(Gjg~ z`A@D;dXz|4)!$OeA5>6)uD8ey!neSh%{CvrI0toYcOgBq+x6wvunSoI!00kK5S9HdMZ}4slI#Uqlp_ za0A@!<@d~PUsY&yty=F$6#U#;Pmwr`kiYF;cV4nmukyMUQ3K^H*k$fZ`R7<6dKuED z3)~qaf+lz=ro?%UZ}cZpMLShF^QPh+J{#zyt=quaV0&|WA4TqBx^K+QZAS(P>D6zSsqfhp2Ph*@H}%vU)S@TrXBI7#1zHSGR+lFDR(Gaz^?}n zjFWXUjD1^yo~s%$snSQ{etB^d`cH7Y&)Y1Z<+D{uQb9#GdER9;+Io^_R~#h8j`YrT z-DzN-PLAKad^>OH;YT+0nL2i3229wjl?O#jPk!igxs#-rY|c?+2q*rTSnzlxYv!i? z`_f+766~&%8+av6<)tKpVIyW)ug~`tBN)rjXb#%UJ^!?AM-l$E3U7(89L9woyGK@{ zsAts<(yYXgZEwc+`X{ehN2I9matBWlY!uo(Eu8JMYYA)WqE=?hoJ*-v-qL&ZRVi$9 zHBYC*`W~+OM@Os3+EVctCOF8Cs1;Djch0egHH#%~KEIqDmozmR)ToMkIVf7!icYsP z{TYJAXXx8KA+y)hFnCD&GLc6_S^O3!%tMT9^lTt@c4A$5A$8U1NtSQTgPq>lSCd{6 zava-2+Rg8{Vofl)2YZFOd0y=p+Se9A)NKwQx_cC!JrGRScLjHj`n^c$E2%fEU+M^4 zuC4~01<3D|2in;MfA}BSf;Px_7P_?Zno~XXR%xCSw`5$@!#5*A zsxGt4w%ZkvGJ;CZ?w39{uS6%8fuoxtT{0GLMRLJc@iHYV7AouYBVW3^>2C6>vxnPV z@i!<`TO@c#>*;@4NNaztcJ;;U8zQ3vRp*~Qs(NZ+r(ZO4FRr58J+@)Fu&*PjkBYd5 z*)tic2>XgiQChU)x%z4I)AC18laU9zbNhi)khMZvQ9JZASE2!YABP#euZ-DxJr9sF z8Cc8u$(py~94mKUl-Eq~M*){MfBjpUJ{0j^BMg$yioFUClEcKR@RPh-`TSMtse^p{ zb=gXa3WGD^>`wAReokw9alR?97SS9o6{Z_pl;>yo^-x3g0)am=ffAwqmEmX`ZMS1H9Ri-Ye(X3-$A`*Y7r~Xo=zAM-OB6F1IxX*t ziH1&$VnTn;NgvPC_TA9qtVmr{Hd%RTX!2&-U%vgrj`|aHWA|$N!H@S}wYHQl z%)Rg$4nsuv5Uar&vEj4%9KZ+vJW9t%KD91F3yZD;U{BhlB za}S)$Tc@Ch%6%pDatxI3YrBH-%elj9WWStkmP#`hnXp^+oQgeeQ_YGdAQ!l(6xJT8 zJPTP8K-7Ld|E}w`IpMNx+QN?~Zoh=|$A)7&eJy}v9j^NH(%aN$MZcpk_(q7v{>sj+-eE+~1it4?9X6u^ z?gwR2K{@kn&=X%49ojaY+-}|`+Pj+n=}VwhOI)!-`anDf)Hz3y?a$x;0tTW@6Mp`O z+&Iy8=9!8u2P49yPj3xv9P^B51Wa;z+<)eRlnK1=9^Qs|)2Wu0mU8F(II`PJmZhMP zmMDmiaHHaC77bC5J-AnMB5Upf#g4{~P{nj_?p6#$K?85Iq&LP$6m%BX73?l?C{91~ za!v6W#uBdc1x9cZ{&MzXn4ci)@(hKenylG zWdFd|oOOhDQw|O7l_Y>DO#TB=xnhRX1GJR$R#-Vuz0xzUgutFoTWpDlcThdqv z-#u#Fmm#?f9e8ygq_-0$xP(kjIw+V1j@QfRFUlAk2ZjS#%D66UeZ^2c(qu8Sbg<3> zPB&-1gxTC&+KfE!9584&CfN>A_y9{2lFa&1dBv;mjQ*~(AzUW{Ce0R9e-eSfO|MQt zg>b!mO;_5GFo_QfjZ&kq1ANLPOR5YV*-`6B>KUYvpf@wc&NH-7;MZvWGQ03#-01o= zQYb2uM*DyRWcCnmqhKCiCzSaFO0n~C5IMopf~Pv%hxQ@~=Jb!vm(=>mh=V?%xAOzr z*~_5<`-);n2yb_8kh*bRIQX9U*MC`snV$c;P55i4CU{g{%RN9v*)fsn{C(XFeWgK?4%EIeZZWspXF@)9*JL7L?P5*xFHzA`5FGC zfZ&8Wivus08ADt;S9MMC_xKrYdgaNQv!~nsJMPPcUFrWI@}&#h&H|pPN~}V{^4VGgfgplnMu4!+Uo;roYlkH4bblhOHqs64;Hg04& z7?-i!9zMUnxbX$>R+yrhfjxGpMXRobK zyn(I7EOS7yPQC+0(TJ=J*wnAcH=BqnSqIy+&WbFA8MNeUoC%f+_3PzW-#10L)$KrB zOYgy$@ui<5&v00z>p6K~MJkwof#+e|=$Y zD!5EIxYi_Tnn83lJOkY_LApYM?w}Q4G|p#G&ggxD;in7TrahghX7tht<&_-M4}dIP z?B1B*hHua)(UBTjkw3Uy@?6uEa>DyZXe~$(&&e`}c`Ht98jRdEAN4|@3IGKLY)k`k z)fj+|6LJqvc|bdn&oLQ|q&sQj1TPoa#*cu4-m$(vY^WMgu+sT~mIU8s6g}(?Ic06P zUp6@)aZ>T;?|%h@IMH1!L}S-4zAf4g+)5A0{-?o)i~pCK_bCIgi4Q2#z?JAp{yy^B zFPp?XhTH?XlAcR*x5uenV7>lI1=T~o3D;gs(G!NJ}Cl^|RT0Wx@_ z|ICm3J<;G6!_9vQ>sHMF4r%?v9)ow%{u1C8<)*myl-i&v{5=)_I--hXUjJalUF#5>o2zmQw zeBa40!=ADO94`U)8OMxIB!rBHun~Ai@!dv!NGwk9^w(tTxNlZ@b&=Yviv4ia~gf@*DOD6i*HZCXQIn+ zNK^0Wf;8A_HSUTe4v~`vOT)dMJ}kw}@4d@YtXsq{PW!hlD?8T0R;Nmq6`Ww`kp1|+ z@K^sox!vSCnm{c3WHgER>m+!sG|9DpDgKiS|C<#5jpph3LpM#sM3qU-snM%3DU&b+ zoa+5P<;EeZ*F;6yW9_WR+(o5(Zd?>e6rlbqgX-^KtG8IN(fz#WsQxLsH&V3ZvuI5` zDrL-DHoneYE_fjA`&NbDbO!T|D;pL?MnE(po&Wi$_>Yn+-4CkwGslW5g(#T&_cth1 zjn*>ywklFB%%iH*Cs7r?bPXj5_`3<2P)XV^=;dwPJO|P>Tdf>P7N8{q+L3}$apuU)9qpC*Kp}HNof;7s%k^hxPRYdOL{~d8z>k%ewcB-Qoj1tls*HY zQ8Rl4*A`wPU*|aW3)#ASe)YNkpl48PgFoTIVFp7c`c2x7%Q&I`*B#e7eR)~VDXo*y znrQ)tK$pH9m(J60bjBC;q~NrO(}{oV08#*7co5{ z*4t08Y}2lHG5Yn>VuDzN-g*|UEg$v`)6@S8wIO*KTQU9fpfAK~ND-s;>S>@^H14uv zYjDXuEE_5r#ILoXV*2T=;o#4{6KDFRj#0f=EICT{5ss$P)o*pmed`BN+E0*Yv=DLn z0VtZ=TawutTba20m5U9V$v#2zYimazFV>zuuCHwzeN_7^^gGD*{e;hRbVg($8nLJ0 z*3RFsc|o7xy!^6Q*6f$)pplDgR`-@)UbSg4NLnx3xa2$G{B8+%S&VJ6V){ex2Vbr2 z8si*ygiOe_4M~KR1F~%VmtNV<3$dy8oF9Sn`1<*{7wY4BPDbs{l2-;BEkzFEWN&af zxQfgmPzr-vDm%gL9OvUJTb%9LIiUQ5g3J9fhZ#Q;eb$IET3>G0O|}wkYw&LRzGy3( zUU*u$><#is6&-BUFP-jC8^zL_#^579I7@Uuf+mHap< zv~u+RWoO|0PuC~6pT`Mpa`;-yiO7PuG`70+h3L7e)K^RoXE}r4E8izjakyLhL9Vig zY;VUFLX_JJdxt|*+t=wZv72tFG1{p!+!6FD6sKRw1?w`UT|y(CADLWVySLPA;A@e)2qj@s$CePlJVc zxc(yXa+K#$KFyHJKBVJ-;~K@m|gNl8fot{cK4!a~9~ghXy! z7m<_^y&)qk%=M2Cj{+t5BmS*HzvOxBy}aCHgoJ#3eFc3*1zkPtghZsJ zrBB)1xN#k%xbEri;$`7?-NlplZzli7N6p65%EQ6U%fZ!!>y)pBrK`7>JP*%lM*sZ$ zqh8K#|IEn6^B>rOC<^&mxCw~}3Jd*{Gnh~g#6(8b!^XnP)#HJytCPatHNWNHVdLxS z;poYwswd7R0;Uoa5tQW8cW|+G_4Sk!`p>EURg8Zzv{AG0vQfApd_(lQu;g`Nu?HgJ zGGY=SQbzwS{oh3YDY=%bwS%qy|2Vn0*gum0JEDIimlHZg=HDUpml#ef`Jcl7-w?F6 z`X_*H-X2bW$;sME$i~UW*~Z1o69iG@KZ0m&C1dOA;cVfh;NWawXCvh1Vkal`ufYFl z`Tj07s3)L`3jI?@|E=i%lknMNg}`qS|Iw~LaQzkn zzeW54H4N4x&O^;-!1 z7V#hL`UBT*A@Ezof3)ilT)&0DZxR2|u0L@776QLT{71X~!1Y@Q{1)*a?fL`PZz1qo z#DBEw4_v>6z;6-%(XKym{T2ehMf^v*{=oHH2>cfDAMN@B*KZ;4Tf~2~>knMNg}`qS z|Iw~LaQzknzeW5N>fsCS>B#pA_yho4fQFj#18_Iw zWRI@NSlZUFD`OWm?9THs(HoytP_n>8aLiH+@sbtr7I~6hrwg?Haqht?+1X|qE>nn~ zSJb7x-t%dkOC3_<44jp9$@H;C=jn+kW!Y}u*rPtwJ{NhhwY~S?m~it!vZ78L=onM=?E;?Tq*O#PG zV0%gg0pM#6&Sqm&gwf;e<;~)y9EnS~!dxvPh?_47z@>XoiQrqUKEZ2FOA1}WnHh4~ z8<cm-bE0ymH3E7+13L;}iq`=~f$z8b4il?b8D_Xu0=%RE;Oc7*n~pZ-xY4xQ_>8d!J_4l<{v2=y5}=9r7|0SY zn{@pe^?^=_1SD4Jp|wh^Z*bIf-Y(!in+S$Ipk?`0I8(;8VKeB@z)`M#~Htpl&GZP zp#TGT0!|tbc>)wp<7O0YY+yiAkPjjK=N5WW(?~CoeoBrOI~+?$%1%;X z>ybd;>b&%@6H34odk;P=;D75I@Z_zJcfiPyVO_zt5r>|pFT^?A4OApPr4aYNFGQQG z(2~m+FiyAUf2jSqqoJXToz^RU`PrzzG#HKk9B0h&7SThf_*!}nEPD*7q8t(9Ls=Iu zY2o<>77X71TI*LbJIeAH-&&oi$-(7#R%w4}%W}P~YM3|^wdl>ILITXZ+KmYeTU*Uy z*Ax>MDf(X4dZ-)n4oa5pipt)ic6yFbX1xUfE_{`l_={Ip@eLMQTGZzLA=}UAAL+YJ zA-vt#ttXqJ%DI}#0J1m$FwrwxsQ#=pesXOwX)w@6OXuPv#L@Oz<>b(onK4Fv8Ih`6 zscO?s)aNWjVgYLKXad?@h#$^_#Iq1^-z|j9v8itnf#VLDku|*i3Hz6v!&#Y#0Jq2h zBjX1nLxhf9T}pStRH)^d%=1;OQP1i!uXkUo#`Wa03dZi9_L8;u0u%UPI}t4b+A?_O_5JSs}8=%VoBHV zB09f2D6Z#90N{~*u&)z$$;eOm?1B%$8!TGD!HGbtMtDXiC8^}~5@JrU4gsS>;B}>Z zyM@V`Mt%kZ2IAOF{JMt~_*`^bsqNXF=55^8}y!oNXOi`pT zb8j#vZ)r^wMHOmJ0X)?O!MnWrY%!0?vOaw!CYslbdbg|?ELYw61uQ(uo~@OJ9-x1F z+PpUww{bqOq~yw-$E~#k4GK~q6m_GtRSIx~auC?K2i}3Ic-9Ry<7pEaHcUPfP61Gu zHBL@vEi5;H;4BR?JjYTFukPze?4RAqfNOS3CI{ar_&4MyFtS_yvI=x;>5^{1Nic&SYNivkg|(ZJ zk9R159qo|Q0+eons;0enm%P21gQ{PG6+#5mC$bzD^Cx8D@fQ`3gusfWc7`c|X6&9E z5%4AzjEl((k=p7(d2VizOFlZfH%O(?_mPf*3Ib?T6&JuH7Uweiy=5AF_lSvx?mdE* zTKIGDQGk)1LR*IVT=F=R2}bBR*{lr6O+`mh#nWwKKzmQ1^#o31qYSr#=r8~?FR*i& z-lV0Iec$A3K1@rnt{7O1lgWdX_1@p-WrkGKNL&ooK)nN1p!C}ID&Y!bu9^bqzkJI%zt!mB3zZ5&%ku5nFZ2l`?bzXGFU`H;>m(?e$ zywGP1{*{m!GQsH_UJ718ZKblds*{w&%-KH_KlkQuJsw6hpIalIPz9Qr7Ec>QZby7( z2=!U#aN$S*9{Rl+gE7ZV6~@sz%r;ZVw zLmrpfv#BLg05ui$1u(5plfN1WXa^ckW^?fdvy%-7u;!chPR-S3WNR1zHwG6L$J zr;0kzR1EeIj5b2B?P|=++}zsAqK{{uV`s8J12x^KKvU)H7I^}1<_2bNfyx)184?kO zP@Vw0w$ZbeIs_9Ll|*RR;Oc(mBsIc$2khZaJJ*N^0Ioy1I+uCzY6E7CT*Z50?TK#* zhh=b6zCD`&H3Wd31JgR)NrcXo!IHhHNDQDmCSW(_9PXgL ztiHq5G<1I-R5@W*9HLpNow7tgTt5qC_LQFjU;qtClNKntpZ6rR+=g~#ig*DMfm6Y9 z-3$TQzxD;esq@bt(HK3{ea;+=;(~A~fd$f6j|Od1L}p0SdnFXvC5M}d2%571(aXOq z2@f*uWBJxJC`>oyf}ud)O}EDvI@gFL1WH=?h?KZaO#r**Y4XjGCgn<+s8E0&USB}q zC)Hh0*HN5_4xf+nNw?!h(75nX5h+8d=mEJ`r;Ojl1cDXA4*e(NdY*|09u4;D;u<;v zLV5I{81S;780uIy3}(4dWM_cPj|Ej%I?cen`{*kg8dP*36O~irnP1M;Us@mc738n1 zqX=YeK;Jsnfi1O*0*LDZ+s8|6y*>xW;9puq9|{VMF6G=Am!D=Grt|oe)#?iOLIq>RD&vNGk;IV<(EFqRi zYP|rIJr~K8#7ehwI0+CAj`_;Uuft0Cu`hG?U6P;xb*Fn_p{sdT)K7_;IhASv(v)xy zfvv3}hda1VWjgvQ)||kv+uRHr@gb%Fh+pX+mvrrAE2Ma6R9t021EsuRLoNV~$+86J zU4Zo+c9B5tA2J16Nx)=4X7x$Ic@xx(rcE8>mMkflJZX9gP3|8wr2|Sre-mmmNd);p zE_M;uwvY%w9s@zJV^D(uD9KL=zYk7VnmU-2iIkeb(4ir^Cdq=$Oi*KFq-d86W;7W^ z!Qzm9I+gj-aUgnkd5#lk{ChB=x1Q?=!tfIuriMs&fYPE&ia)@$G(p7x(IK0F69@_K zJ4AD66hNc|37R|v0aE%P^LDDvFlz?t3!ytHmIYADioeC(OPvaKL_- zMJRSgjli=lz%I&sotUyg6%M#*9#kHx`|s?=zb}BfKQh9woTgeOBqDsAN_fn4@ERwZ zgz^STQ6hZRl0WLP#r3fK>2~FykddMBFLHjz;F#uQktN3hiG|EWfkU;B!};eSB9d$K zpH~8=yq7q(Lk3e98}MLjnZE-{3;CMuM_-6Z_E#vNU|A%SYaSxbS0@B!bMPqSp{-H zMa{JK3$yUV=kS0Vl}3`OKOX!Q-d1Q1xWh~STksS(h)W8^3GKC7>6 zk`C~Fcr|r@ulBi+`Ujm|D+eLJ^S~~2#|pJx3yx(n=#cRzTfzqk`Uvg}bWQQTJtyxSmOzngv2 zTddH?=7kXOI&t!q%zN|h*e^!Voe!BKjfjS)I=`*~3yS^fh|vK_aL5dTvDQ+{1d!N) z$Yk|B6w8nqlc?D^xeu3waFv{xBO2lwL9oCISO%U4=o>KTa)ffb53xe3Rr`-re?}k~ z$5rf#L-%1g6ktT~m5UmylP+*o&N{hE{V_lS0mw2ii%QDPHSc~TwM`DFPa;z}dStvV z&NwN*V}U5qI_2z}<2idjlEFVb0B!COqutiI*S_8oa7M+qg$s&Og)F$8U2*^^g|J$` ziyObGP(M(FpYb~;hG(&V8cQ!Jr~>@0;444`QG+QNWx4(t6vRD<;iHfz5*DEHChmOA%9c5HC~k1Y$x9n2!bQ zfD;hV$``j217z>a86TJ3bZg8Uc~jfb`yEJdec6rj^8q`_ufzpp8UR443kp>cW>Qv5 zV)q2{h8ifZ(fS%20?;GM))H*A**6$sC6iA%BXLAa>kd6@5ys!k z04_}-0tOv5c0vxPn9O2-l;@PU)pa5WcXXAXM-eo1E!Uv{Ps+th#&y653i9?0aO~4j zzq;I;XY&aU_PTnEk9;6S;3aB|fVG8euCAcSCLQqDk!2<^Hu?}$h!?KW1{=hrfK=N?9d8G^7K{OL2W5ml;)X zh(1(2J7NSer2vo;$EDzmZe;ih04HxNyk#D{cNyxFNCj!KKPfA1AO?1xuA(D~Rg?-L zP5Z~lJR&-9V&C?uO_6C?OYvSglKAoAIzj!kyq_OkO}#q`>COl;$l3&UIu=^m6f1BA zRKQNN-MO6Y80nXE>lL_LV&nKGS^$`lY3mLV?){ti@dm{^8JX5%7yx$^nYeyf`#t(# zj276-=d`XVCI)88?u@O-wsZ;rj;=R{oavZ9$v}Y2PmjJ%iy)2-oxHljIRK!^`lumb$q0IFjihRN zYW;K*1^|N3Ey-sp2KYQc`cdYaEJ|N7Pyh~nI`_lC2IL2a&h$_KN)bh%gf>rW5Qy*r zmbB#mN;}1ZMY+Fb2IqrQj6neS*#>8Q06jMm_|3rc-{bKDq5)`fbR<{?k^*l08~;5j z{a=^u->?zT)45B&r~SD(j?Te-L9LaPr)yx2#P8aM8bwtEKAzeL1hnq8E!u z;CV|=TG~7zfYRloQ)98y2)b6fYI+!5UCpnTrDl^0FWt) zof1A&sONjm0oN`l81gwnNrT!2PH|q)LUbc!fm?-~R`#o)f=#*?TAd`idqcqaourmR z-&gZ{06e-;)I`$uB>1OY9o0LFHD&U4&{^J8z=7>q!F=sJL|Q8XtQbZ`XFbqP));}~ z;}a~>V%LHNG~%!u$HG1k0MH*(l##@Bz{J6h3I%yQw9q1ejU)y9TE9kHzEKDAsi2^% zCk`Zpdbv%A;Nwak0(@iBkOb-i=wG3h7O8$i`@}*X+<;ShkiniaYGED#IHjD?PC;Bo zvV_zL3g8wD00;4L#?+)lKv3Uc`>ZA#-DU+R@G8Bq4F&{8GZ7O3H{NRN1gb9;5I~oF z@j2iQEeZV10DURUOIw5sP)Y)wT@zm0NbBQubB`+kt0MW>WDYX`So(qv(jH^}y06IU zU;QyAXMnbs^mU-HK>tH*2|<;ueL5BFrNNA73|XU^1OcEGbj?(WbM>BdnR{FW7DDV! zLN|y2rFuk$9fH#t8;s@_B!$zk(FGrjj|+i}F4bY2tF((MBP5-)33VN5KmmIzw z5(#<=0t-dBca4sQI-svF42(MI>ixJmIjN;@L=g120W5$cq|2A2-M|4(4X?TxJ}KOs zkdy=Ava2$fLyg)=@ZW?0zBBz@)89*A-)F%=ZBtRNCcdlpD(G3413hN$i;F@U=Hq5f z?NV%jhmNyC{m?1{5zxd7&LY>R2}5bAggxwSl!i7rkih{72EF!f<^@AT)g^r<*X9(G zbLH;~rv-ErcZBA^8}ZCc)f7ZcGoWW-T?dA9-Q21&b=bzxyCU$II}Mq~ zP}`3Wz8ny$GePT|lJ4q-)Ed*#h^}hI)6mJ`u1zF>^>$I_=IE6z& z;I^)Oak0k9m)Ad-u7FPNQ-3NeqJat68F8p90M)w77?&!4-Ew_;AMzb!IePMK9;ye+T0nrnznueh24k=knyg-&(C#ixN*~#g?kjSV;BMdII1ooQ z(9Py~Tu%wSDT8@^E+)8@7jIuPE|)9E3)agKtrUXO?4%K@<8YeWNy%P+JMR&xL_n{! zi}%6ix*k;a(}g5_S%(8`7??;XJOEN3c3$eq=S^qVbP^&3ia;|4x=+%CNJCONuJNCs zuIgaS+OSv&g1k4Sc`bqXbHEoxN)?+X2L7KWQ!0C=+%PUxy zgGbj^{sLI+BOO4`LPvHS)Y)W)mmn!4=F&K*ln`2tPf`%V(kXC(v-j=?hUo3%@fQ#v z3js@hco^u{3ma%M$FoBwl;w-BUt>GFphb~Tegcn-J{S8?bu0k_Y?B?EeCtS6gv`VR z(Ll}~K!C3^vs_t)tf<2)024TN=hQs0eMfixT?mTHDiU(kI;52p-nb5H$f2XNfL4*nTO( zylfdJ2TI@nbH#&ir*7rmm_H%En3NuL>Si(S#In~Z??`+ACp4;rZ_s{e*mQn*D)I-_ z66lC|pyQauPX&oD5_tC*T$chTfNR76^>DMhx@?c=H2HB*M|SQx{m*;gG-jrYRAp2g zoF%8N-T)z?JlD^aMK_kbzo_e31v=8V=>h8RcBDo8@GrnB#xWWcjhlWb&2H_+J;Us2 z(C2G%3lQcLqBYS6ydlA$Cqi@xEO<>Vo-0eK@A#7DX7K1upp>hU7jm_b?zL9er#M|G zFf&BNzTi2KU(SLYsy<=>1inX;6>Fwf*Cknki&u_k*f4aHY=4~?TwDV6vxxyhx0}dbxP=rXf)npf!o^7+uyKJ4kQJ*>!8sO66Y^N2 zDLm-p(M5URgExe0B!8_`j&}Nxgsm{GjUnQROTIaJ7yp z3Ye)0g7>Ertuuf4YpKbiNSGc@L3Ui9P_T^E=LY=VXO3iQoZNwokO%!raEyjPMLODD zNr3o;1Bhc%rh(a=Sx}(ZanZoq{G;!_mP-cDnJTqfPb~3ya7R~5BuiNbMQmk1cmO|NV$hyqdW{EOe@+!9( zVkpU{b&FleVwGFSt;plDOT3{Hi88zE;}&)rrZQ{u{vK=hvwzNW&U2n~&hPvC{m%J* zf1fF}vMOv0JFh^v!QTv8HQv|5S=TbM`<5#0P?A&EkZPZcI}?pk;-hR)lA0W| zZQ86oz7Te!QSs=t9r5fl=(EbI16~B`@lgUDjCBe|y+aDoogFhp9c8h-Eq90Nn$vm7 z;c0!o>WDS%OEl{m`mnRib5jFP8C5lF;=8Cz5E6u zUB-sQ^xJhVl4`)0G$<(Qvo1Mi=6>vXi{l8Zt)GZ%U#2lmCW0^lyk%$iGwP(VX^|Va z#Cr@w(8NEd%UXeS7MuX%xnO?Q9|61e}h9}%eXX6yIRrKfrhu zEQDWZ;__WZwd+$oH6FNRK||#J5u3UBgy`Nr7PAG$be)o8;H1esu0RVzg9%Y{f|VGo zBWPvHXT%kTccq|RTL6!q{1fp6lf4!WL8&||f zkv^XM3LGZ|2}^uS7+!VFnFT}^2)LtnCfXKuAkXAKp7kN7lo+_|tF&b?He_8;1ptpA5cVTkhQeD|l%vd7q z?0e8pA5_#e@276xyYKx<)u6JRz$Pp6TdHxO?1Bgl(uLK@Gj_4l&Hzi0ck>a%8n44@ z+4W87BIhetxTW+PR^E@C&N^MYq{`NLR2$z4hs~Ps13%1FfRE{~H2*&D?^rky5F8??0KRI1rleABgmec$z7_j_N@{XW0z`SW>mT#q?OVw5oe07)lDTQ{+j ze+Z2f-&LVeGh%~>Sh-jMP+x%Eq9MdJ?zE$u3jm4Q0I)Lv_$H3n?*WL#1F#eTfH@Zc zB}Qp0l_Y*Z20A;~f}J0yyyGrMoa|ya{u%=SR`!P=L(>h-xTDKRlIIOM!>PxXD<{07KTTX|c;Eg{l-tXviv zcu`tYpE{3Cdzbzmm7bG}WZ7w4#3(-$?T0sr0E}=Z(3DQtAdIgh#C|2U-VJG8I(2la zs%**OGxO!<=AF1jGGVx&^=s+Qk*}pO_jerd@Pxo`=YUF!RLib-3x=#Clqa|(P-nk| zrofhdGMF-v7?H8T?-Rn;cL~!4C!V(}>w?G7P813CJ{>tn2IxIy&_U=;*bFFdlJd4U z{UhNCZ1%1|Ltq6VFaT*@Yh7E26@=mzRL(hB=^Ht9Yc(Tku_K0dO?<`)Rrpgx8CXSJ z$R5{+p#lQ{+$&jGq`Mv!FgjMV+ZunpaDZa_TvN*p1MgWY8uB12V70P8W?gb~A$z1c zkZ{}3mUK7`2jY8r*>Ny`_&IRr@zTaUdg9YyoQ$#v z#ZX^&E^qh8eSwn&)4i#F?3ds{)^E5Q$9x8kP0)uG-8KpNs0<`AEZ4gJSk&~hh|AKk z!YCn6kPwjc0JkYg#@&6aFu%G6*XW$3`Y{ZYX!;>?O?Dh=((VegFaR>%=VkY~IdF;C)q=Hh3xi|Sh2k{-ymyp1w@W|nRJnig|E z)DLW-M3(Dv6vNk_n#Vjl*t1STiS-1bJdQQbA@or!pLTOIwWP}%(V)I5qT$)cDx2xh zk4OY5r%!Q_{mUUTbA)(h_&6jEvIdjwoNdfoB+3Y(Dd95J~d(V{B%Zego0Q-)qxIL^wbzcv;1%*a#iJN&6yZ*&pAeb&t<>fRq) zBHUKN4K^xuEuh)aA@cpbqL3~5bxEO@;EVSW#UjdLIKRg)x^=YN`kG?dP)AyMQ(Pv? zh}TW znJ*93_hYO+sB3*>f%dgA#;hGdJo0GMK|@knLCQ^(9&ac(!kMaTF8`)_9sm6A9sPiFP|MBWI(i$`d6}_&Sl2rVfQt@7{_^Y+ zYjMq>_Q~)kZSx#ePgz3W!4L|*v$35+%#?Wm?Ub65I1wJrm+JDzPNm+MoTv8XjO+eg z9g(-zV2099Dt6#8OI>~~ZP)Lik3uv2E!q+8jR>Vg6kEUxN#%4-`kN`x3_@2Z(SqF< z{O6YW`vb8Hm+ksAd84A>^1d41BLNce1v?s?YwwH|59rJgOwMApdm}Q=#6JlmsHtHD z`_`3@&Mgj|qL~={N8J=no=931Iw`mCRX}g50he<^Lz{w7h!NPF)H}+7*p+&1(kfiK(H@2QO`E7#pNT^|lIz!j()^?7%a1@_MA zg|Zh&b~v9j`~} zZhmz6hHIMY$GPXZAu$zv#`#eetl;z>q=2+uecVX5rcF|9&~XJ-U?OZODe}Dg z<#O_jXCcwk%I6(wl5g^EIX-E9X5E+v)6yB25LbAIXvsVHgIE zR2Lmh6FK7iUm)ZRdP_hoWuJAae^q?^12hG>3Q=38vvdJOMl(x}rL=AY&6jq=M2B7~ zZg#>8y4z(nD?4vvXt-s|pvA_M zW@Muw^y|S2J?|5Wxl+q3)fqZ&+O5E%fG`0Kqv&w&|7w!L~jn%_WlKg{!?D-jc%UkvnD zg%(#f#iZ<@wt#3ipQ(P~1qChhK+v5viLbv^f1DH5#~gI^i3=;yui+)nq<%m7l$p&^ z8dkj})e%|+lab|USxUCwB*Q#?CQBnl_MTMTkzXgz9HgHW+~y{4J0PdLl3epS1fe~J z6|Vwm?a)G0W@zr^j9nK{OU}AbD=Uv>4sf_q^$z%}+kZ~}JPv`g_woHjQo`6*G0Pti z6zNwsdA`fxXc-;+#;hWx8jVdq1UcF(Z@d%e!Mkh2uET%s+Dn-r<%HJO239qH;@%dS zuuZDJ{JjHfc~(V}{~sVCk{%kI@c)3ZnE4m@^N-GYF|hB)1oy~bW*jX#2w2f01A}n(q0vEc bkI2(7fsHSq4oLei(_>$F literal 0 HcmV?d00001 diff --git a/assets/images/logos/logo-96x96.png b/assets/images/logos/logo-96x96.png new file mode 100644 index 0000000000000000000000000000000000000000..48d0cb98e3fd8fde32c3ccabd3307cde2949a384 GIT binary patch literal 3644 zcmZ`+cTm&K7X5`zFdzXbp@m)qrG-G~kkEsGd?E-+3q@KeBArkKqzM8_7ZvHfi9r;EpT+e6a@rR5k!` z;`7_h)hGjWj>h_Efc!U#I!iMs3?tsa+8+RzpnnrIxg2pqA?X7Qu^9RdFcZ6qvTF*w zf#QbR5Up()wD7gSJy`_DJs5c>;J{?;Z@hea34Tc;4k+X`@6(Y@(1|vu<}V9^qs`f; zSq3Bo6nhAsmWKp%3g=CG6SSI6f*!S4B0&#?8C@=zFDT3@+}hguRuOW#YdJP+X?5pd zU+y32Uss0W3cpl@JR#rs?%y#())!t6!ml#{+$0=vJ8BLjPL)AmtOEUs(vQ*vMS_@t zQEq=EI{c5!Q%pFuHa>+pjdm2Y0_kAQ#w7E{AxMm*$HWJO!2}i7dSN1>i@5{Z#@xoh z^@U`41=LFYL9dYEqXF8JVfWp!J1K@ZL#!!s%Cdm;jYsQ$1R`m>;?ypPCUCOW!-fyC=~n06R4feEyTi@va|*XTdA%!tGz*i#1@xLuv(0f7y`iv_?ad-(P#` z4$H|LFu3OGrX^l5Ij#sW!zHJ5?tA>2gwZzhPh*ZG49}vwQGa)o4?MLUn2<4EG$nY1Q7bc9 zd#}(GcMMKLd^TZ6`8_-6#Gd4p*%6dcea9DkotYDEWM0_{VCC4=Cy9{`v}jwNBzOrE z6}ImRGXK^kW|G_5d*bLJfhO6>DO{U2d9CHC$wLhnnBH>uCvkjE6pn#tjDhi-c&1&3 zMcx)~CQ!{DeN)mH$e*QZrp&3++J=tz`xC&m@lc$1=CCP2sjw zL_9mGgZNuE+&3?R_{Bq|pi|ut!va;H}I)!*kr{S zUXA3cD&5U*y)DTsW6f*RR5qP^98XK4CnD3K`eS{wg^(GA?OWh5s-VlVv-b3&a;KCd z-9RU%3wDN!FERz4=my(8d(oSivaA*vL^aIU#ljo6B4cd#V9Du_l~h11C0HpYFBbgD z$sDDW^LruU4|~>g2XAoP8Q9)DCMAFNu_CSnA-G zJfo8BI2S$QuwB8^@@V@^ux{TD^M)Hq9cj)d`^^@aGC(U_-h%~GKs;$uIO-7q>+6Ty!P-lqNVTJZ_PPI2*P zW@{jqL`tl{lVbVpJaO`htqF89Ftrous()0)wIvO5@C%o?U`3)#2zvftj%#zewObOe zjSPRpf3}#a`=wnB-mP^JiS@>ZCP#8K?09lTKF2?r%;84nz!VIdrSc`=d2yA$dl|3F zU)23t+p~J%aKFIJX7iimP|50>&N|oj*NCW(DBA<4ppJ8TL`2qfm0mm~j_wW^llSs9 z%){HdPrsL&Xccv^Cg!6(?w+c)iy#g*wmVs_}9@ofBAdvZlKT^TLxQNw?f9{Twtu{G1Z0_*N0Oi z(iz`7swix|me6_2&9sFlXdu^cYbV#p_rx=|X|*U|^H(@Uf=c?FJ82%2Q-o9Z${>jXEuaCWsluS-$r6PNvQrBG9yC$YrUv zGutJcOHC=83o*3e+;zOZes5@73BnLcWpZGgmmhGcQbEHp4Gz@A6v7nbAk}9meLit$ znyR)kXo^^g9Q_({7&e*-$#C9skNqR!Odrrd4RPV4GG#ut;Q|d(sdfq!o7_L>aGI{^v6HmqiQZ@qp@aOC#xpl6nDjc`>sXVfzUw~Yktytx~gsv+D( zbDdeO6BrBF3v8kqXf&&T#*dZwXsmTw?Xn#Hyq!&A=4h%Bo3PP)D5OKNZNQMg#R&R8 zr5S|bD@T9KV`Zr-&RFt4#Rq%!m0OJ?udLl3uXmtf-#g`)Y`7pL_*7>4i;+~GCNsM~ zSeOLVNZ9dZRV*m`M&sY7I(9xPAClgNF9ChI~45`m0*u8jPKsotnuo2^3W>rb2Uhh)E!)cvA>=d~!cdO7!mZ~n z2nOT`wPuw4OqI+1_mSt9KVh$1HwL@Z=*@h=OC~xVeOW_fd1QzmSV?AmElOHuJdxS? z6BXL)`fk~bYKqkrm6_YHWYsS{ zNcerq(Tn-BV2)KJJOJGiD>~A#L-*~SSzj{IJ2IC;=L7xu z`8Q7{bbfT%EsrOM_KpclZ58hEv2O4SC}j`l=}-GTfJ%t+<*tJ{k}k@qDxR{rPx)Q; z+UI$n=;sZdU^#{2!^W}EWfdl=xh4-?3B>$XEN-pfs|J+{ zFHCT*-FRr<7`ykbZMeVe;_7Ga>z=C_7jH_~ylef|Yl+G9^8tOMmkMY79v641loLJz z8x<+W7PNB^MsZiHZk-mA(w|L)4RGev}35Z zSd~=PkK?oV*~*gq#yF)pY?Ayg!@sil=q7sDA%ATRkJ|~Epk0q`in9!3zFaFv@YHh^ zyX;ljvaET?R*uWlt;TJ=H9*dW6RG#>s{he!mw*TkFN&wH$VV+B6tTjOXL1S*(ic-r zjk?gceP6qbW*FR4bg4TZ^IiA0llpi4rCJR~_~hVs>KdAxllClEK17`)^q6R10*gB1 z3T{Meru)6vNE&&uq5UX(F-^3zw7RlmbFt~?VSGlq$+3lfM~?=-G* zha<%?N!~&;VFj85<+(*x*TuQQzVKUj)D4$-&Xe&PUDvGc2!qZ++RhyMH#=w(FQ=oL zoPC3aAA&DoUsbH6#AdNf;6f_`>kF1Gwcl_(hT~GE824D(OVW9uWZ8$y(UF^K>FolB z#owHg?pwCLp9RKs`{jri5?Uf+ifdcvoHfL~>gX2tStt*uKvjX}bOZI|cZaVCIORkh zROqVvxd1R8vKv_3c_bF}c4f**jP2#_qxBZy&hB-N8^6z}g11OF406I3L+I%RCo`u- zg&;ealfIh=WF(x|U|u!j;pD(>k?F$pL_nPv=H;{Nl8c#dOe@h9gdMZxM6}A{g1Rb0 z`W%hg+ewK@-fyr^!wfrZcJbf9D9Pp{A1t&yG(D)R_B5(Bafq?3Pq)85!KaPtNo)jP z%7|Z6T{sVqyLdCp;e~L6;&zNDLs?%V(_I);#xB~2tqdYT@)b!x7Q{1r#2um>^#RWi z7+qTG(qrlk`23wlekqS3CbPSO+@2rU%Dq{QsXBH&?-wSg`HESD7id8{pwMBpXvN7e zK-u6OM9>t03pHm@mB<^hHnQ*@1vee}Nozd(PA literal 0 HcmV?d00001 diff --git a/config/_default/menus/menus.en.toml b/config/_default/menus/menus.en.toml index 327a5777bc2..a4282c53a93 100644 --- a/config/_default/menus/menus.en.toml +++ b/config/_default/menus/menus.en.toml @@ -45,8 +45,14 @@ url = "/functions/" [[docs]] - name = "Variables" + name = "Methods" weight = 80 + identifier = "methods" + url = "/methods/" + +[[docs]] + name = "Variables" + weight = 85 identifier = "variables" url = "/variables/" diff --git a/config/_default/params.toml b/config/_default/params.toml index 3fddf9dbc6e..b41679c61ce 100644 --- a/config/_default/params.toml +++ b/config/_default/params.toml @@ -22,3 +22,6 @@ flex_box_interior_classes = "flex-auto w-100 w-40-l mr3 mb3 bg-white ba b--moon- [social] twitter = "GoHugoIO" + +[render_hooks.link] +errorLevel = 'warning' # ignore (default), warning, or error (fails the build) diff --git a/content/en/about/hugo-and-gdpr.md b/content/en/about/hugo-and-gdpr.md index 85e996f5949..ea588fb3193 100644 --- a/content/en/about/hugo-and-gdpr.md +++ b/content/en/about/hugo-and-gdpr.md @@ -29,7 +29,7 @@ toc: true Below are all privacy settings and their default value. These settings need to be put in your site configuration (e.g. `hugo.toml`). -{{< code-toggle file="hugo" >}} +{{< code-toggle file=hugo >}} [privacy] [privacy.disqus] disable = false @@ -58,7 +58,7 @@ privacyEnhanced = false An example privacy configuration that disables all the relevant services in Hugo. With this configuration, the other settings will not matter. -{{< code-toggle file="hugo" >}} +{{< code-toggle file=hugo >}} [privacy] [privacy.disqus] disable = true @@ -98,7 +98,7 @@ simple **Note:** If you use the _simple mode_ for Instagram and a site styled with Bootstrap 4, you may want to disable the inline styles provided by Hugo: - {{< code-toggle file="hugo" >}} +{{< code-toggle file=hugo >}} [services] [services.instagram] disableInlineCSS = true @@ -114,7 +114,7 @@ simple **Note:** If you use the _simple mode_ for Twitter, you may want to disable the inline styles provided by Hugo: - {{< code-toggle file="hugo" >}} +{{< code-toggle file=hugo >}} [services] [services.twitter] disableInlineCSS = true diff --git a/content/en/about/license.md b/content/en/about/license.md index dc560b33f93..6e9d2ea19a0 100644 --- a/content/en/about/license.md +++ b/content/en/about/license.md @@ -1,160 +1,80 @@ --- title: License -description: Hugo v0.15 and later are released under the Apache 2.0 license. +description: Hugo is released under the Apache 2.0 license. categories: ["about hugo"] -keywords: ["License","apache"] +keywords: ["license","apache"] menu: docs: parent: about weight: 70 weight: 70 -aliases: [/meta/license] -toc: true --- -{{% note %}} -Hugo v0.15 and later are released under the Apache 2.0 license. -Earlier versions of Hugo were released under the [Simple Public License](https://opensource.org/license/simpl-2-0-html/). -{{% /note %}} - -_Version 2.0, January 2004_
- - -*Terms and Conditions for use, reproduction, and distribution* - -## 1. Definitions - -“License” shall mean the terms and conditions for use, reproduction, and -distribution as defined by Sections 1 through 9 of this document. - -“Licensor” shall mean the copyright owner or entity authorized by the copyright -owner that is granting the License. - -“Legal Entity” shall mean the union of the acting entity and all other entities -that control, are controlled by, or are under common control with that entity. -For the purposes of this definition, “control” means **(i)** the power, direct or -indirect, to cause the direction or management of such entity, whether by -contract or otherwise, or **(ii)** ownership of fifty percent (50%) or more of the -outstanding shares, or **(iii)** beneficial ownership of such entity. - -“You” (or “Your”) shall mean an individual or Legal Entity exercising -permissions granted by this License. - -“Source” form shall mean the preferred form for making modifications, including -but not limited to software source code, documentation source, and configuration -files. - -“Object” form shall mean any form resulting from mechanical transformation or -translation of a Source form, including but not limited to compiled object code, -generated documentation, and conversions to other media types. - -“Work” shall mean the work of authorship, whether in Source or Object form, made -available under the License, as indicated by a copyright notice that is included -in or attached to the work (an example is provided in the Appendix below). - -“Derivative Works” shall mean any work, whether in Source or Object form, that -is based on (or derived from) the Work and for which the editorial revisions, -annotations, elaborations, or other modifications represent, as a whole, an -original work of authorship. For the purposes of this License, Derivative Works -shall not include works that remain separable from, or merely link (or bind by -name) to the interfaces of, the Work and Derivative Works thereof. - -“Contribution” shall mean any work of authorship, including the original version -of the Work and any modifications or additions to that Work or Derivative Works -thereof, that is intentionally submitted to Licensor for inclusion in the Work -by the copyright owner or by an individual or Legal Entity authorized to submit -on behalf of the copyright owner. For the purposes of this definition, -“submitted” means any form of electronic, verbal, or written communication sent -to the Licensor or its representatives, including but not limited to -communication on electronic mailing lists, source code control systems, and -issue tracking systems that are managed by, or on behalf of, the Licensor for -the purpose of discussing and improving the Work, but excluding communication -that is conspicuously marked or otherwise designated in writing by the copyright -owner as “Not a Contribution.” - -“Contributor” shall mean Licensor and any individual or Legal Entity on behalf -of whom a Contribution has been received by Licensor and subsequently -incorporated within the Work. - -## 2. Grant of Copyright License - -Subject to the terms and conditions of this License, each Contributor hereby -grants to You a perpetual, worldwide, non-exclusive, no-charge, royalty-free, -irrevocable copyright license to reproduce, prepare Derivative Works of, -publicly display, publicly perform, sublicense, and distribute the Work and such -Derivative Works in Source or Object form. - -## 3. Grant of Patent License - -Subject to the terms and conditions of this License, each Contributor hereby -grants to You a perpetual, worldwide, non-exclusive, no-charge, royalty-free, -irrevocable (except as stated in this section) patent license to make, have -made, use, offer to sell, sell, import, and otherwise transfer the Work, where -such license applies only to those patent claims licensable by such Contributor -that are necessarily infringed by their Contribution(s) alone or by combination -of their Contribution(s) with the Work to which such Contribution(s) was -submitted. If You institute patent litigation against any entity (including a -cross-claim or counterclaim in a lawsuit) alleging that the Work or a -Contribution incorporated within the Work constitutes direct or contributory -patent infringement, then any patent licenses granted to You under this License -for that Work shall terminate as of the date such litigation is filed. - -## 4. Redistribution - -You may reproduce and distribute copies of the Work or Derivative Works thereof -in any medium, with or without modifications, and in Source or Object form, -provided that You meet the following conditions: - -* **(a)** You must give any other recipients of the Work or Derivative Works a copy of -this License; and -* **(b)** You must cause any modified files to carry prominent notices stating that You -changed the files; and -* **\(c)** You must retain, in the Source form of any Derivative Works that You distribute, -all copyright, patent, trademark, and attribution notices from the Source form -of the Work, excluding those notices that do not pertain to any part of the -Derivative Works; and -* **(d)** If the Work includes a “NOTICE” text file as part of its distribution, then any Derivative Works that You distribute must include a readable copy of the attribution notices contained within such NOTICE file, excluding those notices that do not pertain to any part of the Derivative Works, in at least one of the following places: within a NOTICE text file distributed as part of the Derivative Works; within the Source form or documentation, if provided along with the Derivative Works; or, within a display generated by the Derivative Works, if and wherever such third-party notices normally appear. The contents of the NOTICE file are for informational purposes only and do not modify the License. You may add Your own attribution notices within Derivative Works that You distribute, alongside or as an addendum to the NOTICE text from the Work, provided that such additional attribution notices cannot be construed as modifying the License. +## Apache License -You may add Your own copyright statement to Your modifications and may provide additional or different license terms and conditions for use, reproduction, or distribution of Your modifications, or for any such Derivative Works as a whole, provided Your use, reproduction, and distribution of the Work otherwise complies with the conditions stated in this License. -## 5. Submission of Contributions +_Version 2.0, January 2004_ +__ -Unless You explicitly state otherwise, any Contribution intentionally submitted for inclusion in the Work by You to the Licensor shall be under the terms and conditions of this License, without any additional terms or conditions. Notwithstanding the above, nothing herein shall supersede or modify the terms of any separate license agreement you may have executed with Licensor regarding such Contributions. +### Terms and Conditions for use, reproduction, and distribution -## 6. Trademarks +#### 1. Definitions -This License does not grant permission to use the trade names, trademarks, service marks, or product names of the Licensor, except as required for reasonable and customary use in describing the origin of the Work and reproducing the content of the NOTICE file. +“License” shall mean the terms and conditions for use, reproduction, and distribution as defined by Sections 1 through 9 of this document. -## 7. Disclaimer of Warranty +“Licensor” shall mean the copyright owner or entity authorized by the copyright owner that is granting the License. -Unless required by applicable law or agreed to in writing, Licensor provides the Work (and each Contributor provides its Contributions) on an “AS IS” BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied, including, without limitation, any warranties or conditions of TITLE, NON-INFRINGEMENT, MERCHANTABILITY, or FITNESS FOR A PARTICULAR PURPOSE. You are solely responsible for determining the appropriateness of using or redistributing the Work and assume any risks associated with Your exercise of permissions under this License. +“Legal Entity” shall mean the union of the acting entity and all other entities that control, are controlled by, or are under common control with that entity. For the purposes of this definition, “control” means **(i)** the power, direct or indirect, to cause the direction or management of such entity, whether by contract or otherwise, or **(ii)** ownership of fifty percent (50%) or more of the outstanding shares, or **(iii)** beneficial ownership of such entity. -## 8. Limitation of Liability +“You” (or “Your”) shall mean an individual or Legal Entity exercising permissions granted by this License. -In no event and under no legal theory, whether in tort (including negligence), contract, or otherwise, unless required by applicable law (such as deliberate and grossly negligent acts) or agreed to in writing, shall any Contributor be liable to You for damages, including any direct, indirect, special, incidental, or consequential damages of any character arising as a result of this License or out of the use or inability to use the Work (including but not limited to damages for loss of goodwill, work stoppage, computer failure or malfunction, or any and all other commercial damages or losses), even if such Contributor has been advised of the possibility of such damages. +“Source” form shall mean the preferred form for making modifications, including but not limited to software source code, documentation source, and configuration files. -## 9. Accepting Warranty or Additional Liability +“Object” form shall mean any form resulting from mechanical transformation or translation of a Source form, including but not limited to compiled object code, generated documentation, and conversions to other media types. -While redistributing the Work or Derivative Works thereof, You may choose to offer, and charge a fee for, acceptance of support, warranty, indemnity, or other liability obligations and/or rights consistent with this License. However, in accepting such obligations, You may act only on Your own behalf and on Your sole responsibility, not on behalf of any other Contributor, and only if You agree to indemnify, defend, and hold each Contributor harmless for any liability incurred by, or claims asserted against, such Contributor by reason of your accepting any such warranty or additional liability. +“Work” shall mean the work of authorship, whether in Source or Object form, made available under the License, as indicated by a copyright notice that is included in or attached to the work (an example is provided in the Appendix below). + +“Derivative Works” shall mean any work, whether in Source or Object form, that is based on (or derived from) the Work and for which the editorial revisions, annotations, elaborations, or other modifications represent, as a whole, an original work of authorship. For the purposes of this License, Derivative Works shall not include works that remain separable from, or merely link (or bind by name) to the interfaces of, the Work and Derivative Works thereof. + +“Contribution” shall mean any work of authorship, including the original version of the Work and any modifications or additions to that Work or Derivative Works thereof, that is intentionally submitted to Licensor for inclusion in the Work by the copyright owner or by an individual or Legal Entity authorized to submit on behalf of the copyright owner. For the purposes of this definition, “submitted” means any form of electronic, verbal, or written communication sent to the Licensor or its representatives, including but not limited to communication on electronic mailing lists, source code control systems, and issue tracking systems that are managed by, or on behalf of, the Licensor for the purpose of discussing and improving the Work, but excluding communication that is conspicuously marked or otherwise designated in writing by the copyright owner as “Not a Contribution.” + +“Contributor” shall mean Licensor and any individual or Legal Entity on behalf of whom a Contribution has been received by Licensor and subsequently incorporated within the Work. + +#### 2. Grant of Copyright License + +Subject to the terms and conditions of this License, each Contributor hereby grants to You a perpetual, worldwide, non-exclusive, no-charge, royalty-free, irrevocable copyright license to reproduce, prepare Derivative Works of, publicly display, publicly perform, sublicense, and distribute the Work and such Derivative Works in Source or Object form. + +#### 3. Grant of Patent License + +Subject to the terms and conditions of this License, each Contributor hereby grants to You a perpetual, worldwide, non-exclusive, no-charge, royalty-free, irrevocable (except as stated in this section) patent license to make, have made, use, offer to sell, sell, import, and otherwise transfer the Work, where such license applies only to those patent claims licensable by such Contributor that are necessarily infringed by their Contribution(s) alone or by combination of their Contribution(s) with the Work to which such Contribution(s) was submitted. If You institute patent litigation against any entity (including a cross-claim or counterclaim in a lawsuit) alleging that the Work or a Contribution incorporated within the Work constitutes direct or contributory patent infringement, then any patent licenses granted to You under this License for that Work shall terminate as of the date such litigation is filed. -_END OF TERMS AND CONDITIONS_ +#### 4. Redistribution -## APPENDIX: How to apply the Apache License to your work +You may reproduce and distribute copies of the Work or Derivative Works thereof in any medium, with or without modifications, and in Source or Object form, provided that You meet the following conditions: -To apply the Apache License to your work, attach the following boilerplate notice, with the fields enclosed by brackets `[]` replaced with your own identifying information. (Don't include the brackets!) The text should be enclosed in the appropriate comment syntax for the file format. We also recommend that a file or class name and description of purpose be included on the same “printed page” as the copyright notice for easier identification within third-party archives. +* **(a)** You must give any other recipients of the Work or Derivative Works a copy of this License; and +* **(b)** You must cause any modified files to carry prominent notices stating that You changed the files; and +* **(c)** You must retain, in the Source form of any Derivative Works that You distribute, all copyright, patent, trademark, and attribution notices from the Source form of the Work, excluding those notices that do not pertain to any part of the Derivative Works; and +* **(d)** If the Work includes a “NOTICE” text file as part of its distribution, then any Derivative Works that You distribute must include a readable copy of the attribution notices contained within such NOTICE file, excluding those notices that do not pertain to any part of the Derivative Works, in at least one of the following places: within a NOTICE text file distributed as part of the Derivative Works; within the Source form or documentation, if provided along with the Derivative Works; or, within a display generated by the Derivative Works, if and wherever such third-party notices normally appear. The contents of the NOTICE file are for informational purposes only and do not modify the License. You may add Your own attribution notices within Derivative Works that You distribute, alongside or as an addendum to the NOTICE text from the Work, provided that such additional attribution notices cannot be construed as modifying the License. + +You may add Your own copyright statement to Your modifications and may provide additional or different license terms and conditions for use, reproduction, or distribution of Your modifications, or for any such Derivative Works as a whole, provided Your use, reproduction, and distribution of the Work otherwise complies with the conditions stated in this License. + +#### 5. Submission of Contributions + +Unless You explicitly state otherwise, any Contribution intentionally submitted for inclusion in the Work by You to the Licensor shall be under the terms and conditions of this License, without any additional terms or conditions. Notwithstanding the above, nothing herein shall supersede or modify the terms of any separate license agreement you may have executed with Licensor regarding such Contributions. + +#### 6. Trademarks -{{< code file="apache-notice.txt" >}} -Copyright [yyyy] [name of copyright owner] +This License does not grant permission to use the trade names, trademarks, service marks, or product names of the Licensor, except as required for reasonable and customary use in describing the origin of the Work and reproducing the content of the NOTICE file. + +#### 7. Disclaimer of Warranty -Licensed under the Apache License, Version 2.0 (the "License"); -you may not use this file except in compliance with the License. -You may obtain a copy of the License at +Unless required by applicable law or agreed to in writing, Licensor provides the Work (and each Contributor provides its Contributions) on an “AS IS” BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied, including, without limitation, any warranties or conditions of TITLE, NON-INFRINGEMENT, MERCHANTABILITY, or FITNESS FOR A PARTICULAR PURPOSE. You are solely responsible for determining the appropriateness of using or redistributing the Work and assume any risks associated with Your exercise of permissions under this License. - https://www.apache.org/licenses/LICENSE-2.0 +#### 8. Limitation of Liability -Unless required by applicable law or agreed to in writing, software -distributed under the License is distributed on an "AS IS" BASIS, -WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. -See the License for the specific language governing permissions and -limitations under the License. -{{< /code >}} +In no event and under no legal theory, whether in tort (including negligence), contract, or otherwise, unless required by applicable law (such as deliberate and grossly negligent acts) or agreed to in writing, shall any Contributor be liable to You for damages, including any direct, indirect, special, incidental, or consequential damages of any character arising as a result of this License or out of the use or inability to use the Work (including but not limited to damages for loss of goodwill, work stoppage, computer failure or malfunction, or any and all other commercial damages or losses), even if such Contributor has been advised of the possibility of such damages. + +#### 9. Accepting Warranty or Additional Liability + +While redistributing the Work or Derivative Works thereof, You may choose to offer, and charge a fee for, acceptance of support, warranty, indemnity, or other liability obligations and/or rights consistent with this License. However, in accepting such obligations, You may act only on Your own behalf and on Your sole responsibility, not on behalf of any other Contributor, and only if You agree to indemnify, defend, and hold each Contributor harmless for any liability incurred by, or claims asserted against, such Contributor by reason of your accepting any such warranty or additional liability. diff --git a/content/en/content-management/_common/_index.md b/content/en/content-management/_common/_index.md new file mode 100644 index 00000000000..47d5812fba5 --- /dev/null +++ b/content/en/content-management/_common/_index.md @@ -0,0 +1,13 @@ +--- +cascade: + _build: + list: never + publishResources: false + render: never +--- + + diff --git a/layouts/shortcodes/page-kinds.html b/content/en/content-management/_common/page-kinds.md similarity index 65% rename from layouts/shortcodes/page-kinds.html rename to content/en/content-management/_common/page-kinds.md index 968a7a5fbfa..07a53e8e63e 100644 --- a/layouts/shortcodes/page-kinds.html +++ b/content/en/content-management/_common/page-kinds.md @@ -1,3 +1,7 @@ +--- +# Do not remove front matter. +--- + | Kind | Description | Example | |----------------|--------------------------------------------------------------------|-------------------------------------------------------------------------------| | `home` | The landing page for the home page | `/index.html` | @@ -5,3 +9,9 @@ | `section` | The landing page of a given section | `posts` section (`/posts/index.html`) | | `taxonomy` | The landing page for a taxonomy | `tags` taxonomy (`/tags/index.html`) | | `term` | The landing page for one taxonomy's term | term `awesome` in `tags` taxonomy (`/tags/awesome/index.html`) | + +Four other page kinds unrelated to content are `robotsTXT`, `RSS`, `sitemap`, and `404`. Although primarily for internal use, you can specify the name when disabling one or more page kinds on your site. For example: + +{{< code-toggle file=hugo >}} +disableKinds = ['robotsTXT','404'] +{{< /code-toggle >}} diff --git a/content/en/content-management/archetypes.md b/content/en/content-management/archetypes.md index fe460f91ff4..0a33c9da65e 100644 --- a/content/en/content-management/archetypes.md +++ b/content/en/content-management/archetypes.md @@ -19,7 +19,7 @@ A content file consists of [front matter] and markup. The markup is typically ma The `hugo new content` command creates a new file in the `content` directory, using an archetype as a template. This is the default archetype: -{{< code-toggle file="archetypes/default.md" copy=false fm=true >}} +{{< code-toggle file="archetypes/default.md" fm=true >}} title = '{{ replace .File.ContentBaseName `-` ` ` | title }}' date = '{{ .Date }}' draft = true @@ -27,13 +27,13 @@ draft = true When you create new content, Hugo evaluates the [template actions] within the archetype. For example: -```text +```sh hugo new content posts/my-first-post.md ``` With the default archetype shown above, Hugo creates this content file: -{{< code-toggle file="content/posts/my-first-post.md" copy=false fm=true >}} +{{< code-toggle file="content/posts/my-first-post.md" fm=true >}} title = 'My First Post' date = '2023-08-24T11:49:46-07:00' draft = true @@ -53,7 +53,7 @@ Hugo looks for archetypes in the `archetypes` directory in the root of your proj For example, with this command: -```text +```sh hugo new content posts/my-first-post.md ``` @@ -75,7 +75,7 @@ Archetypes receive the following objects and values in [context]: - `.Date` - `.Type` - `.Site` (see [details](/variables/site/)) -- `.File` (see [details](/variables/files/)) +- `.File` (see [details](/variables/file/)) As shown above, the default archetype passes `.File.ContentBaseName` as the argument to the `replace` function when populating the title in front matter. @@ -85,8 +85,7 @@ Although typically used as a front matter template, you can also use an archetyp For example, in a documentation site you might have a section (content type) for functions. Every page within this section should follow the same format: a brief description, the function signature, examples, and notes. We can pre-populate the page to remind content authors of the standard format. - -{{< code file="archetypes/functions.md" copy=false >}} +{{< code file="archetypes/functions.md" >}} --- date: '{{ .Date }}' draft: true @@ -125,17 +124,17 @@ Create an archetype for galleries: ```text archetypes/ ├── galleries/ -│   ├── images/ -│   │   └── .gitkeep -│   └── index.md <-- same format as default.md +│ ├── images/ +│ │ └── .gitkeep +│ └── index.md <-- same format as default.md └── default.md ``` Subdirectories within an archetype must contain at least one file. Without a file, Hugo will not create the subdirectory when you create new content. The name and size of the file are irrelevant. The example above includes a `.gitkeep` file, an empty file commonly used to preserve otherwise empty directories in a Git repository. - To create a new gallery: -```text + +```sh hugo new galleries/bryce-canyon ``` @@ -166,13 +165,13 @@ archetypes/ To create an article using the articles archetype: -```text +```sh hugo new content articles/something.md ``` To create an article using the tutorials archetype: -```text +```sh hugo new content --kind tutorials articles/something.md ``` diff --git a/content/en/content-management/build-options.md b/content/en/content-management/build-options.md index 378a3114447..fb3cca7cba1 100644 --- a/content/en/content-management/build-options.md +++ b/content/en/content-management/build-options.md @@ -51,7 +51,7 @@ If set to `true` (default) the [Bundle's Resources](/content-management/page-bun Setting this to `false` will still publish Resources on demand (when a resource's `.Permalink` or `.RelPermalink` is invoked from the templates) but will skip the others. {{% note %}} -Any page, regardless of their build options, will always be available using the [`.GetPage`](/functions/getpage) methods. +Any page, regardless of their build options, will always be available using the [`.GetPage`](/methods/page/getpage) methods. {{% /note %}} ### Illustrative use cases @@ -60,14 +60,14 @@ Any page, regardless of their build options, will always be available using the Project needs a "Who We Are" content file for front matter and body to be used by the homepage but nowhere else. -{{< code-toggle file="content/who-we-are.md" fm=true copy=false >}} +{{< code-toggle file="content/who-we-are.md" fm=true >}} title: Who we are _build: list: false render: false {{< /code-toggle >}} -{{< code file="layouts/index.html" copy=false >}} +{{< code file="layouts/index.html" >}}

{{ with site.GetPage "who-we-are" }} {{ .Content }} @@ -91,7 +91,7 @@ cascade: list: true # default {{< /code-toggle >}} -{{< code file="layouts/_defaults/testimonials.html" copy=false >}} +{{< code file="layouts/_defaults/testimonials.html" >}}
{{ range first 5 .Pages }}
diff --git a/content/en/content-management/comments.md b/content/en/content-management/comments.md index 39663013b6b..e85ea5a787b 100644 --- a/content/en/content-management/comments.md +++ b/content/en/content-management/comments.md @@ -24,9 +24,8 @@ Hugo comes with all the code you need to load Disqus into your templates. Before Disqus comments require you set a single value in your [site's configuration file][configuration] like so: -{{< code-toggle file="hugo" >}} -[services.disqus] -shortname = 'your-disqus-shortname' +{{< code-toggle file=hugo >}} +disqusShortname = "yourDisqusShortname" {{}} For many websites, this is enough configuration. However, you also have the option to set the following in the [front matter] of a single content file: diff --git a/content/en/content-management/cross-references.md b/content/en/content-management/cross-references.md index c989cc560d9..32e1d96edc3 100644 --- a/content/en/content-management/cross-references.md +++ b/content/en/content-management/cross-references.md @@ -35,7 +35,6 @@ The `ref` and `relref` shortcodes require a single parameter: the path to a cont The pages can be referenced as follows: - ```text {{}} // <- From pages/document1.md, relative path {{}} @@ -138,7 +137,7 @@ produces this HTML: ## Ref and RelRef Configuration -The behavior can, since Hugo 0.45, be configured in `hugo.toml`: +The behavior can be configured in `hugo.toml`: refLinksErrorLevel ("ERROR") : When using `ref` or `relref` to resolve page links and a link cannot resolved, it will be logged with this log level. Valid values are `ERROR` (default) or `WARNING`. Any `ERROR` will fail the build (`exit -1`). @@ -146,7 +145,6 @@ refLinksErrorLevel ("ERROR") refLinksNotFoundURL : URL to be used as a placeholder when a page reference cannot be found in `ref` or `relref`. Is used as-is. - [lists]: /templates/lists/ [output formats]: /templates/output-formats/ [shortcode]: /content-management/shortcodes/ diff --git a/content/en/content-management/diagrams.md b/content/en/content-management/diagrams.md index c0b2349b0b2..886b4c298f5 100644 --- a/content/en/content-management/diagrams.md +++ b/content/en/content-management/diagrams.md @@ -165,7 +165,6 @@ Created from └─Fedora ``` - ### Sequence diagram @@ -186,7 +185,6 @@ Created from ``` - ### Flowchart @@ -232,7 +230,6 @@ Created from ``` - ### Table diff --git a/content/en/content-management/emoji-shortcodes.md b/content/en/content-management/emoji-shortcodes.md new file mode 100644 index 00000000000..39ed28ee7d5 --- /dev/null +++ b/content/en/content-management/emoji-shortcodes.md @@ -0,0 +1,1621 @@ +--- +title: Emoji shortcodes +description: Include emoji shortcodes in your markdown, or use the emojify function in your templates. +categories: [content management] +keywords: [emoji] +menu: + docs: + parent: content-management + weight: 250 +toc: true +weight: 250 +--- + +With emoji processing enabled, this markdown: + +```md +Hello! :wave: +``` + +Is rendered to: + +```html +Hello! 👋 +``` + +And in your browser... Hello! :wave: + +To enable this capability in your site configuration: + +{{< code-toggle file=hugo >}} +enableEmoji = true +{{< /code-toggle >}} + +To process an emoji shortcode from within a template, use the [`emojify`] function. + +[`emojify`]: /functions/transform/emojify + +## Introduction + +This cheat sheet was automatically generated from [GitHub Emoji API] and [Unicode Full Emoji List]. Specials thanks to [@ikatyang] for making [this list] available to the open source community. + +GitHub [custom emoji] are not supported. + +[custom emoji]: #github-custom-emoji +[@ikatyang]: https://github.com/ikatyang +[github emoji api]: https://api.github.com/emojis +[unicode full emoji list]: https://unicode.org/emoji/charts/full-emoji-list.html +[this list]: https://github.com/ikatyang/emoji-cheat-sheet/blob/master/readme.md + + + +## Smileys & Emotion + +- [Face Smiling](#face-smiling) +- [Face Affection](#face-affection) +- [Face Tongue](#face-tongue) +- [Face Hand](#face-hand) +- [Face Neutral Skeptical](#face-neutral-skeptical) +- [Face Sleepy](#face-sleepy) +- [Face Unwell](#face-unwell) +- [Face Hat](#face-hat) +- [Face Glasses](#face-glasses) +- [Face Concerned](#face-concerned) +- [Face Negative](#face-negative) +- [Face Costume](#face-costume) +- [Cat Face](#cat-face) +- [Monkey Face](#monkey-face) +- [Heart](#heart) +- [Emotion](#emotion) + +### Face Smiling + +| | ico | shortcode | ico | shortcode | | +| - | :-: | - | :-: | - | - | +| [top](#introduction) | :grinning: | `:grinning:` | :smiley: | `:smiley:` | [top](#introduction) | +| [top](#introduction) | :smile: | `:smile:` | :grin: | `:grin:` | [top](#introduction) | +| [top](#introduction) | :laughing: | `:laughing:`
`:satisfied:` | :sweat_smile: | `:sweat_smile:` | [top](#introduction) | +| [top](#introduction) | :rofl: | `:rofl:` | :joy: | `:joy:` | [top](#introduction) | +| [top](#introduction) | :slightly_smiling_face: | `:slightly_smiling_face:` | :upside_down_face: | `:upside_down_face:` | [top](#introduction) | +| [top](#introduction) | :wink: | `:wink:` | :blush: | `:blush:` | [top](#introduction) | +| [top](#introduction) | :innocent: | `:innocent:` | | | [top](#introduction) | + +### Face Affection + +| | ico | shortcode | ico | shortcode | | +| - | :-: | - | :-: | - | - | +| [top](#introduction) | :smiling_face_with_three_hearts: | `:smiling_face_with_three_hearts:` | :heart_eyes: | `:heart_eyes:` | [top](#introduction) | +| [top](#introduction) | :star_struck: | `:star_struck:` | :kissing_heart: | `:kissing_heart:` | [top](#introduction) | +| [top](#introduction) | :kissing: | `:kissing:` | :relaxed: | `:relaxed:` | [top](#introduction) | +| [top](#introduction) | :kissing_closed_eyes: | `:kissing_closed_eyes:` | :kissing_smiling_eyes: | `:kissing_smiling_eyes:` | [top](#introduction) | +| [top](#introduction) | :smiling_face_with_tear: | `:smiling_face_with_tear:` | | | [top](#introduction) | + +### Face Tongue + +| | ico | shortcode | ico | shortcode | | +| - | :-: | - | :-: | - | - | +| [top](#introduction) | :yum: | `:yum:` | :stuck_out_tongue: | `:stuck_out_tongue:` | [top](#introduction) | +| [top](#introduction) | :stuck_out_tongue_winking_eye: | `:stuck_out_tongue_winking_eye:` | :zany_face: | `:zany_face:` | [top](#introduction) | +| [top](#introduction) | :stuck_out_tongue_closed_eyes: | `:stuck_out_tongue_closed_eyes:` | :money_mouth_face: | `:money_mouth_face:` | [top](#introduction) | + +### Face Hand + +| | ico | shortcode | ico | shortcode | | +| - | :-: | - | :-: | - | - | +| [top](#introduction) | :hugs: | `:hugs:` | :hand_over_mouth: | `:hand_over_mouth:` | [top](#introduction) | +| [top](#introduction) | :shushing_face: | `:shushing_face:` | :thinking: | `:thinking:` | [top](#introduction) | + +### Face Neutral Skeptical + +| | ico | shortcode | ico | shortcode | | +| - | :-: | - | :-: | - | - | +| [top](#introduction) | :zipper_mouth_face: | `:zipper_mouth_face:` | :raised_eyebrow: | `:raised_eyebrow:` | [top](#introduction) | +| [top](#introduction) | :neutral_face: | `:neutral_face:` | :expressionless: | `:expressionless:` | [top](#introduction) | +| [top](#introduction) | :no_mouth: | `:no_mouth:` | :face_in_clouds: | `:face_in_clouds:` | [top](#introduction) | +| [top](#introduction) | :smirk: | `:smirk:` | :unamused: | `:unamused:` | [top](#introduction) | +| [top](#introduction) | :roll_eyes: | `:roll_eyes:` | :grimacing: | `:grimacing:` | [top](#introduction) | +| [top](#introduction) | :face_exhaling: | `:face_exhaling:` | :lying_face: | `:lying_face:` | [top](#introduction) | + +### Face Sleepy + +| | ico | shortcode | ico | shortcode | | +| - | :-: | - | :-: | - | - | +| [top](#introduction) | :relieved: | `:relieved:` | :pensive: | `:pensive:` | [top](#introduction) | +| [top](#introduction) | :sleepy: | `:sleepy:` | :drooling_face: | `:drooling_face:` | [top](#introduction) | +| [top](#introduction) | :sleeping: | `:sleeping:` | | | [top](#introduction) | + +### Face Unwell + +| | ico | shortcode | ico | shortcode | | +| - | :-: | - | :-: | - | - | +| [top](#introduction) | :mask: | `:mask:` | :face_with_thermometer: | `:face_with_thermometer:` | [top](#introduction) | +| [top](#introduction) | :face_with_head_bandage: | `:face_with_head_bandage:` | :nauseated_face: | `:nauseated_face:` | [top](#introduction) | +| [top](#introduction) | :vomiting_face: | `:vomiting_face:` | :sneezing_face: | `:sneezing_face:` | [top](#introduction) | +| [top](#introduction) | :hot_face: | `:hot_face:` | :cold_face: | `:cold_face:` | [top](#introduction) | +| [top](#introduction) | :woozy_face: | `:woozy_face:` | :dizzy_face: | `:dizzy_face:` | [top](#introduction) | +| [top](#introduction) | :face_with_spiral_eyes: | `:face_with_spiral_eyes:` | :exploding_head: | `:exploding_head:` | [top](#introduction) | + +### Face Hat + +| | ico | shortcode | ico | shortcode | | +| - | :-: | - | :-: | - | - | +| [top](#introduction) | :cowboy_hat_face: | `:cowboy_hat_face:` | :partying_face: | `:partying_face:` | [top](#introduction) | +| [top](#introduction) | :disguised_face: | `:disguised_face:` | | | [top](#introduction) | + +### Face Glasses + +| | ico | shortcode | ico | shortcode | | +| - | :-: | - | :-: | - | - | +| [top](#introduction) | :sunglasses: | `:sunglasses:` | :nerd_face: | `:nerd_face:` | [top](#introduction) | +| [top](#introduction) | :monocle_face: | `:monocle_face:` | | | [top](#introduction) | + +### Face Concerned + +| | ico | shortcode | ico | shortcode | | +| - | :-: | - | :-: | - | - | +| [top](#introduction) | :confused: | `:confused:` | :worried: | `:worried:` | [top](#introduction) | +| [top](#introduction) | :slightly_frowning_face: | `:slightly_frowning_face:` | :frowning_face: | `:frowning_face:` | [top](#introduction) | +| [top](#introduction) | :open_mouth: | `:open_mouth:` | :hushed: | `:hushed:` | [top](#introduction) | +| [top](#introduction) | :astonished: | `:astonished:` | :flushed: | `:flushed:` | [top](#introduction) | +| [top](#introduction) | :pleading_face: | `:pleading_face:` | :frowning: | `:frowning:` | [top](#introduction) | +| [top](#introduction) | :anguished: | `:anguished:` | :fearful: | `:fearful:` | [top](#introduction) | +| [top](#introduction) | :cold_sweat: | `:cold_sweat:` | :disappointed_relieved: | `:disappointed_relieved:` | [top](#introduction) | +| [top](#introduction) | :cry: | `:cry:` | :sob: | `:sob:` | [top](#introduction) | +| [top](#introduction) | :scream: | `:scream:` | :confounded: | `:confounded:` | [top](#introduction) | +| [top](#introduction) | :persevere: | `:persevere:` | :disappointed: | `:disappointed:` | [top](#introduction) | +| [top](#introduction) | :sweat: | `:sweat:` | :weary: | `:weary:` | [top](#introduction) | +| [top](#introduction) | :tired_face: | `:tired_face:` | :yawning_face: | `:yawning_face:` | [top](#introduction) | + +### Face Negative + +| | ico | shortcode | ico | shortcode | | +| - | :-: | - | :-: | - | - | +| [top](#introduction) | :triumph: | `:triumph:` | :pout: | `:pout:`
`:rage:` | [top](#introduction) | +| [top](#introduction) | :angry: | `:angry:` | :cursing_face: | `:cursing_face:` | [top](#introduction) | +| [top](#introduction) | :smiling_imp: | `:smiling_imp:` | :imp: | `:imp:` | [top](#introduction) | +| [top](#introduction) | :skull: | `:skull:` | :skull_and_crossbones: | `:skull_and_crossbones:` | [top](#introduction) | + +### Face Costume + +| | ico | shortcode | ico | shortcode | | +| - | :-: | - | :-: | - | - | +| [top](#introduction) | :hankey: | `:hankey:`
`:poop:`
`:shit:` | :clown_face: | `:clown_face:` | [top](#introduction) | +| [top](#introduction) | :japanese_ogre: | `:japanese_ogre:` | :japanese_goblin: | `:japanese_goblin:` | [top](#introduction) | +| [top](#introduction) | :ghost: | `:ghost:` | :alien: | `:alien:` | [top](#introduction) | +| [top](#introduction) | :space_invader: | `:space_invader:` | :robot: | `:robot:` | [top](#introduction) | + +### Cat Face + +| | ico | shortcode | ico | shortcode | | +| - | :-: | - | :-: | - | - | +| [top](#introduction) | :smiley_cat: | `:smiley_cat:` | :smile_cat: | `:smile_cat:` | [top](#introduction) | +| [top](#introduction) | :joy_cat: | `:joy_cat:` | :heart_eyes_cat: | `:heart_eyes_cat:` | [top](#introduction) | +| [top](#introduction) | :smirk_cat: | `:smirk_cat:` | :kissing_cat: | `:kissing_cat:` | [top](#introduction) | +| [top](#introduction) | :scream_cat: | `:scream_cat:` | :crying_cat_face: | `:crying_cat_face:` | [top](#introduction) | +| [top](#introduction) | :pouting_cat: | `:pouting_cat:` | | | [top](#introduction) | + +### Monkey Face + +| | ico | shortcode | ico | shortcode | | +| - | :-: | - | :-: | - | - | +| [top](#introduction) | :see_no_evil: | `:see_no_evil:` | :hear_no_evil: | `:hear_no_evil:` | [top](#introduction) | +| [top](#introduction) | :speak_no_evil: | `:speak_no_evil:` | | | [top](#introduction) | + +### Heart + +| | ico | shortcode | ico | shortcode | | +| - | :-: | - | :-: | - | - | +| [top](#introduction) | :love_letter: | `:love_letter:` | :cupid: | `:cupid:` | [top](#introduction) | +| [top](#introduction) | :gift_heart: | `:gift_heart:` | :sparkling_heart: | `:sparkling_heart:` | [top](#introduction) | +| [top](#introduction) | :heartpulse: | `:heartpulse:` | :heartbeat: | `:heartbeat:` | [top](#introduction) | +| [top](#introduction) | :revolving_hearts: | `:revolving_hearts:` | :two_hearts: | `:two_hearts:` | [top](#introduction) | +| [top](#introduction) | :heart_decoration: | `:heart_decoration:` | :heavy_heart_exclamation: | `:heavy_heart_exclamation:` | [top](#introduction) | +| [top](#introduction) | :broken_heart: | `:broken_heart:` | :heart_on_fire: | `:heart_on_fire:` | [top](#introduction) | +| [top](#introduction) | :mending_heart: | `:mending_heart:` | :heart: | `:heart:` | [top](#introduction) | +| [top](#introduction) | :orange_heart: | `:orange_heart:` | :yellow_heart: | `:yellow_heart:` | [top](#introduction) | +| [top](#introduction) | :green_heart: | `:green_heart:` | :blue_heart: | `:blue_heart:` | [top](#introduction) | +| [top](#introduction) | :purple_heart: | `:purple_heart:` | :brown_heart: | `:brown_heart:` | [top](#introduction) | +| [top](#introduction) | :black_heart: | `:black_heart:` | :white_heart: | `:white_heart:` | [top](#introduction) | + +### Emotion + +| | ico | shortcode | ico | shortcode | | +| - | :-: | - | :-: | - | - | +| [top](#introduction) | :kiss: | `:kiss:` | :100: | `:100:` | [top](#introduction) | +| [top](#introduction) | :anger: | `:anger:` | :boom: | `:boom:`
`:collision:` | [top](#introduction) | +| [top](#introduction) | :dizzy: | `:dizzy:` | :sweat_drops: | `:sweat_drops:` | [top](#introduction) | +| [top](#introduction) | :dash: | `:dash:` | :hole: | `:hole:` | [top](#introduction) | +| [top](#introduction) | :speech_balloon: | `:speech_balloon:` | :eye_speech_bubble: | `:eye_speech_bubble:` | [top](#introduction) | +| [top](#introduction) | :left_speech_bubble: | `:left_speech_bubble:` | :right_anger_bubble: | `:right_anger_bubble:` | [top](#introduction) | +| [top](#introduction) | :thought_balloon: | `:thought_balloon:` | :zzz: | `:zzz:` | [top](#introduction) | + +## People & Body + +- [Hand Fingers Open](#hand-fingers-open) +- [Hand Fingers Partial](#hand-fingers-partial) +- [Hand Single Finger](#hand-single-finger) +- [Hand Fingers Closed](#hand-fingers-closed) +- [Hands](#hands) +- [Hand Prop](#hand-prop) +- [Body Parts](#body-parts) +- [Person](#person) +- [Person Gesture](#person-gesture) +- [Person Role](#person-role) +- [Person Fantasy](#person-fantasy) +- [Person Activity](#person-activity) +- [Person Sport](#person-sport) +- [Person Resting](#person-resting) +- [Family](#family) +- [Person Symbol](#person-symbol) + +### Hand Fingers Open + +| | ico | shortcode | ico | shortcode | | +| - | :-: | - | :-: | - | - | +| [top](#people--body) | :wave: | `:wave:` | :raised_back_of_hand: | `:raised_back_of_hand:` | [top](#introduction) | +| [top](#people--body) | :raised_hand_with_fingers_splayed: | `:raised_hand_with_fingers_splayed:` | :hand: | `:hand:`
`:raised_hand:` | [top](#introduction) | +| [top](#people--body) | :vulcan_salute: | `:vulcan_salute:` | | | [top](#introduction) | + +### Hand Fingers Partial + +| | ico | shortcode | ico | shortcode | | +| - | :-: | - | :-: | - | - | +| [top](#people--body) | :ok_hand: | `:ok_hand:` | :pinched_fingers: | `:pinched_fingers:` | [top](#introduction) | +| [top](#people--body) | :pinching_hand: | `:pinching_hand:` | :v: | `:v:` | [top](#introduction) | +| [top](#people--body) | :crossed_fingers: | `:crossed_fingers:` | :love_you_gesture: | `:love_you_gesture:` | [top](#introduction) | +| [top](#people--body) | :metal: | `:metal:` | :call_me_hand: | `:call_me_hand:` | [top](#introduction) | + +### Hand Single Finger + +| | ico | shortcode | ico | shortcode | | +| - | :-: | - | :-: | - | - | +| [top](#people--body) | :point_left: | `:point_left:` | :point_right: | `:point_right:` | [top](#introduction) | +| [top](#people--body) | :point_up_2: | `:point_up_2:` | :fu: | `:fu:`
`:middle_finger:` | [top](#introduction) | +| [top](#people--body) | :point_down: | `:point_down:` | :point_up: | `:point_up:` | [top](#introduction) | + +### Hand Fingers Closed + +| | ico | shortcode | ico | shortcode | | +| - | :-: | - | :-: | - | - | +| [top](#people--body) | :+1: | `:+1:`
`:thumbsup:` | :-1: | `:-1:`
`:thumbsdown:` | [top](#introduction) | +| [top](#people--body) | :fist: | `:fist:`
`:fist_raised:` | :facepunch: | `:facepunch:`
`:fist_oncoming:`
`:punch:` | [top](#introduction) | +| [top](#people--body) | :fist_left: | `:fist_left:` | :fist_right: | `:fist_right:` | [top](#introduction) | + +### Hands + +| | ico | shortcode | ico | shortcode | | +| - | :-: | - | :-: | - | - | +| [top](#people--body) | :clap: | `:clap:` | :raised_hands: | `:raised_hands:` | [top](#introduction) | +| [top](#people--body) | :open_hands: | `:open_hands:` | :palms_up_together: | `:palms_up_together:` | [top](#introduction) | +| [top](#people--body) | :handshake: | `:handshake:` | :pray: | `:pray:` | [top](#introduction) | + +### Hand Prop + +| | ico | shortcode | ico | shortcode | | +| - | :-: | - | :-: | - | - | +| [top](#people--body) | :writing_hand: | `:writing_hand:` | :nail_care: | `:nail_care:` | [top](#introduction) | +| [top](#people--body) | :selfie: | `:selfie:` | | | [top](#introduction) | + +### Body Parts + +| | ico | shortcode | ico | shortcode | | +| - | :-: | - | :-: | - | - | +| [top](#people--body) | :muscle: | `:muscle:` | :mechanical_arm: | `:mechanical_arm:` | [top](#introduction) | +| [top](#people--body) | :mechanical_leg: | `:mechanical_leg:` | :leg: | `:leg:` | [top](#introduction) | +| [top](#people--body) | :foot: | `:foot:` | :ear: | `:ear:` | [top](#introduction) | +| [top](#people--body) | :ear_with_hearing_aid: | `:ear_with_hearing_aid:` | :nose: | `:nose:` | [top](#introduction) | +| [top](#people--body) | :brain: | `:brain:` | :anatomical_heart: | `:anatomical_heart:` | [top](#introduction) | +| [top](#people--body) | :lungs: | `:lungs:` | :tooth: | `:tooth:` | [top](#introduction) | +| [top](#people--body) | :bone: | `:bone:` | :eyes: | `:eyes:` | [top](#introduction) | +| [top](#people--body) | :eye: | `:eye:` | :tongue: | `:tongue:` | [top](#introduction) | +| [top](#people--body) | :lips: | `:lips:` | | | [top](#introduction) | + +### Person + +| | ico | shortcode | ico | shortcode | | +| - | :-: | - | :-: | - | - | +| [top](#people--body) | :baby: | `:baby:` | :child: | `:child:` | [top](#introduction) | +| [top](#people--body) | :boy: | `:boy:` | :girl: | `:girl:` | [top](#introduction) | +| [top](#people--body) | :adult: | `:adult:` | :blond_haired_person: | `:blond_haired_person:` | [top](#introduction) | +| [top](#people--body) | :man: | `:man:` | :bearded_person: | `:bearded_person:` | [top](#introduction) | +| [top](#people--body) | :man_beard: | `:man_beard:` | :woman_beard: | `:woman_beard:` | [top](#introduction) | +| [top](#people--body) | :red_haired_man: | `:red_haired_man:` | :curly_haired_man: | `:curly_haired_man:` | [top](#introduction) | +| [top](#people--body) | :white_haired_man: | `:white_haired_man:` | :bald_man: | `:bald_man:` | [top](#introduction) | +| [top](#people--body) | :woman: | `:woman:` | :red_haired_woman: | `:red_haired_woman:` | [top](#introduction) | +| [top](#people--body) | :person_red_hair: | `:person_red_hair:` | :curly_haired_woman: | `:curly_haired_woman:` | [top](#introduction) | +| [top](#people--body) | :person_curly_hair: | `:person_curly_hair:` | :white_haired_woman: | `:white_haired_woman:` | [top](#introduction) | +| [top](#people--body) | :person_white_hair: | `:person_white_hair:` | :bald_woman: | `:bald_woman:` | [top](#introduction) | +| [top](#people--body) | :person_bald: | `:person_bald:` | :blond_haired_woman: | `:blond_haired_woman:`
`:blonde_woman:` | [top](#introduction) | +| [top](#people--body) | :blond_haired_man: | `:blond_haired_man:` | :older_adult: | `:older_adult:` | [top](#introduction) | +| [top](#people--body) | :older_man: | `:older_man:` | :older_woman: | `:older_woman:` | [top](#introduction) | + +### Person Gesture + +| | ico | shortcode | ico | shortcode | | +| - | :-: | - | :-: | - | - | +| [top](#people--body) | :frowning_person: | `:frowning_person:` | :frowning_man: | `:frowning_man:` | [top](#introduction) | +| [top](#people--body) | :frowning_woman: | `:frowning_woman:` | :pouting_face: | `:pouting_face:` | [top](#introduction) | +| [top](#people--body) | :pouting_man: | `:pouting_man:` | :pouting_woman: | `:pouting_woman:` | [top](#introduction) | +| [top](#people--body) | :no_good: | `:no_good:` | :ng_man: | `:ng_man:`
`:no_good_man:` | [top](#introduction) | +| [top](#people--body) | :ng_woman: | `:ng_woman:`
`:no_good_woman:` | :ok_person: | `:ok_person:` | [top](#introduction) | +| [top](#people--body) | :ok_man: | `:ok_man:` | :ok_woman: | `:ok_woman:` | [top](#introduction) | +| [top](#people--body) | :information_desk_person: | `:information_desk_person:`
`:tipping_hand_person:` | :sassy_man: | `:sassy_man:`
`:tipping_hand_man:` | [top](#introduction) | +| [top](#people--body) | :sassy_woman: | `:sassy_woman:`
`:tipping_hand_woman:` | :raising_hand: | `:raising_hand:` | [top](#introduction) | +| [top](#people--body) | :raising_hand_man: | `:raising_hand_man:` | :raising_hand_woman: | `:raising_hand_woman:` | [top](#introduction) | +| [top](#people--body) | :deaf_person: | `:deaf_person:` | :deaf_man: | `:deaf_man:` | [top](#introduction) | +| [top](#people--body) | :deaf_woman: | `:deaf_woman:` | :bow: | `:bow:` | [top](#introduction) | +| [top](#people--body) | :bowing_man: | `:bowing_man:` | :bowing_woman: | `:bowing_woman:` | [top](#introduction) | +| [top](#people--body) | :facepalm: | `:facepalm:` | :man_facepalming: | `:man_facepalming:` | [top](#introduction) | +| [top](#people--body) | :woman_facepalming: | `:woman_facepalming:` | :shrug: | `:shrug:` | [top](#introduction) | +| [top](#people--body) | :man_shrugging: | `:man_shrugging:` | :woman_shrugging: | `:woman_shrugging:` | [top](#introduction) | + +### Person Role + +| | ico | shortcode | ico | shortcode | | +| - | :-: | - | :-: | - | - | +| [top](#people--body) | :health_worker: | `:health_worker:` | :man_health_worker: | `:man_health_worker:` | [top](#introduction) | +| [top](#people--body) | :woman_health_worker: | `:woman_health_worker:` | :student: | `:student:` | [top](#introduction) | +| [top](#people--body) | :man_student: | `:man_student:` | :woman_student: | `:woman_student:` | [top](#introduction) | +| [top](#people--body) | :teacher: | `:teacher:` | :man_teacher: | `:man_teacher:` | [top](#introduction) | +| [top](#people--body) | :woman_teacher: | `:woman_teacher:` | :judge: | `:judge:` | [top](#introduction) | +| [top](#people--body) | :man_judge: | `:man_judge:` | :woman_judge: | `:woman_judge:` | [top](#introduction) | +| [top](#people--body) | :farmer: | `:farmer:` | :man_farmer: | `:man_farmer:` | [top](#introduction) | +| [top](#people--body) | :woman_farmer: | `:woman_farmer:` | :cook: | `:cook:` | [top](#introduction) | +| [top](#people--body) | :man_cook: | `:man_cook:` | :woman_cook: | `:woman_cook:` | [top](#introduction) | +| [top](#people--body) | :mechanic: | `:mechanic:` | :man_mechanic: | `:man_mechanic:` | [top](#introduction) | +| [top](#people--body) | :woman_mechanic: | `:woman_mechanic:` | :factory_worker: | `:factory_worker:` | [top](#introduction) | +| [top](#people--body) | :man_factory_worker: | `:man_factory_worker:` | :woman_factory_worker: | `:woman_factory_worker:` | [top](#introduction) | +| [top](#people--body) | :office_worker: | `:office_worker:` | :man_office_worker: | `:man_office_worker:` | [top](#introduction) | +| [top](#people--body) | :woman_office_worker: | `:woman_office_worker:` | :scientist: | `:scientist:` | [top](#introduction) | +| [top](#people--body) | :man_scientist: | `:man_scientist:` | :woman_scientist: | `:woman_scientist:` | [top](#introduction) | +| [top](#people--body) | :technologist: | `:technologist:` | :man_technologist: | `:man_technologist:` | [top](#introduction) | +| [top](#people--body) | :woman_technologist: | `:woman_technologist:` | :singer: | `:singer:` | [top](#introduction) | +| [top](#people--body) | :man_singer: | `:man_singer:` | :woman_singer: | `:woman_singer:` | [top](#introduction) | +| [top](#people--body) | :artist: | `:artist:` | :man_artist: | `:man_artist:` | [top](#introduction) | +| [top](#people--body) | :woman_artist: | `:woman_artist:` | :pilot: | `:pilot:` | [top](#introduction) | +| [top](#people--body) | :man_pilot: | `:man_pilot:` | :woman_pilot: | `:woman_pilot:` | [top](#introduction) | +| [top](#people--body) | :astronaut: | `:astronaut:` | :man_astronaut: | `:man_astronaut:` | [top](#introduction) | +| [top](#people--body) | :woman_astronaut: | `:woman_astronaut:` | :firefighter: | `:firefighter:` | [top](#introduction) | +| [top](#people--body) | :man_firefighter: | `:man_firefighter:` | :woman_firefighter: | `:woman_firefighter:` | [top](#introduction) | +| [top](#people--body) | :cop: | `:cop:`
`:police_officer:` | :policeman: | `:policeman:` | [top](#introduction) | +| [top](#people--body) | :policewoman: | `:policewoman:` | :detective: | `:detective:` | [top](#introduction) | +| [top](#people--body) | :male_detective: | `:male_detective:` | :female_detective: | `:female_detective:` | [top](#introduction) | +| [top](#people--body) | :guard: | `:guard:` | :guardsman: | `:guardsman:` | [top](#introduction) | +| [top](#people--body) | :guardswoman: | `:guardswoman:` | :ninja: | `:ninja:` | [top](#introduction) | +| [top](#people--body) | :construction_worker: | `:construction_worker:` | :construction_worker_man: | `:construction_worker_man:` | [top](#introduction) | +| [top](#people--body) | :construction_worker_woman: | `:construction_worker_woman:` | :prince: | `:prince:` | [top](#introduction) | +| [top](#people--body) | :princess: | `:princess:` | :person_with_turban: | `:person_with_turban:` | [top](#introduction) | +| [top](#people--body) | :man_with_turban: | `:man_with_turban:` | :woman_with_turban: | `:woman_with_turban:` | [top](#introduction) | +| [top](#people--body) | :man_with_gua_pi_mao: | `:man_with_gua_pi_mao:` | :woman_with_headscarf: | `:woman_with_headscarf:` | [top](#introduction) | +| [top](#people--body) | :person_in_tuxedo: | `:person_in_tuxedo:` | :man_in_tuxedo: | `:man_in_tuxedo:` | [top](#introduction) | +| [top](#people--body) | :woman_in_tuxedo: | `:woman_in_tuxedo:` | :person_with_veil: | `:person_with_veil:` | [top](#introduction) | +| [top](#people--body) | :man_with_veil: | `:man_with_veil:` | :bride_with_veil: | `:bride_with_veil:`
`:woman_with_veil:` | [top](#introduction) | +| [top](#people--body) | :pregnant_woman: | `:pregnant_woman:` | :breast_feeding: | `:breast_feeding:` | [top](#introduction) | +| [top](#people--body) | :woman_feeding_baby: | `:woman_feeding_baby:` | :man_feeding_baby: | `:man_feeding_baby:` | [top](#introduction) | +| [top](#people--body) | :person_feeding_baby: | `:person_feeding_baby:` | | | [top](#introduction) | + +### Person Fantasy + +| | ico | shortcode | ico | shortcode | | +| - | :-: | - | :-: | - | - | +| [top](#people--body) | :angel: | `:angel:` | :santa: | `:santa:` | [top](#introduction) | +| [top](#people--body) | :mrs_claus: | `:mrs_claus:` | :mx_claus: | `:mx_claus:` | [top](#introduction) | +| [top](#people--body) | :superhero: | `:superhero:` | :superhero_man: | `:superhero_man:` | [top](#introduction) | +| [top](#people--body) | :superhero_woman: | `:superhero_woman:` | :supervillain: | `:supervillain:` | [top](#introduction) | +| [top](#people--body) | :supervillain_man: | `:supervillain_man:` | :supervillain_woman: | `:supervillain_woman:` | [top](#introduction) | +| [top](#people--body) | :mage: | `:mage:` | :mage_man: | `:mage_man:` | [top](#introduction) | +| [top](#people--body) | :mage_woman: | `:mage_woman:` | :fairy: | `:fairy:` | [top](#introduction) | +| [top](#people--body) | :fairy_man: | `:fairy_man:` | :fairy_woman: | `:fairy_woman:` | [top](#introduction) | +| [top](#people--body) | :vampire: | `:vampire:` | :vampire_man: | `:vampire_man:` | [top](#introduction) | +| [top](#people--body) | :vampire_woman: | `:vampire_woman:` | :merperson: | `:merperson:` | [top](#introduction) | +| [top](#people--body) | :merman: | `:merman:` | :mermaid: | `:mermaid:` | [top](#introduction) | +| [top](#people--body) | :elf: | `:elf:` | :elf_man: | `:elf_man:` | [top](#introduction) | +| [top](#people--body) | :elf_woman: | `:elf_woman:` | :genie: | `:genie:` | [top](#introduction) | +| [top](#people--body) | :genie_man: | `:genie_man:` | :genie_woman: | `:genie_woman:` | [top](#introduction) | +| [top](#people--body) | :zombie: | `:zombie:` | :zombie_man: | `:zombie_man:` | [top](#introduction) | +| [top](#people--body) | :zombie_woman: | `:zombie_woman:` | | | [top](#introduction) | + +### Person Activity + +| | ico | shortcode | ico | shortcode | | +| - | :-: | - | :-: | - | - | +| [top](#people--body) | :massage: | `:massage:` | :massage_man: | `:massage_man:` | [top](#introduction) | +| [top](#people--body) | :massage_woman: | `:massage_woman:` | :haircut: | `:haircut:` | [top](#introduction) | +| [top](#people--body) | :haircut_man: | `:haircut_man:` | :haircut_woman: | `:haircut_woman:` | [top](#introduction) | +| [top](#people--body) | :walking: | `:walking:` | :walking_man: | `:walking_man:` | [top](#introduction) | +| [top](#people--body) | :walking_woman: | `:walking_woman:` | :standing_person: | `:standing_person:` | [top](#introduction) | +| [top](#people--body) | :standing_man: | `:standing_man:` | :standing_woman: | `:standing_woman:` | [top](#introduction) | +| [top](#people--body) | :kneeling_person: | `:kneeling_person:` | :kneeling_man: | `:kneeling_man:` | [top](#introduction) | +| [top](#people--body) | :kneeling_woman: | `:kneeling_woman:` | :person_with_probing_cane: | `:person_with_probing_cane:` | [top](#introduction) | +| [top](#people--body) | :man_with_probing_cane: | `:man_with_probing_cane:` | :woman_with_probing_cane: | `:woman_with_probing_cane:` | [top](#introduction) | +| [top](#people--body) | :person_in_motorized_wheelchair: | `:person_in_motorized_wheelchair:` | :man_in_motorized_wheelchair: | `:man_in_motorized_wheelchair:` | [top](#introduction) | +| [top](#people--body) | :woman_in_motorized_wheelchair: | `:woman_in_motorized_wheelchair:` | :person_in_manual_wheelchair: | `:person_in_manual_wheelchair:` | [top](#introduction) | +| [top](#people--body) | :man_in_manual_wheelchair: | `:man_in_manual_wheelchair:` | :woman_in_manual_wheelchair: | `:woman_in_manual_wheelchair:` | [top](#introduction) | +| [top](#people--body) | :runner: | `:runner:`
`:running:` | :running_man: | `:running_man:` | [top](#introduction) | +| [top](#people--body) | :running_woman: | `:running_woman:` | :dancer: | `:dancer:`
`:woman_dancing:` | [top](#introduction) | +| [top](#people--body) | :man_dancing: | `:man_dancing:` | :business_suit_levitating: | `:business_suit_levitating:` | [top](#introduction) | +| [top](#people--body) | :dancers: | `:dancers:` | :dancing_men: | `:dancing_men:` | [top](#introduction) | +| [top](#people--body) | :dancing_women: | `:dancing_women:` | :sauna_person: | `:sauna_person:` | [top](#introduction) | +| [top](#people--body) | :sauna_man: | `:sauna_man:` | :sauna_woman: | `:sauna_woman:` | [top](#introduction) | +| [top](#people--body) | :climbing: | `:climbing:` | :climbing_man: | `:climbing_man:` | [top](#introduction) | +| [top](#people--body) | :climbing_woman: | `:climbing_woman:` | | | [top](#introduction) | + +### Person Sport + +| | ico | shortcode | ico | shortcode | | +| - | :-: | - | :-: | - | - | +| [top](#people--body) | :person_fencing: | `:person_fencing:` | :horse_racing: | `:horse_racing:` | [top](#introduction) | +| [top](#people--body) | :skier: | `:skier:` | :snowboarder: | `:snowboarder:` | [top](#introduction) | +| [top](#people--body) | :golfing: | `:golfing:` | :golfing_man: | `:golfing_man:` | [top](#introduction) | +| [top](#people--body) | :golfing_woman: | `:golfing_woman:` | :surfer: | `:surfer:` | [top](#introduction) | +| [top](#people--body) | :surfing_man: | `:surfing_man:` | :surfing_woman: | `:surfing_woman:` | [top](#introduction) | +| [top](#people--body) | :rowboat: | `:rowboat:` | :rowing_man: | `:rowing_man:` | [top](#introduction) | +| [top](#people--body) | :rowing_woman: | `:rowing_woman:` | :swimmer: | `:swimmer:` | [top](#introduction) | +| [top](#people--body) | :swimming_man: | `:swimming_man:` | :swimming_woman: | `:swimming_woman:` | [top](#introduction) | +| [top](#people--body) | :bouncing_ball_person: | `:bouncing_ball_person:` | :basketball_man: | `:basketball_man:`
`:bouncing_ball_man:` | [top](#introduction) | +| [top](#people--body) | :basketball_woman: | `:basketball_woman:`
`:bouncing_ball_woman:` | :weight_lifting: | `:weight_lifting:` | [top](#introduction) | +| [top](#people--body) | :weight_lifting_man: | `:weight_lifting_man:` | :weight_lifting_woman: | `:weight_lifting_woman:` | [top](#introduction) | +| [top](#people--body) | :bicyclist: | `:bicyclist:` | :biking_man: | `:biking_man:` | [top](#introduction) | +| [top](#people--body) | :biking_woman: | `:biking_woman:` | :mountain_bicyclist: | `:mountain_bicyclist:` | [top](#introduction) | +| [top](#people--body) | :mountain_biking_man: | `:mountain_biking_man:` | :mountain_biking_woman: | `:mountain_biking_woman:` | [top](#introduction) | +| [top](#people--body) | :cartwheeling: | `:cartwheeling:` | :man_cartwheeling: | `:man_cartwheeling:` | [top](#introduction) | +| [top](#people--body) | :woman_cartwheeling: | `:woman_cartwheeling:` | :wrestling: | `:wrestling:` | [top](#introduction) | +| [top](#people--body) | :men_wrestling: | `:men_wrestling:` | :women_wrestling: | `:women_wrestling:` | [top](#introduction) | +| [top](#people--body) | :water_polo: | `:water_polo:` | :man_playing_water_polo: | `:man_playing_water_polo:` | [top](#introduction) | +| [top](#people--body) | :woman_playing_water_polo: | `:woman_playing_water_polo:` | :handball_person: | `:handball_person:` | [top](#introduction) | +| [top](#people--body) | :man_playing_handball: | `:man_playing_handball:` | :woman_playing_handball: | `:woman_playing_handball:` | [top](#introduction) | +| [top](#people--body) | :juggling_person: | `:juggling_person:` | :man_juggling: | `:man_juggling:` | [top](#introduction) | +| [top](#people--body) | :woman_juggling: | `:woman_juggling:` | | | [top](#introduction) | + +### Person Resting + +| | ico | shortcode | ico | shortcode | | +| - | :-: | - | :-: | - | - | +| [top](#people--body) | :lotus_position: | `:lotus_position:` | :lotus_position_man: | `:lotus_position_man:` | [top](#introduction) | +| [top](#people--body) | :lotus_position_woman: | `:lotus_position_woman:` | :bath: | `:bath:` | [top](#introduction) | +| [top](#people--body) | :sleeping_bed: | `:sleeping_bed:` | | | [top](#introduction) | + +### Family + +| | ico | shortcode | ico | shortcode | | +| - | :-: | - | :-: | - | - | +| [top](#people--body) | :people_holding_hands: | `:people_holding_hands:` | :two_women_holding_hands: | `:two_women_holding_hands:` | [top](#introduction) | +| [top](#people--body) | :couple: | `:couple:` | :two_men_holding_hands: | `:two_men_holding_hands:` | [top](#introduction) | +| [top](#people--body) | :couplekiss: | `:couplekiss:` | :couplekiss_man_woman: | `:couplekiss_man_woman:` | [top](#introduction) | +| [top](#people--body) | :couplekiss_man_man: | `:couplekiss_man_man:` | :couplekiss_woman_woman: | `:couplekiss_woman_woman:` | [top](#introduction) | +| [top](#people--body) | :couple_with_heart: | `:couple_with_heart:` | :couple_with_heart_woman_man: | `:couple_with_heart_woman_man:` | [top](#introduction) | +| [top](#people--body) | :couple_with_heart_man_man: | `:couple_with_heart_man_man:` | :couple_with_heart_woman_woman: | `:couple_with_heart_woman_woman:` | [top](#introduction) | +| [top](#people--body) | :family_man_woman_boy: | `:family_man_woman_boy:` | :family_man_woman_girl: | `:family_man_woman_girl:` | [top](#introduction) | +| [top](#people--body) | :family_man_woman_girl_boy: | `:family_man_woman_girl_boy:` | :family_man_woman_boy_boy: | `:family_man_woman_boy_boy:` | [top](#introduction) | +| [top](#people--body) | :family_man_woman_girl_girl: | `:family_man_woman_girl_girl:` | :family_man_man_boy: | `:family_man_man_boy:` | [top](#introduction) | +| [top](#people--body) | :family_man_man_girl: | `:family_man_man_girl:` | :family_man_man_girl_boy: | `:family_man_man_girl_boy:` | [top](#introduction) | +| [top](#people--body) | :family_man_man_boy_boy: | `:family_man_man_boy_boy:` | :family_man_man_girl_girl: | `:family_man_man_girl_girl:` | [top](#introduction) | +| [top](#people--body) | :family_woman_woman_boy: | `:family_woman_woman_boy:` | :family_woman_woman_girl: | `:family_woman_woman_girl:` | [top](#introduction) | +| [top](#people--body) | :family_woman_woman_girl_boy: | `:family_woman_woman_girl_boy:` | :family_woman_woman_boy_boy: | `:family_woman_woman_boy_boy:` | [top](#introduction) | +| [top](#people--body) | :family_woman_woman_girl_girl: | `:family_woman_woman_girl_girl:` | :family_man_boy: | `:family_man_boy:` | [top](#introduction) | +| [top](#people--body) | :family_man_boy_boy: | `:family_man_boy_boy:` | :family_man_girl: | `:family_man_girl:` | [top](#introduction) | +| [top](#people--body) | :family_man_girl_boy: | `:family_man_girl_boy:` | :family_man_girl_girl: | `:family_man_girl_girl:` | [top](#introduction) | +| [top](#people--body) | :family_woman_boy: | `:family_woman_boy:` | :family_woman_boy_boy: | `:family_woman_boy_boy:` | [top](#introduction) | +| [top](#people--body) | :family_woman_girl: | `:family_woman_girl:` | :family_woman_girl_boy: | `:family_woman_girl_boy:` | [top](#introduction) | +| [top](#people--body) | :family_woman_girl_girl: | `:family_woman_girl_girl:` | | | [top](#introduction) | + +### Person Symbol + +| | ico | shortcode | ico | shortcode | | +| - | :-: | - | :-: | - | - | +| [top](#people--body) | :speaking_head: | `:speaking_head:` | :bust_in_silhouette: | `:bust_in_silhouette:` | [top](#introduction) | +| [top](#people--body) | :busts_in_silhouette: | `:busts_in_silhouette:` | :people_hugging: | `:people_hugging:` | [top](#introduction) | +| [top](#people--body) | :family: | `:family:` | :footprints: | `:footprints:` | [top](#introduction) | + +## Animals & Nature + +- [Animal Mammal](#animal-mammal) +- [Animal Bird](#animal-bird) +- [Animal Amphibian](#animal-amphibian) +- [Animal Reptile](#animal-reptile) +- [Animal Marine](#animal-marine) +- [Animal Bug](#animal-bug) +- [Plant Flower](#plant-flower) +- [Plant Other](#plant-other) + +### Animal Mammal + +| | ico | shortcode | ico | shortcode | | +| - | :-: | - | :-: | - | - | +| [top](#animals--nature) | :monkey_face: | `:monkey_face:` | :monkey: | `:monkey:` | [top](#introduction) | +| [top](#animals--nature) | :gorilla: | `:gorilla:` | :orangutan: | `:orangutan:` | [top](#introduction) | +| [top](#animals--nature) | :dog: | `:dog:` | :dog2: | `:dog2:` | [top](#introduction) | +| [top](#animals--nature) | :guide_dog: | `:guide_dog:` | :service_dog: | `:service_dog:` | [top](#introduction) | +| [top](#animals--nature) | :poodle: | `:poodle:` | :wolf: | `:wolf:` | [top](#introduction) | +| [top](#animals--nature) | :fox_face: | `:fox_face:` | :raccoon: | `:raccoon:` | [top](#introduction) | +| [top](#animals--nature) | :cat: | `:cat:` | :cat2: | `:cat2:` | [top](#introduction) | +| [top](#animals--nature) | :black_cat: | `:black_cat:` | :lion: | `:lion:` | [top](#introduction) | +| [top](#animals--nature) | :tiger: | `:tiger:` | :tiger2: | `:tiger2:` | [top](#introduction) | +| [top](#animals--nature) | :leopard: | `:leopard:` | :horse: | `:horse:` | [top](#introduction) | +| [top](#animals--nature) | :racehorse: | `:racehorse:` | :unicorn: | `:unicorn:` | [top](#introduction) | +| [top](#animals--nature) | :zebra: | `:zebra:` | :deer: | `:deer:` | [top](#introduction) | +| [top](#animals--nature) | :bison: | `:bison:` | :cow: | `:cow:` | [top](#introduction) | +| [top](#animals--nature) | :ox: | `:ox:` | :water_buffalo: | `:water_buffalo:` | [top](#introduction) | +| [top](#animals--nature) | :cow2: | `:cow2:` | :pig: | `:pig:` | [top](#introduction) | +| [top](#animals--nature) | :pig2: | `:pig2:` | :boar: | `:boar:` | [top](#introduction) | +| [top](#animals--nature) | :pig_nose: | `:pig_nose:` | :ram: | `:ram:` | [top](#introduction) | +| [top](#animals--nature) | :sheep: | `:sheep:` | :goat: | `:goat:` | [top](#introduction) | +| [top](#animals--nature) | :dromedary_camel: | `:dromedary_camel:` | :camel: | `:camel:` | [top](#introduction) | +| [top](#animals--nature) | :llama: | `:llama:` | :giraffe: | `:giraffe:` | [top](#introduction) | +| [top](#animals--nature) | :elephant: | `:elephant:` | :mammoth: | `:mammoth:` | [top](#introduction) | +| [top](#animals--nature) | :rhinoceros: | `:rhinoceros:` | :hippopotamus: | `:hippopotamus:` | [top](#introduction) | +| [top](#animals--nature) | :mouse: | `:mouse:` | :mouse2: | `:mouse2:` | [top](#introduction) | +| [top](#animals--nature) | :rat: | `:rat:` | :hamster: | `:hamster:` | [top](#introduction) | +| [top](#animals--nature) | :rabbit: | `:rabbit:` | :rabbit2: | `:rabbit2:` | [top](#introduction) | +| [top](#animals--nature) | :chipmunk: | `:chipmunk:` | :beaver: | `:beaver:` | [top](#introduction) | +| [top](#animals--nature) | :hedgehog: | `:hedgehog:` | :bat: | `:bat:` | [top](#introduction) | +| [top](#animals--nature) | :bear: | `:bear:` | :polar_bear: | `:polar_bear:` | [top](#introduction) | +| [top](#animals--nature) | :koala: | `:koala:` | :panda_face: | `:panda_face:` | [top](#introduction) | +| [top](#animals--nature) | :sloth: | `:sloth:` | :otter: | `:otter:` | [top](#introduction) | +| [top](#animals--nature) | :skunk: | `:skunk:` | :kangaroo: | `:kangaroo:` | [top](#introduction) | +| [top](#animals--nature) | :badger: | `:badger:` | :feet: | `:feet:`
`:paw_prints:` | [top](#introduction) | + +### Animal Bird + +| | ico | shortcode | ico | shortcode | | +| - | :-: | - | :-: | - | - | +| [top](#animals--nature) | :turkey: | `:turkey:` | :chicken: | `:chicken:` | [top](#introduction) | +| [top](#animals--nature) | :rooster: | `:rooster:` | :hatching_chick: | `:hatching_chick:` | [top](#introduction) | +| [top](#animals--nature) | :baby_chick: | `:baby_chick:` | :hatched_chick: | `:hatched_chick:` | [top](#introduction) | +| [top](#animals--nature) | :bird: | `:bird:` | :penguin: | `:penguin:` | [top](#introduction) | +| [top](#animals--nature) | :dove: | `:dove:` | :eagle: | `:eagle:` | [top](#introduction) | +| [top](#animals--nature) | :duck: | `:duck:` | :swan: | `:swan:` | [top](#introduction) | +| [top](#animals--nature) | :owl: | `:owl:` | :dodo: | `:dodo:` | [top](#introduction) | +| [top](#animals--nature) | :feather: | `:feather:` | :flamingo: | `:flamingo:` | [top](#introduction) | +| [top](#animals--nature) | :peacock: | `:peacock:` | :parrot: | `:parrot:` | [top](#introduction) | + +### Animal Amphibian + +| | ico | shortcode | | +| - | :-: | - | - | +| [top](#animals--nature) | :frog: | `:frog:` | [top](#introduction) | + +### Animal Reptile + +| | ico | shortcode | ico | shortcode | | +| - | :-: | - | :-: | - | - | +| [top](#animals--nature) | :crocodile: | `:crocodile:` | :turtle: | `:turtle:` | [top](#introduction) | +| [top](#animals--nature) | :lizard: | `:lizard:` | :snake: | `:snake:` | [top](#introduction) | +| [top](#animals--nature) | :dragon_face: | `:dragon_face:` | :dragon: | `:dragon:` | [top](#introduction) | +| [top](#animals--nature) | :sauropod: | `:sauropod:` | :t-rex: | `:t-rex:` | [top](#introduction) | + +### Animal Marine + +| | ico | shortcode | ico | shortcode | | +| - | :-: | - | :-: | - | - | +| [top](#animals--nature) | :whale: | `:whale:` | :whale2: | `:whale2:` | [top](#introduction) | +| [top](#animals--nature) | :dolphin: | `:dolphin:`
`:flipper:` | :seal: | `:seal:` | [top](#introduction) | +| [top](#animals--nature) | :fish: | `:fish:` | :tropical_fish: | `:tropical_fish:` | [top](#introduction) | +| [top](#animals--nature) | :blowfish: | `:blowfish:` | :shark: | `:shark:` | [top](#introduction) | +| [top](#animals--nature) | :octopus: | `:octopus:` | :shell: | `:shell:` | [top](#introduction) | + +### Animal Bug + +| | ico | shortcode | ico | shortcode | | +| - | :-: | - | :-: | - | - | +| [top](#animals--nature) | :snail: | `:snail:` | :butterfly: | `:butterfly:` | [top](#introduction) | +| [top](#animals--nature) | :bug: | `:bug:` | :ant: | `:ant:` | [top](#introduction) | +| [top](#animals--nature) | :bee: | `:bee:`
`:honeybee:` | :beetle: | `:beetle:` | [top](#introduction) | +| [top](#animals--nature) | :lady_beetle: | `:lady_beetle:` | :cricket: | `:cricket:` | [top](#introduction) | +| [top](#animals--nature) | :cockroach: | `:cockroach:` | :spider: | `:spider:` | [top](#introduction) | +| [top](#animals--nature) | :spider_web: | `:spider_web:` | :scorpion: | `:scorpion:` | [top](#introduction) | +| [top](#animals--nature) | :mosquito: | `:mosquito:` | :fly: | `:fly:` | [top](#introduction) | +| [top](#animals--nature) | :worm: | `:worm:` | :microbe: | `:microbe:` | [top](#introduction) | + +### Plant Flower + +| | ico | shortcode | ico | shortcode | | +| - | :-: | - | :-: | - | - | +| [top](#animals--nature) | :bouquet: | `:bouquet:` | :cherry_blossom: | `:cherry_blossom:` | [top](#introduction) | +| [top](#animals--nature) | :white_flower: | `:white_flower:` | :rosette: | `:rosette:` | [top](#introduction) | +| [top](#animals--nature) | :rose: | `:rose:` | :wilted_flower: | `:wilted_flower:` | [top](#introduction) | +| [top](#animals--nature) | :hibiscus: | `:hibiscus:` | :sunflower: | `:sunflower:` | [top](#introduction) | +| [top](#animals--nature) | :blossom: | `:blossom:` | :tulip: | `:tulip:` | [top](#introduction) | + +### Plant Other + +| | ico | shortcode | ico | shortcode | | +| - | :-: | - | :-: | - | - | +| [top](#animals--nature) | :seedling: | `:seedling:` | :potted_plant: | `:potted_plant:` | [top](#introduction) | +| [top](#animals--nature) | :evergreen_tree: | `:evergreen_tree:` | :deciduous_tree: | `:deciduous_tree:` | [top](#introduction) | +| [top](#animals--nature) | :palm_tree: | `:palm_tree:` | :cactus: | `:cactus:` | [top](#introduction) | +| [top](#animals--nature) | :ear_of_rice: | `:ear_of_rice:` | :herb: | `:herb:` | [top](#introduction) | +| [top](#animals--nature) | :shamrock: | `:shamrock:` | :four_leaf_clover: | `:four_leaf_clover:` | [top](#introduction) | +| [top](#animals--nature) | :maple_leaf: | `:maple_leaf:` | :fallen_leaf: | `:fallen_leaf:` | [top](#introduction) | +| [top](#animals--nature) | :leaves: | `:leaves:` | :mushroom: | `:mushroom:` | [top](#introduction) | + +## Food & Drink + +- [Food Fruit](#food-fruit) +- [Food Vegetable](#food-vegetable) +- [Food Prepared](#food-prepared) +- [Food Asian](#food-asian) +- [Food Marine](#food-marine) +- [Food Sweet](#food-sweet) +- [Drink](#drink) +- [Dishware](#dishware) + +### Food Fruit + +| | ico | shortcode | ico | shortcode | | +| - | :-: | - | :-: | - | - | +| [top](#food--drink) | :grapes: | `:grapes:` | :melon: | `:melon:` | [top](#introduction) | +| [top](#food--drink) | :watermelon: | `:watermelon:` | :mandarin: | `:mandarin:`
`:orange:`
`:tangerine:` | [top](#introduction) | +| [top](#food--drink) | :lemon: | `:lemon:` | :banana: | `:banana:` | [top](#introduction) | +| [top](#food--drink) | :pineapple: | `:pineapple:` | :mango: | `:mango:` | [top](#introduction) | +| [top](#food--drink) | :apple: | `:apple:` | :green_apple: | `:green_apple:` | [top](#introduction) | +| [top](#food--drink) | :pear: | `:pear:` | :peach: | `:peach:` | [top](#introduction) | +| [top](#food--drink) | :cherries: | `:cherries:` | :strawberry: | `:strawberry:` | [top](#introduction) | +| [top](#food--drink) | :blueberries: | `:blueberries:` | :kiwi_fruit: | `:kiwi_fruit:` | [top](#introduction) | +| [top](#food--drink) | :tomato: | `:tomato:` | :olive: | `:olive:` | [top](#introduction) | +| [top](#food--drink) | :coconut: | `:coconut:` | | | [top](#introduction) | + +### Food Vegetable + +| | ico | shortcode | ico | shortcode | | +| - | :-: | - | :-: | - | - | +| [top](#food--drink) | :avocado: | `:avocado:` | :eggplant: | `:eggplant:` | [top](#introduction) | +| [top](#food--drink) | :potato: | `:potato:` | :carrot: | `:carrot:` | [top](#introduction) | +| [top](#food--drink) | :corn: | `:corn:` | :hot_pepper: | `:hot_pepper:` | [top](#introduction) | +| [top](#food--drink) | :bell_pepper: | `:bell_pepper:` | :cucumber: | `:cucumber:` | [top](#introduction) | +| [top](#food--drink) | :leafy_green: | `:leafy_green:` | :broccoli: | `:broccoli:` | [top](#introduction) | +| [top](#food--drink) | :garlic: | `:garlic:` | :onion: | `:onion:` | [top](#introduction) | +| [top](#food--drink) | :peanuts: | `:peanuts:` | :chestnut: | `:chestnut:` | [top](#introduction) | + +### Food Prepared + +| | ico | shortcode | ico | shortcode | | +| - | :-: | - | :-: | - | - | +| [top](#food--drink) | :bread: | `:bread:` | :croissant: | `:croissant:` | [top](#introduction) | +| [top](#food--drink) | :baguette_bread: | `:baguette_bread:` | :flatbread: | `:flatbread:` | [top](#introduction) | +| [top](#food--drink) | :pretzel: | `:pretzel:` | :bagel: | `:bagel:` | [top](#introduction) | +| [top](#food--drink) | :pancakes: | `:pancakes:` | :waffle: | `:waffle:` | [top](#introduction) | +| [top](#food--drink) | :cheese: | `:cheese:` | :meat_on_bone: | `:meat_on_bone:` | [top](#introduction) | +| [top](#food--drink) | :poultry_leg: | `:poultry_leg:` | :cut_of_meat: | `:cut_of_meat:` | [top](#introduction) | +| [top](#food--drink) | :bacon: | `:bacon:` | :hamburger: | `:hamburger:` | [top](#introduction) | +| [top](#food--drink) | :fries: | `:fries:` | :pizza: | `:pizza:` | [top](#introduction) | +| [top](#food--drink) | :hotdog: | `:hotdog:` | :sandwich: | `:sandwich:` | [top](#introduction) | +| [top](#food--drink) | :taco: | `:taco:` | :burrito: | `:burrito:` | [top](#introduction) | +| [top](#food--drink) | :tamale: | `:tamale:` | :stuffed_flatbread: | `:stuffed_flatbread:` | [top](#introduction) | +| [top](#food--drink) | :falafel: | `:falafel:` | :egg: | `:egg:` | [top](#introduction) | +| [top](#food--drink) | :fried_egg: | `:fried_egg:` | :shallow_pan_of_food: | `:shallow_pan_of_food:` | [top](#introduction) | +| [top](#food--drink) | :stew: | `:stew:` | :fondue: | `:fondue:` | [top](#introduction) | +| [top](#food--drink) | :bowl_with_spoon: | `:bowl_with_spoon:` | :green_salad: | `:green_salad:` | [top](#introduction) | +| [top](#food--drink) | :popcorn: | `:popcorn:` | :butter: | `:butter:` | [top](#introduction) | +| [top](#food--drink) | :salt: | `:salt:` | :canned_food: | `:canned_food:` | [top](#introduction) | + +### Food Asian + +| | ico | shortcode | ico | shortcode | | +| - | :-: | - | :-: | - | - | +| [top](#food--drink) | :bento: | `:bento:` | :rice_cracker: | `:rice_cracker:` | [top](#introduction) | +| [top](#food--drink) | :rice_ball: | `:rice_ball:` | :rice: | `:rice:` | [top](#introduction) | +| [top](#food--drink) | :curry: | `:curry:` | :ramen: | `:ramen:` | [top](#introduction) | +| [top](#food--drink) | :spaghetti: | `:spaghetti:` | :sweet_potato: | `:sweet_potato:` | [top](#introduction) | +| [top](#food--drink) | :oden: | `:oden:` | :sushi: | `:sushi:` | [top](#introduction) | +| [top](#food--drink) | :fried_shrimp: | `:fried_shrimp:` | :fish_cake: | `:fish_cake:` | [top](#introduction) | +| [top](#food--drink) | :moon_cake: | `:moon_cake:` | :dango: | `:dango:` | [top](#introduction) | +| [top](#food--drink) | :dumpling: | `:dumpling:` | :fortune_cookie: | `:fortune_cookie:` | [top](#introduction) | +| [top](#food--drink) | :takeout_box: | `:takeout_box:` | | | [top](#introduction) | + +### Food Marine + +| | ico | shortcode | ico | shortcode | | +| - | :-: | - | :-: | - | - | +| [top](#food--drink) | :crab: | `:crab:` | :lobster: | `:lobster:` | [top](#introduction) | +| [top](#food--drink) | :shrimp: | `:shrimp:` | :squid: | `:squid:` | [top](#introduction) | +| [top](#food--drink) | :oyster: | `:oyster:` | | | [top](#introduction) | + +### Food Sweet + +| | ico | shortcode | ico | shortcode | | +| - | :-: | - | :-: | - | - | +| [top](#food--drink) | :icecream: | `:icecream:` | :shaved_ice: | `:shaved_ice:` | [top](#introduction) | +| [top](#food--drink) | :ice_cream: | `:ice_cream:` | :doughnut: | `:doughnut:` | [top](#introduction) | +| [top](#food--drink) | :cookie: | `:cookie:` | :birthday: | `:birthday:` | [top](#introduction) | +| [top](#food--drink) | :cake: | `:cake:` | :cupcake: | `:cupcake:` | [top](#introduction) | +| [top](#food--drink) | :pie: | `:pie:` | :chocolate_bar: | `:chocolate_bar:` | [top](#introduction) | +| [top](#food--drink) | :candy: | `:candy:` | :lollipop: | `:lollipop:` | [top](#introduction) | +| [top](#food--drink) | :custard: | `:custard:` | :honey_pot: | `:honey_pot:` | [top](#introduction) | + +### Drink + +| | ico | shortcode | ico | shortcode | | +| - | :-: | - | :-: | - | - | +| [top](#food--drink) | :baby_bottle: | `:baby_bottle:` | :milk_glass: | `:milk_glass:` | [top](#introduction) | +| [top](#food--drink) | :coffee: | `:coffee:` | :teapot: | `:teapot:` | [top](#introduction) | +| [top](#food--drink) | :tea: | `:tea:` | :sake: | `:sake:` | [top](#introduction) | +| [top](#food--drink) | :champagne: | `:champagne:` | :wine_glass: | `:wine_glass:` | [top](#introduction) | +| [top](#food--drink) | :cocktail: | `:cocktail:` | :tropical_drink: | `:tropical_drink:` | [top](#introduction) | +| [top](#food--drink) | :beer: | `:beer:` | :beers: | `:beers:` | [top](#introduction) | +| [top](#food--drink) | :clinking_glasses: | `:clinking_glasses:` | :tumbler_glass: | `:tumbler_glass:` | [top](#introduction) | +| [top](#food--drink) | :cup_with_straw: | `:cup_with_straw:` | :bubble_tea: | `:bubble_tea:` | [top](#introduction) | +| [top](#food--drink) | :beverage_box: | `:beverage_box:` | :mate: | `:mate:` | [top](#introduction) | +| [top](#food--drink) | :ice_cube: | `:ice_cube:` | | | [top](#introduction) | + +### Dishware + +| | ico | shortcode | ico | shortcode | | +| - | :-: | - | :-: | - | - | +| [top](#food--drink) | :chopsticks: | `:chopsticks:` | :plate_with_cutlery: | `:plate_with_cutlery:` | [top](#introduction) | +| [top](#food--drink) | :fork_and_knife: | `:fork_and_knife:` | :spoon: | `:spoon:` | [top](#introduction) | +| [top](#food--drink) | :hocho: | `:hocho:`
`:knife:` | :amphora: | `:amphora:` | [top](#introduction) | + +## Travel & Places + +- [Place Map](#place-map) +- [Place Geographic](#place-geographic) +- [Place Building](#place-building) +- [Place Religious](#place-religious) +- [Place Other](#place-other) +- [Transport Ground](#transport-ground) +- [Transport Water](#transport-water) +- [Transport Air](#transport-air) +- [Hotel](#hotel) +- [Time](#time) +- [Sky & Weather](#sky--weather) + +### Place Map + +| | ico | shortcode | ico | shortcode | | +| - | :-: | - | :-: | - | - | +| [top](#travel--places) | :earth_africa: | `:earth_africa:` | :earth_americas: | `:earth_americas:` | [top](#introduction) | +| [top](#travel--places) | :earth_asia: | `:earth_asia:` | :globe_with_meridians: | `:globe_with_meridians:` | [top](#introduction) | +| [top](#travel--places) | :world_map: | `:world_map:` | :japan: | `:japan:` | [top](#introduction) | +| [top](#travel--places) | :compass: | `:compass:` | | | [top](#introduction) | + +### Place Geographic + +| | ico | shortcode | ico | shortcode | | +| - | :-: | - | :-: | - | - | +| [top](#travel--places) | :mountain_snow: | `:mountain_snow:` | :mountain: | `:mountain:` | [top](#introduction) | +| [top](#travel--places) | :volcano: | `:volcano:` | :mount_fuji: | `:mount_fuji:` | [top](#introduction) | +| [top](#travel--places) | :camping: | `:camping:` | :beach_umbrella: | `:beach_umbrella:` | [top](#introduction) | +| [top](#travel--places) | :desert: | `:desert:` | :desert_island: | `:desert_island:` | [top](#introduction) | +| [top](#travel--places) | :national_park: | `:national_park:` | | | [top](#introduction) | + +### Place Building + +| | ico | shortcode | ico | shortcode | | +| - | :-: | - | :-: | - | - | +| [top](#travel--places) | :stadium: | `:stadium:` | :classical_building: | `:classical_building:` | [top](#introduction) | +| [top](#travel--places) | :building_construction: | `:building_construction:` | :bricks: | `:bricks:` | [top](#introduction) | +| [top](#travel--places) | :rock: | `:rock:` | :wood: | `:wood:` | [top](#introduction) | +| [top](#travel--places) | :hut: | `:hut:` | :houses: | `:houses:` | [top](#introduction) | +| [top](#travel--places) | :derelict_house: | `:derelict_house:` | :house: | `:house:` | [top](#introduction) | +| [top](#travel--places) | :house_with_garden: | `:house_with_garden:` | :office: | `:office:` | [top](#introduction) | +| [top](#travel--places) | :post_office: | `:post_office:` | :european_post_office: | `:european_post_office:` | [top](#introduction) | +| [top](#travel--places) | :hospital: | `:hospital:` | :bank: | `:bank:` | [top](#introduction) | +| [top](#travel--places) | :hotel: | `:hotel:` | :love_hotel: | `:love_hotel:` | [top](#introduction) | +| [top](#travel--places) | :convenience_store: | `:convenience_store:` | :school: | `:school:` | [top](#introduction) | +| [top](#travel--places) | :department_store: | `:department_store:` | :factory: | `:factory:` | [top](#introduction) | +| [top](#travel--places) | :japanese_castle: | `:japanese_castle:` | :european_castle: | `:european_castle:` | [top](#introduction) | +| [top](#travel--places) | :wedding: | `:wedding:` | :tokyo_tower: | `:tokyo_tower:` | [top](#introduction) | +| [top](#travel--places) | :statue_of_liberty: | `:statue_of_liberty:` | | | [top](#introduction) | + +### Place Religious + +| | ico | shortcode | ico | shortcode | | +| - | :-: | - | :-: | - | - | +| [top](#travel--places) | :church: | `:church:` | :mosque: | `:mosque:` | [top](#introduction) | +| [top](#travel--places) | :hindu_temple: | `:hindu_temple:` | :synagogue: | `:synagogue:` | [top](#introduction) | +| [top](#travel--places) | :shinto_shrine: | `:shinto_shrine:` | :kaaba: | `:kaaba:` | [top](#introduction) | + +### Place Other + +| | ico | shortcode | ico | shortcode | | +| - | :-: | - | :-: | - | - | +| [top](#travel--places) | :fountain: | `:fountain:` | :tent: | `:tent:` | [top](#introduction) | +| [top](#travel--places) | :foggy: | `:foggy:` | :night_with_stars: | `:night_with_stars:` | [top](#introduction) | +| [top](#travel--places) | :cityscape: | `:cityscape:` | :sunrise_over_mountains: | `:sunrise_over_mountains:` | [top](#introduction) | +| [top](#travel--places) | :sunrise: | `:sunrise:` | :city_sunset: | `:city_sunset:` | [top](#introduction) | +| [top](#travel--places) | :city_sunrise: | `:city_sunrise:` | :bridge_at_night: | `:bridge_at_night:` | [top](#introduction) | +| [top](#travel--places) | :hotsprings: | `:hotsprings:` | :carousel_horse: | `:carousel_horse:` | [top](#introduction) | +| [top](#travel--places) | :ferris_wheel: | `:ferris_wheel:` | :roller_coaster: | `:roller_coaster:` | [top](#introduction) | +| [top](#travel--places) | :barber: | `:barber:` | :circus_tent: | `:circus_tent:` | [top](#introduction) | + +### Transport Ground + +| | ico | shortcode | ico | shortcode | | +| - | :-: | - | :-: | - | - | +| [top](#travel--places) | :steam_locomotive: | `:steam_locomotive:` | :railway_car: | `:railway_car:` | [top](#introduction) | +| [top](#travel--places) | :bullettrain_side: | `:bullettrain_side:` | :bullettrain_front: | `:bullettrain_front:` | [top](#introduction) | +| [top](#travel--places) | :train2: | `:train2:` | :metro: | `:metro:` | [top](#introduction) | +| [top](#travel--places) | :light_rail: | `:light_rail:` | :station: | `:station:` | [top](#introduction) | +| [top](#travel--places) | :tram: | `:tram:` | :monorail: | `:monorail:` | [top](#introduction) | +| [top](#travel--places) | :mountain_railway: | `:mountain_railway:` | :train: | `:train:` | [top](#introduction) | +| [top](#travel--places) | :bus: | `:bus:` | :oncoming_bus: | `:oncoming_bus:` | [top](#introduction) | +| [top](#travel--places) | :trolleybus: | `:trolleybus:` | :minibus: | `:minibus:` | [top](#introduction) | +| [top](#travel--places) | :ambulance: | `:ambulance:` | :fire_engine: | `:fire_engine:` | [top](#introduction) | +| [top](#travel--places) | :police_car: | `:police_car:` | :oncoming_police_car: | `:oncoming_police_car:` | [top](#introduction) | +| [top](#travel--places) | :taxi: | `:taxi:` | :oncoming_taxi: | `:oncoming_taxi:` | [top](#introduction) | +| [top](#travel--places) | :car: | `:car:`
`:red_car:` | :oncoming_automobile: | `:oncoming_automobile:` | [top](#introduction) | +| [top](#travel--places) | :blue_car: | `:blue_car:` | :pickup_truck: | `:pickup_truck:` | [top](#introduction) | +| [top](#travel--places) | :truck: | `:truck:` | :articulated_lorry: | `:articulated_lorry:` | [top](#introduction) | +| [top](#travel--places) | :tractor: | `:tractor:` | :racing_car: | `:racing_car:` | [top](#introduction) | +| [top](#travel--places) | :motorcycle: | `:motorcycle:` | :motor_scooter: | `:motor_scooter:` | [top](#introduction) | +| [top](#travel--places) | :manual_wheelchair: | `:manual_wheelchair:` | :motorized_wheelchair: | `:motorized_wheelchair:` | [top](#introduction) | +| [top](#travel--places) | :auto_rickshaw: | `:auto_rickshaw:` | :bike: | `:bike:` | [top](#introduction) | +| [top](#travel--places) | :kick_scooter: | `:kick_scooter:` | :skateboard: | `:skateboard:` | [top](#introduction) | +| [top](#travel--places) | :roller_skate: | `:roller_skate:` | :busstop: | `:busstop:` | [top](#introduction) | +| [top](#travel--places) | :motorway: | `:motorway:` | :railway_track: | `:railway_track:` | [top](#introduction) | +| [top](#travel--places) | :oil_drum: | `:oil_drum:` | :fuelpump: | `:fuelpump:` | [top](#introduction) | +| [top](#travel--places) | :rotating_light: | `:rotating_light:` | :traffic_light: | `:traffic_light:` | [top](#introduction) | +| [top](#travel--places) | :vertical_traffic_light: | `:vertical_traffic_light:` | :stop_sign: | `:stop_sign:` | [top](#introduction) | +| [top](#travel--places) | :construction: | `:construction:` | | | [top](#introduction) | + +### Transport Water + +| | ico | shortcode | ico | shortcode | | +| - | :-: | - | :-: | - | - | +| [top](#travel--places) | :anchor: | `:anchor:` | :boat: | `:boat:`
`:sailboat:` | [top](#introduction) | +| [top](#travel--places) | :canoe: | `:canoe:` | :speedboat: | `:speedboat:` | [top](#introduction) | +| [top](#travel--places) | :passenger_ship: | `:passenger_ship:` | :ferry: | `:ferry:` | [top](#introduction) | +| [top](#travel--places) | :motor_boat: | `:motor_boat:` | :ship: | `:ship:` | [top](#introduction) | + +### Transport Air + +| | ico | shortcode | ico | shortcode | | +| - | :-: | - | :-: | - | - | +| [top](#travel--places) | :airplane: | `:airplane:` | :small_airplane: | `:small_airplane:` | [top](#introduction) | +| [top](#travel--places) | :flight_departure: | `:flight_departure:` | :flight_arrival: | `:flight_arrival:` | [top](#introduction) | +| [top](#travel--places) | :parachute: | `:parachute:` | :seat: | `:seat:` | [top](#introduction) | +| [top](#travel--places) | :helicopter: | `:helicopter:` | :suspension_railway: | `:suspension_railway:` | [top](#introduction) | +| [top](#travel--places) | :mountain_cableway: | `:mountain_cableway:` | :aerial_tramway: | `:aerial_tramway:` | [top](#introduction) | +| [top](#travel--places) | :artificial_satellite: | `:artificial_satellite:` | :rocket: | `:rocket:` | [top](#introduction) | +| [top](#travel--places) | :flying_saucer: | `:flying_saucer:` | | | [top](#introduction) | + +### Hotel + +| | ico | shortcode | ico | shortcode | | +| - | :-: | - | :-: | - | - | +| [top](#travel--places) | :bellhop_bell: | `:bellhop_bell:` | :luggage: | `:luggage:` | [top](#introduction) | + +### Time + +| | ico | shortcode | ico | shortcode | | +| - | :-: | - | :-: | - | - | +| [top](#travel--places) | :hourglass: | `:hourglass:` | :hourglass_flowing_sand: | `:hourglass_flowing_sand:` | [top](#introduction) | +| [top](#travel--places) | :watch: | `:watch:` | :alarm_clock: | `:alarm_clock:` | [top](#introduction) | +| [top](#travel--places) | :stopwatch: | `:stopwatch:` | :timer_clock: | `:timer_clock:` | [top](#introduction) | +| [top](#travel--places) | :mantelpiece_clock: | `:mantelpiece_clock:` | :clock12: | `:clock12:` | [top](#introduction) | +| [top](#travel--places) | :clock1230: | `:clock1230:` | :clock1: | `:clock1:` | [top](#introduction) | +| [top](#travel--places) | :clock130: | `:clock130:` | :clock2: | `:clock2:` | [top](#introduction) | +| [top](#travel--places) | :clock230: | `:clock230:` | :clock3: | `:clock3:` | [top](#introduction) | +| [top](#travel--places) | :clock330: | `:clock330:` | :clock4: | `:clock4:` | [top](#introduction) | +| [top](#travel--places) | :clock430: | `:clock430:` | :clock5: | `:clock5:` | [top](#introduction) | +| [top](#travel--places) | :clock530: | `:clock530:` | :clock6: | `:clock6:` | [top](#introduction) | +| [top](#travel--places) | :clock630: | `:clock630:` | :clock7: | `:clock7:` | [top](#introduction) | +| [top](#travel--places) | :clock730: | `:clock730:` | :clock8: | `:clock8:` | [top](#introduction) | +| [top](#travel--places) | :clock830: | `:clock830:` | :clock9: | `:clock9:` | [top](#introduction) | +| [top](#travel--places) | :clock930: | `:clock930:` | :clock10: | `:clock10:` | [top](#introduction) | +| [top](#travel--places) | :clock1030: | `:clock1030:` | :clock11: | `:clock11:` | [top](#introduction) | +| [top](#travel--places) | :clock1130: | `:clock1130:` | | | [top](#introduction) | + +### Sky & Weather + +| | ico | shortcode | ico | shortcode | | +| - | :-: | - | :-: | - | - | +| [top](#travel--places) | :new_moon: | `:new_moon:` | :waxing_crescent_moon: | `:waxing_crescent_moon:` | [top](#introduction) | +| [top](#travel--places) | :first_quarter_moon: | `:first_quarter_moon:` | :moon: | `:moon:`
`:waxing_gibbous_moon:` | [top](#introduction) | +| [top](#travel--places) | :full_moon: | `:full_moon:` | :waning_gibbous_moon: | `:waning_gibbous_moon:` | [top](#introduction) | +| [top](#travel--places) | :last_quarter_moon: | `:last_quarter_moon:` | :waning_crescent_moon: | `:waning_crescent_moon:` | [top](#introduction) | +| [top](#travel--places) | :crescent_moon: | `:crescent_moon:` | :new_moon_with_face: | `:new_moon_with_face:` | [top](#introduction) | +| [top](#travel--places) | :first_quarter_moon_with_face: | `:first_quarter_moon_with_face:` | :last_quarter_moon_with_face: | `:last_quarter_moon_with_face:` | [top](#introduction) | +| [top](#travel--places) | :thermometer: | `:thermometer:` | :sunny: | `:sunny:` | [top](#introduction) | +| [top](#travel--places) | :full_moon_with_face: | `:full_moon_with_face:` | :sun_with_face: | `:sun_with_face:` | [top](#introduction) | +| [top](#travel--places) | :ringed_planet: | `:ringed_planet:` | :star: | `:star:` | [top](#introduction) | +| [top](#travel--places) | :star2: | `:star2:` | :stars: | `:stars:` | [top](#introduction) | +| [top](#travel--places) | :milky_way: | `:milky_way:` | :cloud: | `:cloud:` | [top](#introduction) | +| [top](#travel--places) | :partly_sunny: | `:partly_sunny:` | :cloud_with_lightning_and_rain: | `:cloud_with_lightning_and_rain:` | [top](#introduction) | +| [top](#travel--places) | :sun_behind_small_cloud: | `:sun_behind_small_cloud:` | :sun_behind_large_cloud: | `:sun_behind_large_cloud:` | [top](#introduction) | +| [top](#travel--places) | :sun_behind_rain_cloud: | `:sun_behind_rain_cloud:` | :cloud_with_rain: | `:cloud_with_rain:` | [top](#introduction) | +| [top](#travel--places) | :cloud_with_snow: | `:cloud_with_snow:` | :cloud_with_lightning: | `:cloud_with_lightning:` | [top](#introduction) | +| [top](#travel--places) | :tornado: | `:tornado:` | :fog: | `:fog:` | [top](#introduction) | +| [top](#travel--places) | :wind_face: | `:wind_face:` | :cyclone: | `:cyclone:` | [top](#introduction) | +| [top](#travel--places) | :rainbow: | `:rainbow:` | :closed_umbrella: | `:closed_umbrella:` | [top](#introduction) | +| [top](#travel--places) | :open_umbrella: | `:open_umbrella:` | :umbrella: | `:umbrella:` | [top](#introduction) | +| [top](#travel--places) | :parasol_on_ground: | `:parasol_on_ground:` | :zap: | `:zap:` | [top](#introduction) | +| [top](#travel--places) | :snowflake: | `:snowflake:` | :snowman_with_snow: | `:snowman_with_snow:` | [top](#introduction) | +| [top](#travel--places) | :snowman: | `:snowman:` | :comet: | `:comet:` | [top](#introduction) | +| [top](#travel--places) | :fire: | `:fire:` | :droplet: | `:droplet:` | [top](#introduction) | +| [top](#travel--places) | :ocean: | `:ocean:` | | | [top](#introduction) | + +## Activities + +- [Event](#event) +- [Award Medal](#award-medal) +- [Sport](#sport) +- [Game](#game) +- [Arts & Crafts](#arts--crafts) + +### Event + +| | ico | shortcode | ico | shortcode | | +| - | :-: | - | :-: | - | - | +| [top](#activities) | :jack_o_lantern: | `:jack_o_lantern:` | :christmas_tree: | `:christmas_tree:` | [top](#introduction) | +| [top](#activities) | :fireworks: | `:fireworks:` | :sparkler: | `:sparkler:` | [top](#introduction) | +| [top](#activities) | :firecracker: | `:firecracker:` | :sparkles: | `:sparkles:` | [top](#introduction) | +| [top](#activities) | :balloon: | `:balloon:` | :tada: | `:tada:` | [top](#introduction) | +| [top](#activities) | :confetti_ball: | `:confetti_ball:` | :tanabata_tree: | `:tanabata_tree:` | [top](#introduction) | +| [top](#activities) | :bamboo: | `:bamboo:` | :dolls: | `:dolls:` | [top](#introduction) | +| [top](#activities) | :flags: | `:flags:` | :wind_chime: | `:wind_chime:` | [top](#introduction) | +| [top](#activities) | :rice_scene: | `:rice_scene:` | :red_envelope: | `:red_envelope:` | [top](#introduction) | +| [top](#activities) | :ribbon: | `:ribbon:` | :gift: | `:gift:` | [top](#introduction) | +| [top](#activities) | :reminder_ribbon: | `:reminder_ribbon:` | :tickets: | `:tickets:` | [top](#introduction) | +| [top](#activities) | :ticket: | `:ticket:` | | | [top](#introduction) | + +### Award Medal + +| | ico | shortcode | ico | shortcode | | +| - | :-: | - | :-: | - | - | +| [top](#activities) | :medal_military: | `:medal_military:` | :trophy: | `:trophy:` | [top](#introduction) | +| [top](#activities) | :medal_sports: | `:medal_sports:` | :1st_place_medal: | `:1st_place_medal:` | [top](#introduction) | +| [top](#activities) | :2nd_place_medal: | `:2nd_place_medal:` | :3rd_place_medal: | `:3rd_place_medal:` | [top](#introduction) | + +### Sport + +| | ico | shortcode | ico | shortcode | | +| - | :-: | - | :-: | - | - | +| [top](#activities) | :soccer: | `:soccer:` | :baseball: | `:baseball:` | [top](#introduction) | +| [top](#activities) | :softball: | `:softball:` | :basketball: | `:basketball:` | [top](#introduction) | +| [top](#activities) | :volleyball: | `:volleyball:` | :football: | `:football:` | [top](#introduction) | +| [top](#activities) | :rugby_football: | `:rugby_football:` | :tennis: | `:tennis:` | [top](#introduction) | +| [top](#activities) | :flying_disc: | `:flying_disc:` | :bowling: | `:bowling:` | [top](#introduction) | +| [top](#activities) | :cricket_game: | `:cricket_game:` | :field_hockey: | `:field_hockey:` | [top](#introduction) | +| [top](#activities) | :ice_hockey: | `:ice_hockey:` | :lacrosse: | `:lacrosse:` | [top](#introduction) | +| [top](#activities) | :ping_pong: | `:ping_pong:` | :badminton: | `:badminton:` | [top](#introduction) | +| [top](#activities) | :boxing_glove: | `:boxing_glove:` | :martial_arts_uniform: | `:martial_arts_uniform:` | [top](#introduction) | +| [top](#activities) | :goal_net: | `:goal_net:` | :golf: | `:golf:` | [top](#introduction) | +| [top](#activities) | :ice_skate: | `:ice_skate:` | :fishing_pole_and_fish: | `:fishing_pole_and_fish:` | [top](#introduction) | +| [top](#activities) | :diving_mask: | `:diving_mask:` | :running_shirt_with_sash: | `:running_shirt_with_sash:` | [top](#introduction) | +| [top](#activities) | :ski: | `:ski:` | :sled: | `:sled:` | [top](#introduction) | +| [top](#activities) | :curling_stone: | `:curling_stone:` | | | [top](#introduction) | + +### Game + +| | ico | shortcode | ico | shortcode | | +| - | :-: | - | :-: | - | - | +| [top](#activities) | :dart: | `:dart:` | :yo_yo: | `:yo_yo:` | [top](#introduction) | +| [top](#activities) | :kite: | `:kite:` | :gun: | `:gun:` | [top](#introduction) | +| [top](#activities) | :8ball: | `:8ball:` | :crystal_ball: | `:crystal_ball:` | [top](#introduction) | +| [top](#activities) | :magic_wand: | `:magic_wand:` | :video_game: | `:video_game:` | [top](#introduction) | +| [top](#activities) | :joystick: | `:joystick:` | :slot_machine: | `:slot_machine:` | [top](#introduction) | +| [top](#activities) | :game_die: | `:game_die:` | :jigsaw: | `:jigsaw:` | [top](#introduction) | +| [top](#activities) | :teddy_bear: | `:teddy_bear:` | :pinata: | `:pinata:` | [top](#introduction) | +| [top](#activities) | :nesting_dolls: | `:nesting_dolls:` | :spades: | `:spades:` | [top](#introduction) | +| [top](#activities) | :hearts: | `:hearts:` | :diamonds: | `:diamonds:` | [top](#introduction) | +| [top](#activities) | :clubs: | `:clubs:` | :chess_pawn: | `:chess_pawn:` | [top](#introduction) | +| [top](#activities) | :black_joker: | `:black_joker:` | :mahjong: | `:mahjong:` | [top](#introduction) | +| [top](#activities) | :flower_playing_cards: | `:flower_playing_cards:` | | | [top](#introduction) | + +### Arts & Crafts + +| | ico | shortcode | ico | shortcode | | +| - | :-: | - | :-: | - | - | +| [top](#activities) | :performing_arts: | `:performing_arts:` | :framed_picture: | `:framed_picture:` | [top](#introduction) | +| [top](#activities) | :art: | `:art:` | :thread: | `:thread:` | [top](#introduction) | +| [top](#activities) | :sewing_needle: | `:sewing_needle:` | :yarn: | `:yarn:` | [top](#introduction) | +| [top](#activities) | :knot: | `:knot:` | | | [top](#introduction) | + +## Objects + +- [Clothing](#clothing) +- [Sound](#sound) +- [Music](#music) +- [Musical Instrument](#musical-instrument) +- [Phone](#phone) +- [Computer](#computer) +- [Light & Video](#light--video) +- [Book Paper](#book-paper) +- [Money](#money) +- [Mail](#mail) +- [Writing](#writing) +- [Office](#office) +- [Lock](#lock) +- [Tool](#tool) +- [Science](#science) +- [Medical](#medical) +- [Household](#household) +- [Other Object](#other-object) + +### Clothing + +| | ico | shortcode | ico | shortcode | | +| - | :-: | - | :-: | - | - | +| [top](#objects) | :eyeglasses: | `:eyeglasses:` | :dark_sunglasses: | `:dark_sunglasses:` | [top](#introduction) | +| [top](#objects) | :goggles: | `:goggles:` | :lab_coat: | `:lab_coat:` | [top](#introduction) | +| [top](#objects) | :safety_vest: | `:safety_vest:` | :necktie: | `:necktie:` | [top](#introduction) | +| [top](#objects) | :shirt: | `:shirt:`
`:tshirt:` | :jeans: | `:jeans:` | [top](#introduction) | +| [top](#objects) | :scarf: | `:scarf:` | :gloves: | `:gloves:` | [top](#introduction) | +| [top](#objects) | :coat: | `:coat:` | :socks: | `:socks:` | [top](#introduction) | +| [top](#objects) | :dress: | `:dress:` | :kimono: | `:kimono:` | [top](#introduction) | +| [top](#objects) | :sari: | `:sari:` | :one_piece_swimsuit: | `:one_piece_swimsuit:` | [top](#introduction) | +| [top](#objects) | :swim_brief: | `:swim_brief:` | :shorts: | `:shorts:` | [top](#introduction) | +| [top](#objects) | :bikini: | `:bikini:` | :womans_clothes: | `:womans_clothes:` | [top](#introduction) | +| [top](#objects) | :purse: | `:purse:` | :handbag: | `:handbag:` | [top](#introduction) | +| [top](#objects) | :pouch: | `:pouch:` | :shopping: | `:shopping:` | [top](#introduction) | +| [top](#objects) | :school_satchel: | `:school_satchel:` | :thong_sandal: | `:thong_sandal:` | [top](#introduction) | +| [top](#objects) | :mans_shoe: | `:mans_shoe:`
`:shoe:` | :athletic_shoe: | `:athletic_shoe:` | [top](#introduction) | +| [top](#objects) | :hiking_boot: | `:hiking_boot:` | :flat_shoe: | `:flat_shoe:` | [top](#introduction) | +| [top](#objects) | :high_heel: | `:high_heel:` | :sandal: | `:sandal:` | [top](#introduction) | +| [top](#objects) | :ballet_shoes: | `:ballet_shoes:` | :boot: | `:boot:` | [top](#introduction) | +| [top](#objects) | :crown: | `:crown:` | :womans_hat: | `:womans_hat:` | [top](#introduction) | +| [top](#objects) | :tophat: | `:tophat:` | :mortar_board: | `:mortar_board:` | [top](#introduction) | +| [top](#objects) | :billed_cap: | `:billed_cap:` | :military_helmet: | `:military_helmet:` | [top](#introduction) | +| [top](#objects) | :rescue_worker_helmet: | `:rescue_worker_helmet:` | :prayer_beads: | `:prayer_beads:` | [top](#introduction) | +| [top](#objects) | :lipstick: | `:lipstick:` | :ring: | `:ring:` | [top](#introduction) | +| [top](#objects) | :gem: | `:gem:` | | | [top](#introduction) | + +### Sound + +| | ico | shortcode | ico | shortcode | | +| - | :-: | - | :-: | - | - | +| [top](#objects) | :mute: | `:mute:` | :speaker: | `:speaker:` | [top](#introduction) | +| [top](#objects) | :sound: | `:sound:` | :loud_sound: | `:loud_sound:` | [top](#introduction) | +| [top](#objects) | :loudspeaker: | `:loudspeaker:` | :mega: | `:mega:` | [top](#introduction) | +| [top](#objects) | :postal_horn: | `:postal_horn:` | :bell: | `:bell:` | [top](#introduction) | +| [top](#objects) | :no_bell: | `:no_bell:` | | | [top](#introduction) | + +### Music + +| | ico | shortcode | ico | shortcode | | +| - | :-: | - | :-: | - | - | +| [top](#objects) | :musical_score: | `:musical_score:` | :musical_note: | `:musical_note:` | [top](#introduction) | +| [top](#objects) | :notes: | `:notes:` | :studio_microphone: | `:studio_microphone:` | [top](#introduction) | +| [top](#objects) | :level_slider: | `:level_slider:` | :control_knobs: | `:control_knobs:` | [top](#introduction) | +| [top](#objects) | :microphone: | `:microphone:` | :headphones: | `:headphones:` | [top](#introduction) | +| [top](#objects) | :radio: | `:radio:` | | | [top](#introduction) | + +### Musical Instrument + +| | ico | shortcode | ico | shortcode | | +| - | :-: | - | :-: | - | - | +| [top](#objects) | :saxophone: | `:saxophone:` | :accordion: | `:accordion:` | [top](#introduction) | +| [top](#objects) | :guitar: | `:guitar:` | :musical_keyboard: | `:musical_keyboard:` | [top](#introduction) | +| [top](#objects) | :trumpet: | `:trumpet:` | :violin: | `:violin:` | [top](#introduction) | +| [top](#objects) | :banjo: | `:banjo:` | :drum: | `:drum:` | [top](#introduction) | +| [top](#objects) | :long_drum: | `:long_drum:` | | | [top](#introduction) | + +### Phone + +| | ico | shortcode | ico | shortcode | | +| - | :-: | - | :-: | - | - | +| [top](#objects) | :iphone: | `:iphone:` | :calling: | `:calling:` | [top](#introduction) | +| [top](#objects) | :phone: | `:phone:`
`:telephone:` | :telephone_receiver: | `:telephone_receiver:` | [top](#introduction) | +| [top](#objects) | :pager: | `:pager:` | :fax: | `:fax:` | [top](#introduction) | + +### Computer + +| | ico | shortcode | ico | shortcode | | +| - | :-: | - | :-: | - | - | +| [top](#objects) | :battery: | `:battery:` | :electric_plug: | `:electric_plug:` | [top](#introduction) | +| [top](#objects) | :computer: | `:computer:` | :desktop_computer: | `:desktop_computer:` | [top](#introduction) | +| [top](#objects) | :printer: | `:printer:` | :keyboard: | `:keyboard:` | [top](#introduction) | +| [top](#objects) | :computer_mouse: | `:computer_mouse:` | :trackball: | `:trackball:` | [top](#introduction) | +| [top](#objects) | :minidisc: | `:minidisc:` | :floppy_disk: | `:floppy_disk:` | [top](#introduction) | +| [top](#objects) | :cd: | `:cd:` | :dvd: | `:dvd:` | [top](#introduction) | +| [top](#objects) | :abacus: | `:abacus:` | | | [top](#introduction) | + +### Light & Video + +| | ico | shortcode | ico | shortcode | | +| - | :-: | - | :-: | - | - | +| [top](#objects) | :movie_camera: | `:movie_camera:` | :film_strip: | `:film_strip:` | [top](#introduction) | +| [top](#objects) | :film_projector: | `:film_projector:` | :clapper: | `:clapper:` | [top](#introduction) | +| [top](#objects) | :tv: | `:tv:` | :camera: | `:camera:` | [top](#introduction) | +| [top](#objects) | :camera_flash: | `:camera_flash:` | :video_camera: | `:video_camera:` | [top](#introduction) | +| [top](#objects) | :vhs: | `:vhs:` | :mag: | `:mag:` | [top](#introduction) | +| [top](#objects) | :mag_right: | `:mag_right:` | :candle: | `:candle:` | [top](#introduction) | +| [top](#objects) | :bulb: | `:bulb:` | :flashlight: | `:flashlight:` | [top](#introduction) | +| [top](#objects) | :izakaya_lantern: | `:izakaya_lantern:`
`:lantern:` | :diya_lamp: | `:diya_lamp:` | [top](#introduction) | + +### Book Paper + +| | ico | shortcode | ico | shortcode | | +| - | :-: | - | :-: | - | - | +| [top](#objects) | :notebook_with_decorative_cover: | `:notebook_with_decorative_cover:` | :closed_book: | `:closed_book:` | [top](#introduction) | +| [top](#objects) | :book: | `:book:`
`:open_book:` | :green_book: | `:green_book:` | [top](#introduction) | +| [top](#objects) | :blue_book: | `:blue_book:` | :orange_book: | `:orange_book:` | [top](#introduction) | +| [top](#objects) | :books: | `:books:` | :notebook: | `:notebook:` | [top](#introduction) | +| [top](#objects) | :ledger: | `:ledger:` | :page_with_curl: | `:page_with_curl:` | [top](#introduction) | +| [top](#objects) | :scroll: | `:scroll:` | :page_facing_up: | `:page_facing_up:` | [top](#introduction) | +| [top](#objects) | :newspaper: | `:newspaper:` | :newspaper_roll: | `:newspaper_roll:` | [top](#introduction) | +| [top](#objects) | :bookmark_tabs: | `:bookmark_tabs:` | :bookmark: | `:bookmark:` | [top](#introduction) | +| [top](#objects) | :label: | `:label:` | | | [top](#introduction) | + +### Money + +| | ico | shortcode | ico | shortcode | | +| - | :-: | - | :-: | - | - | +| [top](#objects) | :moneybag: | `:moneybag:` | :coin: | `:coin:` | [top](#introduction) | +| [top](#objects) | :yen: | `:yen:` | :dollar: | `:dollar:` | [top](#introduction) | +| [top](#objects) | :euro: | `:euro:` | :pound: | `:pound:` | [top](#introduction) | +| [top](#objects) | :money_with_wings: | `:money_with_wings:` | :credit_card: | `:credit_card:` | [top](#introduction) | +| [top](#objects) | :receipt: | `:receipt:` | :chart: | `:chart:` | [top](#introduction) | + +### Mail + +| | ico | shortcode | ico | shortcode | | +| - | :-: | - | :-: | - | - | +| [top](#objects) | :envelope: | `:envelope:` | :e-mail: | `:e-mail:`
`:email:` | [top](#introduction) | +| [top](#objects) | :incoming_envelope: | `:incoming_envelope:` | :envelope_with_arrow: | `:envelope_with_arrow:` | [top](#introduction) | +| [top](#objects) | :outbox_tray: | `:outbox_tray:` | :inbox_tray: | `:inbox_tray:` | [top](#introduction) | +| [top](#objects) | :package: | `:package:` | :mailbox: | `:mailbox:` | [top](#introduction) | +| [top](#objects) | :mailbox_closed: | `:mailbox_closed:` | :mailbox_with_mail: | `:mailbox_with_mail:` | [top](#introduction) | +| [top](#objects) | :mailbox_with_no_mail: | `:mailbox_with_no_mail:` | :postbox: | `:postbox:` | [top](#introduction) | +| [top](#objects) | :ballot_box: | `:ballot_box:` | | | [top](#introduction) | + +### Writing + +| | ico | shortcode | ico | shortcode | | +| - | :-: | - | :-: | - | - | +| [top](#objects) | :pencil2: | `:pencil2:` | :black_nib: | `:black_nib:` | [top](#introduction) | +| [top](#objects) | :fountain_pen: | `:fountain_pen:` | :pen: | `:pen:` | [top](#introduction) | +| [top](#objects) | :paintbrush: | `:paintbrush:` | :crayon: | `:crayon:` | [top](#introduction) | +| [top](#objects) | :memo: | `:memo:`
`:pencil:` | | | [top](#introduction) | + +### Office + +| | ico | shortcode | ico | shortcode | | +| - | :-: | - | :-: | - | - | +| [top](#objects) | :briefcase: | `:briefcase:` | :file_folder: | `:file_folder:` | [top](#introduction) | +| [top](#objects) | :open_file_folder: | `:open_file_folder:` | :card_index_dividers: | `:card_index_dividers:` | [top](#introduction) | +| [top](#objects) | :date: | `:date:` | :calendar: | `:calendar:` | [top](#introduction) | +| [top](#objects) | :spiral_notepad: | `:spiral_notepad:` | :spiral_calendar: | `:spiral_calendar:` | [top](#introduction) | +| [top](#objects) | :card_index: | `:card_index:` | :chart_with_upwards_trend: | `:chart_with_upwards_trend:` | [top](#introduction) | +| [top](#objects) | :chart_with_downwards_trend: | `:chart_with_downwards_trend:` | :bar_chart: | `:bar_chart:` | [top](#introduction) | +| [top](#objects) | :clipboard: | `:clipboard:` | :pushpin: | `:pushpin:` | [top](#introduction) | +| [top](#objects) | :round_pushpin: | `:round_pushpin:` | :paperclip: | `:paperclip:` | [top](#introduction) | +| [top](#objects) | :paperclips: | `:paperclips:` | :straight_ruler: | `:straight_ruler:` | [top](#introduction) | +| [top](#objects) | :triangular_ruler: | `:triangular_ruler:` | :scissors: | `:scissors:` | [top](#introduction) | +| [top](#objects) | :card_file_box: | `:card_file_box:` | :file_cabinet: | `:file_cabinet:` | [top](#introduction) | +| [top](#objects) | :wastebasket: | `:wastebasket:` | | | [top](#introduction) | + +### Lock + +| | ico | shortcode | ico | shortcode | | +| - | :-: | - | :-: | - | - | +| [top](#objects) | :lock: | `:lock:` | :unlock: | `:unlock:` | [top](#introduction) | +| [top](#objects) | :lock_with_ink_pen: | `:lock_with_ink_pen:` | :closed_lock_with_key: | `:closed_lock_with_key:` | [top](#introduction) | +| [top](#objects) | :key: | `:key:` | :old_key: | `:old_key:` | [top](#introduction) | + +### Tool + +| | ico | shortcode | ico | shortcode | | +| - | :-: | - | :-: | - | - | +| [top](#objects) | :hammer: | `:hammer:` | :axe: | `:axe:` | [top](#introduction) | +| [top](#objects) | :pick: | `:pick:` | :hammer_and_pick: | `:hammer_and_pick:` | [top](#introduction) | +| [top](#objects) | :hammer_and_wrench: | `:hammer_and_wrench:` | :dagger: | `:dagger:` | [top](#introduction) | +| [top](#objects) | :crossed_swords: | `:crossed_swords:` | :bomb: | `:bomb:` | [top](#introduction) | +| [top](#objects) | :boomerang: | `:boomerang:` | :bow_and_arrow: | `:bow_and_arrow:` | [top](#introduction) | +| [top](#objects) | :shield: | `:shield:` | :carpentry_saw: | `:carpentry_saw:` | [top](#introduction) | +| [top](#objects) | :wrench: | `:wrench:` | :screwdriver: | `:screwdriver:` | [top](#introduction) | +| [top](#objects) | :nut_and_bolt: | `:nut_and_bolt:` | :gear: | `:gear:` | [top](#introduction) | +| [top](#objects) | :clamp: | `:clamp:` | :balance_scale: | `:balance_scale:` | [top](#introduction) | +| [top](#objects) | :probing_cane: | `:probing_cane:` | :link: | `:link:` | [top](#introduction) | +| [top](#objects) | :chains: | `:chains:` | :hook: | `:hook:` | [top](#introduction) | +| [top](#objects) | :toolbox: | `:toolbox:` | :magnet: | `:magnet:` | [top](#introduction) | +| [top](#objects) | :ladder: | `:ladder:` | | | [top](#introduction) | + +### Science + +| | ico | shortcode | ico | shortcode | | +| - | :-: | - | :-: | - | - | +| [top](#objects) | :alembic: | `:alembic:` | :test_tube: | `:test_tube:` | [top](#introduction) | +| [top](#objects) | :petri_dish: | `:petri_dish:` | :dna: | `:dna:` | [top](#introduction) | +| [top](#objects) | :microscope: | `:microscope:` | :telescope: | `:telescope:` | [top](#introduction) | +| [top](#objects) | :satellite: | `:satellite:` | | | [top](#introduction) | + +### Medical + +| | ico | shortcode | ico | shortcode | | +| - | :-: | - | :-: | - | - | +| [top](#objects) | :syringe: | `:syringe:` | :drop_of_blood: | `:drop_of_blood:` | [top](#introduction) | +| [top](#objects) | :pill: | `:pill:` | :adhesive_bandage: | `:adhesive_bandage:` | [top](#introduction) | +| [top](#objects) | :stethoscope: | `:stethoscope:` | | | [top](#introduction) | + +### Household + +| | ico | shortcode | ico | shortcode | | +| - | :-: | - | :-: | - | - | +| [top](#objects) | :door: | `:door:` | :elevator: | `:elevator:` | [top](#introduction) | +| [top](#objects) | :mirror: | `:mirror:` | :window: | `:window:` | [top](#introduction) | +| [top](#objects) | :bed: | `:bed:` | :couch_and_lamp: | `:couch_and_lamp:` | [top](#introduction) | +| [top](#objects) | :chair: | `:chair:` | :toilet: | `:toilet:` | [top](#introduction) | +| [top](#objects) | :plunger: | `:plunger:` | :shower: | `:shower:` | [top](#introduction) | +| [top](#objects) | :bathtub: | `:bathtub:` | :mouse_trap: | `:mouse_trap:` | [top](#introduction) | +| [top](#objects) | :razor: | `:razor:` | :lotion_bottle: | `:lotion_bottle:` | [top](#introduction) | +| [top](#objects) | :safety_pin: | `:safety_pin:` | :broom: | `:broom:` | [top](#introduction) | +| [top](#objects) | :basket: | `:basket:` | :roll_of_paper: | `:roll_of_paper:` | [top](#introduction) | +| [top](#objects) | :bucket: | `:bucket:` | :soap: | `:soap:` | [top](#introduction) | +| [top](#objects) | :toothbrush: | `:toothbrush:` | :sponge: | `:sponge:` | [top](#introduction) | +| [top](#objects) | :fire_extinguisher: | `:fire_extinguisher:` | :shopping_cart: | `:shopping_cart:` | [top](#introduction) | + +### Other Object + +| | ico | shortcode | ico | shortcode | | +| - | :-: | - | :-: | - | - | +| [top](#objects) | :smoking: | `:smoking:` | :coffin: | `:coffin:` | [top](#introduction) | +| [top](#objects) | :headstone: | `:headstone:` | :funeral_urn: | `:funeral_urn:` | [top](#introduction) | +| [top](#objects) | :nazar_amulet: | `:nazar_amulet:` | :moyai: | `:moyai:` | [top](#introduction) | +| [top](#objects) | :placard: | `:placard:` | | | [top](#introduction) | + +## Symbols + +- [Transport Sign](#transport-sign) +- [Warning](#warning) +- [Arrow](#arrow) +- [Religion](#religion) +- [Zodiac](#zodiac) +- [Av Symbol](#av-symbol) +- [Gender](#gender) +- [Math](#math) +- [Punctuation](#punctuation) +- [Currency](#currency) +- [Other Symbol](#other-symbol) +- [Keycap](#keycap) +- [Alphanum](#alphanum) +- [Geometric](#geometric) + +### Transport Sign + +| | ico | shortcode | ico | shortcode | | +| - | :-: | - | :-: | - | - | +| [top](#symbols) | :atm: | `:atm:` | :put_litter_in_its_place: | `:put_litter_in_its_place:` | [top](#introduction) | +| [top](#symbols) | :potable_water: | `:potable_water:` | :wheelchair: | `:wheelchair:` | [top](#introduction) | +| [top](#symbols) | :mens: | `:mens:` | :womens: | `:womens:` | [top](#introduction) | +| [top](#symbols) | :restroom: | `:restroom:` | :baby_symbol: | `:baby_symbol:` | [top](#introduction) | +| [top](#symbols) | :wc: | `:wc:` | :passport_control: | `:passport_control:` | [top](#introduction) | +| [top](#symbols) | :customs: | `:customs:` | :baggage_claim: | `:baggage_claim:` | [top](#introduction) | +| [top](#symbols) | :left_luggage: | `:left_luggage:` | | | [top](#introduction) | + +### Warning + +| | ico | shortcode | ico | shortcode | | +| - | :-: | - | :-: | - | - | +| [top](#symbols) | :warning: | `:warning:` | :children_crossing: | `:children_crossing:` | [top](#introduction) | +| [top](#symbols) | :no_entry: | `:no_entry:` | :no_entry_sign: | `:no_entry_sign:` | [top](#introduction) | +| [top](#symbols) | :no_bicycles: | `:no_bicycles:` | :no_smoking: | `:no_smoking:` | [top](#introduction) | +| [top](#symbols) | :do_not_litter: | `:do_not_litter:` | :non-potable_water: | `:non-potable_water:` | [top](#introduction) | +| [top](#symbols) | :no_pedestrians: | `:no_pedestrians:` | :no_mobile_phones: | `:no_mobile_phones:` | [top](#introduction) | +| [top](#symbols) | :underage: | `:underage:` | :radioactive: | `:radioactive:` | [top](#introduction) | +| [top](#symbols) | :biohazard: | `:biohazard:` | | | [top](#introduction) | + +### Arrow + +| | ico | shortcode | ico | shortcode | | +| - | :-: | - | :-: | - | - | +| [top](#symbols) | :arrow_up: | `:arrow_up:` | :arrow_upper_right: | `:arrow_upper_right:` | [top](#introduction) | +| [top](#symbols) | :arrow_right: | `:arrow_right:` | :arrow_lower_right: | `:arrow_lower_right:` | [top](#introduction) | +| [top](#symbols) | :arrow_down: | `:arrow_down:` | :arrow_lower_left: | `:arrow_lower_left:` | [top](#introduction) | +| [top](#symbols) | :arrow_left: | `:arrow_left:` | :arrow_upper_left: | `:arrow_upper_left:` | [top](#introduction) | +| [top](#symbols) | :arrow_up_down: | `:arrow_up_down:` | :left_right_arrow: | `:left_right_arrow:` | [top](#introduction) | +| [top](#symbols) | :leftwards_arrow_with_hook: | `:leftwards_arrow_with_hook:` | :arrow_right_hook: | `:arrow_right_hook:` | [top](#introduction) | +| [top](#symbols) | :arrow_heading_up: | `:arrow_heading_up:` | :arrow_heading_down: | `:arrow_heading_down:` | [top](#introduction) | +| [top](#symbols) | :arrows_clockwise: | `:arrows_clockwise:` | :arrows_counterclockwise: | `:arrows_counterclockwise:` | [top](#introduction) | +| [top](#symbols) | :back: | `:back:` | :end: | `:end:` | [top](#introduction) | +| [top](#symbols) | :on: | `:on:` | :soon: | `:soon:` | [top](#introduction) | +| [top](#symbols) | :top: | `:top:` | | | [top](#introduction) | + +### Religion + +| | ico | shortcode | ico | shortcode | | +| - | :-: | - | :-: | - | - | +| [top](#symbols) | :place_of_worship: | `:place_of_worship:` | :atom_symbol: | `:atom_symbol:` | [top](#introduction) | +| [top](#symbols) | :om: | `:om:` | :star_of_david: | `:star_of_david:` | [top](#introduction) | +| [top](#symbols) | :wheel_of_dharma: | `:wheel_of_dharma:` | :yin_yang: | `:yin_yang:` | [top](#introduction) | +| [top](#symbols) | :latin_cross: | `:latin_cross:` | :orthodox_cross: | `:orthodox_cross:` | [top](#introduction) | +| [top](#symbols) | :star_and_crescent: | `:star_and_crescent:` | :peace_symbol: | `:peace_symbol:` | [top](#introduction) | +| [top](#symbols) | :menorah: | `:menorah:` | :six_pointed_star: | `:six_pointed_star:` | [top](#introduction) | + +### Zodiac + +| | ico | shortcode | ico | shortcode | | +| - | :-: | - | :-: | - | - | +| [top](#symbols) | :aries: | `:aries:` | :taurus: | `:taurus:` | [top](#introduction) | +| [top](#symbols) | :gemini: | `:gemini:` | :cancer: | `:cancer:` | [top](#introduction) | +| [top](#symbols) | :leo: | `:leo:` | :virgo: | `:virgo:` | [top](#introduction) | +| [top](#symbols) | :libra: | `:libra:` | :scorpius: | `:scorpius:` | [top](#introduction) | +| [top](#symbols) | :sagittarius: | `:sagittarius:` | :capricorn: | `:capricorn:` | [top](#introduction) | +| [top](#symbols) | :aquarius: | `:aquarius:` | :pisces: | `:pisces:` | [top](#introduction) | +| [top](#symbols) | :ophiuchus: | `:ophiuchus:` | | | [top](#introduction) | + +### Av Symbol + +| | ico | shortcode | ico | shortcode | | +| - | :-: | - | :-: | - | - | +| [top](#symbols) | :twisted_rightwards_arrows: | `:twisted_rightwards_arrows:` | :repeat: | `:repeat:` | [top](#introduction) | +| [top](#symbols) | :repeat_one: | `:repeat_one:` | :arrow_forward: | `:arrow_forward:` | [top](#introduction) | +| [top](#symbols) | :fast_forward: | `:fast_forward:` | :next_track_button: | `:next_track_button:` | [top](#introduction) | +| [top](#symbols) | :play_or_pause_button: | `:play_or_pause_button:` | :arrow_backward: | `:arrow_backward:` | [top](#introduction) | +| [top](#symbols) | :rewind: | `:rewind:` | :previous_track_button: | `:previous_track_button:` | [top](#introduction) | +| [top](#symbols) | :arrow_up_small: | `:arrow_up_small:` | :arrow_double_up: | `:arrow_double_up:` | [top](#introduction) | +| [top](#symbols) | :arrow_down_small: | `:arrow_down_small:` | :arrow_double_down: | `:arrow_double_down:` | [top](#introduction) | +| [top](#symbols) | :pause_button: | `:pause_button:` | :stop_button: | `:stop_button:` | [top](#introduction) | +| [top](#symbols) | :record_button: | `:record_button:` | :eject_button: | `:eject_button:` | [top](#introduction) | +| [top](#symbols) | :cinema: | `:cinema:` | :low_brightness: | `:low_brightness:` | [top](#introduction) | +| [top](#symbols) | :high_brightness: | `:high_brightness:` | :signal_strength: | `:signal_strength:` | [top](#introduction) | +| [top](#symbols) | :vibration_mode: | `:vibration_mode:` | :mobile_phone_off: | `:mobile_phone_off:` | [top](#introduction) | + +### Gender + +| | ico | shortcode | ico | shortcode | | +| - | :-: | - | :-: | - | - | +| [top](#symbols) | :female_sign: | `:female_sign:` | :male_sign: | `:male_sign:` | [top](#introduction) | +| [top](#symbols) | :transgender_symbol: | `:transgender_symbol:` | | | [top](#introduction) | + +### Math + +| | ico | shortcode | ico | shortcode | | +| - | :-: | - | :-: | - | - | +| [top](#symbols) | :heavy_multiplication_x: | `:heavy_multiplication_x:` | :heavy_plus_sign: | `:heavy_plus_sign:` | [top](#introduction) | +| [top](#symbols) | :heavy_minus_sign: | `:heavy_minus_sign:` | :heavy_division_sign: | `:heavy_division_sign:` | [top](#introduction) | +| [top](#symbols) | :infinity: | `:infinity:` | | | [top](#introduction) | + +### Punctuation + +| | ico | shortcode | ico | shortcode | | +| - | :-: | - | :-: | - | - | +| [top](#symbols) | :bangbang: | `:bangbang:` | :interrobang: | `:interrobang:` | [top](#introduction) | +| [top](#symbols) | :question: | `:question:` | :grey_question: | `:grey_question:` | [top](#introduction) | +| [top](#symbols) | :grey_exclamation: | `:grey_exclamation:` | :exclamation: | `:exclamation:`
`:heavy_exclamation_mark:` | [top](#introduction) | +| [top](#symbols) | :wavy_dash: | `:wavy_dash:` | | | [top](#introduction) | + +### Currency + +| | ico | shortcode | ico | shortcode | | +| - | :-: | - | :-: | - | - | +| [top](#symbols) | :currency_exchange: | `:currency_exchange:` | :heavy_dollar_sign: | `:heavy_dollar_sign:` | [top](#introduction) | + +### Other Symbol + +| | ico | shortcode | ico | shortcode | | +| - | :-: | - | :-: | - | - | +| [top](#symbols) | :medical_symbol: | `:medical_symbol:` | :recycle: | `:recycle:` | [top](#introduction) | +| [top](#symbols) | :fleur_de_lis: | `:fleur_de_lis:` | :trident: | `:trident:` | [top](#introduction) | +| [top](#symbols) | :name_badge: | `:name_badge:` | :beginner: | `:beginner:` | [top](#introduction) | +| [top](#symbols) | :o: | `:o:` | :white_check_mark: | `:white_check_mark:` | [top](#introduction) | +| [top](#symbols) | :ballot_box_with_check: | `:ballot_box_with_check:` | :heavy_check_mark: | `:heavy_check_mark:` | [top](#introduction) | +| [top](#symbols) | :x: | `:x:` | :negative_squared_cross_mark: | `:negative_squared_cross_mark:` | [top](#introduction) | +| [top](#symbols) | :curly_loop: | `:curly_loop:` | :loop: | `:loop:` | [top](#introduction) | +| [top](#symbols) | :part_alternation_mark: | `:part_alternation_mark:` | :eight_spoked_asterisk: | `:eight_spoked_asterisk:` | [top](#introduction) | +| [top](#symbols) | :eight_pointed_black_star: | `:eight_pointed_black_star:` | :sparkle: | `:sparkle:` | [top](#introduction) | +| [top](#symbols) | :copyright: | `:copyright:` | :registered: | `:registered:` | [top](#introduction) | +| [top](#symbols) | :tm: | `:tm:` | | | [top](#introduction) | + +### Keycap + +| | ico | shortcode | ico | shortcode | | +| - | :-: | - | :-: | - | - | +| [top](#symbols) | :hash: | `:hash:` | :asterisk: | `:asterisk:` | [top](#introduction) | +| [top](#symbols) | :zero: | `:zero:` | :one: | `:one:` | [top](#introduction) | +| [top](#symbols) | :two: | `:two:` | :three: | `:three:` | [top](#introduction) | +| [top](#symbols) | :four: | `:four:` | :five: | `:five:` | [top](#introduction) | +| [top](#symbols) | :six: | `:six:` | :seven: | `:seven:` | [top](#introduction) | +| [top](#symbols) | :eight: | `:eight:` | :nine: | `:nine:` | [top](#introduction) | +| [top](#symbols) | :keycap_ten: | `:keycap_ten:` | | | [top](#introduction) | + +### Alphanum + +| | ico | shortcode | ico | shortcode | | +| - | :-: | - | :-: | - | - | +| [top](#symbols) | :capital_abcd: | `:capital_abcd:` | :abcd: | `:abcd:` | [top](#introduction) | +| [top](#symbols) | :1234: | `:1234:` | :symbols: | `:symbols:` | [top](#introduction) | +| [top](#symbols) | :abc: | `:abc:` | :a: | `:a:` | [top](#introduction) | +| [top](#symbols) | :ab: | `:ab:` | :b: | `:b:` | [top](#introduction) | +| [top](#symbols) | :cl: | `:cl:` | :cool: | `:cool:` | [top](#introduction) | +| [top](#symbols) | :free: | `:free:` | :information_source: | `:information_source:` | [top](#introduction) | +| [top](#symbols) | :id: | `:id:` | :m: | `:m:` | [top](#introduction) | +| [top](#symbols) | :new: | `:new:` | :ng: | `:ng:` | [top](#introduction) | +| [top](#symbols) | :o2: | `:o2:` | :ok: | `:ok:` | [top](#introduction) | +| [top](#symbols) | :parking: | `:parking:` | :sos: | `:sos:` | [top](#introduction) | +| [top](#symbols) | :up: | `:up:` | :vs: | `:vs:` | [top](#introduction) | +| [top](#symbols) | :koko: | `:koko:` | :sa: | `:sa:` | [top](#introduction) | +| [top](#symbols) | :u6708: | `:u6708:` | :u6709: | `:u6709:` | [top](#introduction) | +| [top](#symbols) | :u6307: | `:u6307:` | :ideograph_advantage: | `:ideograph_advantage:` | [top](#introduction) | +| [top](#symbols) | :u5272: | `:u5272:` | :u7121: | `:u7121:` | [top](#introduction) | +| [top](#symbols) | :u7981: | `:u7981:` | :accept: | `:accept:` | [top](#introduction) | +| [top](#symbols) | :u7533: | `:u7533:` | :u5408: | `:u5408:` | [top](#introduction) | +| [top](#symbols) | :u7a7a: | `:u7a7a:` | :congratulations: | `:congratulations:` | [top](#introduction) | +| [top](#symbols) | :secret: | `:secret:` | :u55b6: | `:u55b6:` | [top](#introduction) | +| [top](#symbols) | :u6e80: | `:u6e80:` | | | [top](#introduction) | + +### Geometric + +| | ico | shortcode | ico | shortcode | | +| - | :-: | - | :-: | - | - | +| [top](#symbols) | :red_circle: | `:red_circle:` | :orange_circle: | `:orange_circle:` | [top](#introduction) | +| [top](#symbols) | :yellow_circle: | `:yellow_circle:` | :green_circle: | `:green_circle:` | [top](#introduction) | +| [top](#symbols) | :large_blue_circle: | `:large_blue_circle:` | :purple_circle: | `:purple_circle:` | [top](#introduction) | +| [top](#symbols) | :brown_circle: | `:brown_circle:` | :black_circle: | `:black_circle:` | [top](#introduction) | +| [top](#symbols) | :white_circle: | `:white_circle:` | :red_square: | `:red_square:` | [top](#introduction) | +| [top](#symbols) | :orange_square: | `:orange_square:` | :yellow_square: | `:yellow_square:` | [top](#introduction) | +| [top](#symbols) | :green_square: | `:green_square:` | :blue_square: | `:blue_square:` | [top](#introduction) | +| [top](#symbols) | :purple_square: | `:purple_square:` | :brown_square: | `:brown_square:` | [top](#introduction) | +| [top](#symbols) | :black_large_square: | `:black_large_square:` | :white_large_square: | `:white_large_square:` | [top](#introduction) | +| [top](#symbols) | :black_medium_square: | `:black_medium_square:` | :white_medium_square: | `:white_medium_square:` | [top](#introduction) | +| [top](#symbols) | :black_medium_small_square: | `:black_medium_small_square:` | :white_medium_small_square: | `:white_medium_small_square:` | [top](#introduction) | +| [top](#symbols) | :black_small_square: | `:black_small_square:` | :white_small_square: | `:white_small_square:` | [top](#introduction) | +| [top](#symbols) | :large_orange_diamond: | `:large_orange_diamond:` | :large_blue_diamond: | `:large_blue_diamond:` | [top](#introduction) | +| [top](#symbols) | :small_orange_diamond: | `:small_orange_diamond:` | :small_blue_diamond: | `:small_blue_diamond:` | [top](#introduction) | +| [top](#symbols) | :small_red_triangle: | `:small_red_triangle:` | :small_red_triangle_down: | `:small_red_triangle_down:` | [top](#introduction) | +| [top](#symbols) | :diamond_shape_with_a_dot_inside: | `:diamond_shape_with_a_dot_inside:` | :radio_button: | `:radio_button:` | [top](#introduction) | +| [top](#symbols) | :white_square_button: | `:white_square_button:` | :black_square_button: | `:black_square_button:` | [top](#introduction) | + +## Flags + +- [Flag](#flag) +- [Country Flag](#country-flag) +- [Subdivision Flag](#subdivision-flag) + +### Flag + +| | ico | shortcode | ico | shortcode | | +| - | :-: | - | :-: | - | - | +| [top](#flags) | :checkered_flag: | `:checkered_flag:` | :triangular_flag_on_post: | `:triangular_flag_on_post:` | [top](#introduction) | +| [top](#flags) | :crossed_flags: | `:crossed_flags:` | :black_flag: | `:black_flag:` | [top](#introduction) | +| [top](#flags) | :white_flag: | `:white_flag:` | :rainbow_flag: | `:rainbow_flag:` | [top](#introduction) | +| [top](#flags) | :transgender_flag: | `:transgender_flag:` | :pirate_flag: | `:pirate_flag:` | [top](#introduction) | + +### Country Flag + +| | ico | shortcode | ico | shortcode | | +| - | :-: | - | :-: | - | - | +| [top](#flags) | :ascension_island: | `:ascension_island:` | :andorra: | `:andorra:` | [top](#introduction) | +| [top](#flags) | :united_arab_emirates: | `:united_arab_emirates:` | :afghanistan: | `:afghanistan:` | [top](#introduction) | +| [top](#flags) | :antigua_barbuda: | `:antigua_barbuda:` | :anguilla: | `:anguilla:` | [top](#introduction) | +| [top](#flags) | :albania: | `:albania:` | :armenia: | `:armenia:` | [top](#introduction) | +| [top](#flags) | :angola: | `:angola:` | :antarctica: | `:antarctica:` | [top](#introduction) | +| [top](#flags) | :argentina: | `:argentina:` | :american_samoa: | `:american_samoa:` | [top](#introduction) | +| [top](#flags) | :austria: | `:austria:` | :australia: | `:australia:` | [top](#introduction) | +| [top](#flags) | :aruba: | `:aruba:` | :aland_islands: | `:aland_islands:` | [top](#introduction) | +| [top](#flags) | :azerbaijan: | `:azerbaijan:` | :bosnia_herzegovina: | `:bosnia_herzegovina:` | [top](#introduction) | +| [top](#flags) | :barbados: | `:barbados:` | :bangladesh: | `:bangladesh:` | [top](#introduction) | +| [top](#flags) | :belgium: | `:belgium:` | :burkina_faso: | `:burkina_faso:` | [top](#introduction) | +| [top](#flags) | :bulgaria: | `:bulgaria:` | :bahrain: | `:bahrain:` | [top](#introduction) | +| [top](#flags) | :burundi: | `:burundi:` | :benin: | `:benin:` | [top](#introduction) | +| [top](#flags) | :st_barthelemy: | `:st_barthelemy:` | :bermuda: | `:bermuda:` | [top](#introduction) | +| [top](#flags) | :brunei: | `:brunei:` | :bolivia: | `:bolivia:` | [top](#introduction) | +| [top](#flags) | :caribbean_netherlands: | `:caribbean_netherlands:` | :brazil: | `:brazil:` | [top](#introduction) | +| [top](#flags) | :bahamas: | `:bahamas:` | :bhutan: | `:bhutan:` | [top](#introduction) | +| [top](#flags) | :bouvet_island: | `:bouvet_island:` | :botswana: | `:botswana:` | [top](#introduction) | +| [top](#flags) | :belarus: | `:belarus:` | :belize: | `:belize:` | [top](#introduction) | +| [top](#flags) | :canada: | `:canada:` | :cocos_islands: | `:cocos_islands:` | [top](#introduction) | +| [top](#flags) | :congo_kinshasa: | `:congo_kinshasa:` | :central_african_republic: | `:central_african_republic:` | [top](#introduction) | +| [top](#flags) | :congo_brazzaville: | `:congo_brazzaville:` | :switzerland: | `:switzerland:` | [top](#introduction) | +| [top](#flags) | :cote_divoire: | `:cote_divoire:` | :cook_islands: | `:cook_islands:` | [top](#introduction) | +| [top](#flags) | :chile: | `:chile:` | :cameroon: | `:cameroon:` | [top](#introduction) | +| [top](#flags) | :cn: | `:cn:` | :colombia: | `:colombia:` | [top](#introduction) | +| [top](#flags) | :clipperton_island: | `:clipperton_island:` | :costa_rica: | `:costa_rica:` | [top](#introduction) | +| [top](#flags) | :cuba: | `:cuba:` | :cape_verde: | `:cape_verde:` | [top](#introduction) | +| [top](#flags) | :curacao: | `:curacao:` | :christmas_island: | `:christmas_island:` | [top](#introduction) | +| [top](#flags) | :cyprus: | `:cyprus:` | :czech_republic: | `:czech_republic:` | [top](#introduction) | +| [top](#flags) | :de: | `:de:` | :diego_garcia: | `:diego_garcia:` | [top](#introduction) | +| [top](#flags) | :djibouti: | `:djibouti:` | :denmark: | `:denmark:` | [top](#introduction) | +| [top](#flags) | :dominica: | `:dominica:` | :dominican_republic: | `:dominican_republic:` | [top](#introduction) | +| [top](#flags) | :algeria: | `:algeria:` | :ceuta_melilla: | `:ceuta_melilla:` | [top](#introduction) | +| [top](#flags) | :ecuador: | `:ecuador:` | :estonia: | `:estonia:` | [top](#introduction) | +| [top](#flags) | :egypt: | `:egypt:` | :western_sahara: | `:western_sahara:` | [top](#introduction) | +| [top](#flags) | :eritrea: | `:eritrea:` | :es: | `:es:` | [top](#introduction) | +| [top](#flags) | :ethiopia: | `:ethiopia:` | :eu: | `:eu:`
`:european_union:` | [top](#introduction) | +| [top](#flags) | :finland: | `:finland:` | :fiji: | `:fiji:` | [top](#introduction) | +| [top](#flags) | :falkland_islands: | `:falkland_islands:` | :micronesia: | `:micronesia:` | [top](#introduction) | +| [top](#flags) | :faroe_islands: | `:faroe_islands:` | :fr: | `:fr:` | [top](#introduction) | +| [top](#flags) | :gabon: | `:gabon:` | :gb: | `:gb:`
`:uk:` | [top](#introduction) | +| [top](#flags) | :grenada: | `:grenada:` | :georgia: | `:georgia:` | [top](#introduction) | +| [top](#flags) | :french_guiana: | `:french_guiana:` | :guernsey: | `:guernsey:` | [top](#introduction) | +| [top](#flags) | :ghana: | `:ghana:` | :gibraltar: | `:gibraltar:` | [top](#introduction) | +| [top](#flags) | :greenland: | `:greenland:` | :gambia: | `:gambia:` | [top](#introduction) | +| [top](#flags) | :guinea: | `:guinea:` | :guadeloupe: | `:guadeloupe:` | [top](#introduction) | +| [top](#flags) | :equatorial_guinea: | `:equatorial_guinea:` | :greece: | `:greece:` | [top](#introduction) | +| [top](#flags) | :south_georgia_south_sandwich_islands: | `:south_georgia_south_sandwich_islands:` | :guatemala: | `:guatemala:` | [top](#introduction) | +| [top](#flags) | :guam: | `:guam:` | :guinea_bissau: | `:guinea_bissau:` | [top](#introduction) | +| [top](#flags) | :guyana: | `:guyana:` | :hong_kong: | `:hong_kong:` | [top](#introduction) | +| [top](#flags) | :heard_mcdonald_islands: | `:heard_mcdonald_islands:` | :honduras: | `:honduras:` | [top](#introduction) | +| [top](#flags) | :croatia: | `:croatia:` | :haiti: | `:haiti:` | [top](#introduction) | +| [top](#flags) | :hungary: | `:hungary:` | :canary_islands: | `:canary_islands:` | [top](#introduction) | +| [top](#flags) | :indonesia: | `:indonesia:` | :ireland: | `:ireland:` | [top](#introduction) | +| [top](#flags) | :israel: | `:israel:` | :isle_of_man: | `:isle_of_man:` | [top](#introduction) | +| [top](#flags) | :india: | `:india:` | :british_indian_ocean_territory: | `:british_indian_ocean_territory:` | [top](#introduction) | +| [top](#flags) | :iraq: | `:iraq:` | :iran: | `:iran:` | [top](#introduction) | +| [top](#flags) | :iceland: | `:iceland:` | :it: | `:it:` | [top](#introduction) | +| [top](#flags) | :jersey: | `:jersey:` | :jamaica: | `:jamaica:` | [top](#introduction) | +| [top](#flags) | :jordan: | `:jordan:` | :jp: | `:jp:` | [top](#introduction) | +| [top](#flags) | :kenya: | `:kenya:` | :kyrgyzstan: | `:kyrgyzstan:` | [top](#introduction) | +| [top](#flags) | :cambodia: | `:cambodia:` | :kiribati: | `:kiribati:` | [top](#introduction) | +| [top](#flags) | :comoros: | `:comoros:` | :st_kitts_nevis: | `:st_kitts_nevis:` | [top](#introduction) | +| [top](#flags) | :north_korea: | `:north_korea:` | :kr: | `:kr:` | [top](#introduction) | +| [top](#flags) | :kuwait: | `:kuwait:` | :cayman_islands: | `:cayman_islands:` | [top](#introduction) | +| [top](#flags) | :kazakhstan: | `:kazakhstan:` | :laos: | `:laos:` | [top](#introduction) | +| [top](#flags) | :lebanon: | `:lebanon:` | :st_lucia: | `:st_lucia:` | [top](#introduction) | +| [top](#flags) | :liechtenstein: | `:liechtenstein:` | :sri_lanka: | `:sri_lanka:` | [top](#introduction) | +| [top](#flags) | :liberia: | `:liberia:` | :lesotho: | `:lesotho:` | [top](#introduction) | +| [top](#flags) | :lithuania: | `:lithuania:` | :luxembourg: | `:luxembourg:` | [top](#introduction) | +| [top](#flags) | :latvia: | `:latvia:` | :libya: | `:libya:` | [top](#introduction) | +| [top](#flags) | :morocco: | `:morocco:` | :monaco: | `:monaco:` | [top](#introduction) | +| [top](#flags) | :moldova: | `:moldova:` | :montenegro: | `:montenegro:` | [top](#introduction) | +| [top](#flags) | :st_martin: | `:st_martin:` | :madagascar: | `:madagascar:` | [top](#introduction) | +| [top](#flags) | :marshall_islands: | `:marshall_islands:` | :macedonia: | `:macedonia:` | [top](#introduction) | +| [top](#flags) | :mali: | `:mali:` | :myanmar: | `:myanmar:` | [top](#introduction) | +| [top](#flags) | :mongolia: | `:mongolia:` | :macau: | `:macau:` | [top](#introduction) | +| [top](#flags) | :northern_mariana_islands: | `:northern_mariana_islands:` | :martinique: | `:martinique:` | [top](#introduction) | +| [top](#flags) | :mauritania: | `:mauritania:` | :montserrat: | `:montserrat:` | [top](#introduction) | +| [top](#flags) | :malta: | `:malta:` | :mauritius: | `:mauritius:` | [top](#introduction) | +| [top](#flags) | :maldives: | `:maldives:` | :malawi: | `:malawi:` | [top](#introduction) | +| [top](#flags) | :mexico: | `:mexico:` | :malaysia: | `:malaysia:` | [top](#introduction) | +| [top](#flags) | :mozambique: | `:mozambique:` | :namibia: | `:namibia:` | [top](#introduction) | +| [top](#flags) | :new_caledonia: | `:new_caledonia:` | :niger: | `:niger:` | [top](#introduction) | +| [top](#flags) | :norfolk_island: | `:norfolk_island:` | :nigeria: | `:nigeria:` | [top](#introduction) | +| [top](#flags) | :nicaragua: | `:nicaragua:` | :netherlands: | `:netherlands:` | [top](#introduction) | +| [top](#flags) | :norway: | `:norway:` | :nepal: | `:nepal:` | [top](#introduction) | +| [top](#flags) | :nauru: | `:nauru:` | :niue: | `:niue:` | [top](#introduction) | +| [top](#flags) | :new_zealand: | `:new_zealand:` | :oman: | `:oman:` | [top](#introduction) | +| [top](#flags) | :panama: | `:panama:` | :peru: | `:peru:` | [top](#introduction) | +| [top](#flags) | :french_polynesia: | `:french_polynesia:` | :papua_new_guinea: | `:papua_new_guinea:` | [top](#introduction) | +| [top](#flags) | :philippines: | `:philippines:` | :pakistan: | `:pakistan:` | [top](#introduction) | +| [top](#flags) | :poland: | `:poland:` | :st_pierre_miquelon: | `:st_pierre_miquelon:` | [top](#introduction) | +| [top](#flags) | :pitcairn_islands: | `:pitcairn_islands:` | :puerto_rico: | `:puerto_rico:` | [top](#introduction) | +| [top](#flags) | :palestinian_territories: | `:palestinian_territories:` | :portugal: | `:portugal:` | [top](#introduction) | +| [top](#flags) | :palau: | `:palau:` | :paraguay: | `:paraguay:` | [top](#introduction) | +| [top](#flags) | :qatar: | `:qatar:` | :reunion: | `:reunion:` | [top](#introduction) | +| [top](#flags) | :romania: | `:romania:` | :serbia: | `:serbia:` | [top](#introduction) | +| [top](#flags) | :ru: | `:ru:` | :rwanda: | `:rwanda:` | [top](#introduction) | +| [top](#flags) | :saudi_arabia: | `:saudi_arabia:` | :solomon_islands: | `:solomon_islands:` | [top](#introduction) | +| [top](#flags) | :seychelles: | `:seychelles:` | :sudan: | `:sudan:` | [top](#introduction) | +| [top](#flags) | :sweden: | `:sweden:` | :singapore: | `:singapore:` | [top](#introduction) | +| [top](#flags) | :st_helena: | `:st_helena:` | :slovenia: | `:slovenia:` | [top](#introduction) | +| [top](#flags) | :svalbard_jan_mayen: | `:svalbard_jan_mayen:` | :slovakia: | `:slovakia:` | [top](#introduction) | +| [top](#flags) | :sierra_leone: | `:sierra_leone:` | :san_marino: | `:san_marino:` | [top](#introduction) | +| [top](#flags) | :senegal: | `:senegal:` | :somalia: | `:somalia:` | [top](#introduction) | +| [top](#flags) | :suriname: | `:suriname:` | :south_sudan: | `:south_sudan:` | [top](#introduction) | +| [top](#flags) | :sao_tome_principe: | `:sao_tome_principe:` | :el_salvador: | `:el_salvador:` | [top](#introduction) | +| [top](#flags) | :sint_maarten: | `:sint_maarten:` | :syria: | `:syria:` | [top](#introduction) | +| [top](#flags) | :swaziland: | `:swaziland:` | :tristan_da_cunha: | `:tristan_da_cunha:` | [top](#introduction) | +| [top](#flags) | :turks_caicos_islands: | `:turks_caicos_islands:` | :chad: | `:chad:` | [top](#introduction) | +| [top](#flags) | :french_southern_territories: | `:french_southern_territories:` | :togo: | `:togo:` | [top](#introduction) | +| [top](#flags) | :thailand: | `:thailand:` | :tajikistan: | `:tajikistan:` | [top](#introduction) | +| [top](#flags) | :tokelau: | `:tokelau:` | :timor_leste: | `:timor_leste:` | [top](#introduction) | +| [top](#flags) | :turkmenistan: | `:turkmenistan:` | :tunisia: | `:tunisia:` | [top](#introduction) | +| [top](#flags) | :tonga: | `:tonga:` | :tr: | `:tr:` | [top](#introduction) | +| [top](#flags) | :trinidad_tobago: | `:trinidad_tobago:` | :tuvalu: | `:tuvalu:` | [top](#introduction) | +| [top](#flags) | :taiwan: | `:taiwan:` | :tanzania: | `:tanzania:` | [top](#introduction) | +| [top](#flags) | :ukraine: | `:ukraine:` | :uganda: | `:uganda:` | [top](#introduction) | +| [top](#flags) | :us_outlying_islands: | `:us_outlying_islands:` | :united_nations: | `:united_nations:` | [top](#introduction) | +| [top](#flags) | :us: | `:us:` | :uruguay: | `:uruguay:` | [top](#introduction) | +| [top](#flags) | :uzbekistan: | `:uzbekistan:` | :vatican_city: | `:vatican_city:` | [top](#introduction) | +| [top](#flags) | :st_vincent_grenadines: | `:st_vincent_grenadines:` | :venezuela: | `:venezuela:` | [top](#introduction) | +| [top](#flags) | :british_virgin_islands: | `:british_virgin_islands:` | :us_virgin_islands: | `:us_virgin_islands:` | [top](#introduction) | +| [top](#flags) | :vietnam: | `:vietnam:` | :vanuatu: | `:vanuatu:` | [top](#introduction) | +| [top](#flags) | :wallis_futuna: | `:wallis_futuna:` | :samoa: | `:samoa:` | [top](#introduction) | +| [top](#flags) | :kosovo: | `:kosovo:` | :yemen: | `:yemen:` | [top](#introduction) | +| [top](#flags) | :mayotte: | `:mayotte:` | :south_africa: | `:south_africa:` | [top](#introduction) | +| [top](#flags) | :zambia: | `:zambia:` | :zimbabwe: | `:zimbabwe:` | [top](#introduction) | + +### Subdivision Flag + +| | ico | shortcode | ico | shortcode | | +| - | :-: | - | :-: | - | - | +| [top](#flags) | :england: | `:england:` | :scotland: | `:scotland:` | [top](#introduction) | +| [top](#flags) | :wales: | `:wales:` | | | [top](#introduction) | + +## GitHub Custom Emoji + +| | ico | shortcode | ico | shortcode | | +| - | :-: | - | :-: | - | - | +| [top](#github-custom-emoji) | :accessibility: | `:accessibility:` | :atom: | `:atom:` | [top](#introduction) | +| [top](#github-custom-emoji) | :basecamp: | `:basecamp:` | :basecampy: | `:basecampy:` | [top](#introduction) | +| [top](#github-custom-emoji) | :bowtie: | `:bowtie:` | :dependabot: | `:dependabot:` | [top](#introduction) | +| [top](#github-custom-emoji) | :electron: | `:electron:` | :feelsgood: | `:feelsgood:` | [top](#introduction) | +| [top](#github-custom-emoji) | :finnadie: | `:finnadie:` | :fishsticks: | `:fishsticks:` | [top](#introduction) | +| [top](#github-custom-emoji) | :goberserk: | `:goberserk:` | :godmode: | `:godmode:` | [top](#introduction) | +| [top](#github-custom-emoji) | :hurtrealbad: | `:hurtrealbad:` | :neckbeard: | `:neckbeard:` | [top](#introduction) | +| [top](#github-custom-emoji) | :octocat: | `:octocat:` | :rage1: | `:rage1:` | [top](#introduction) | +| [top](#github-custom-emoji) | :rage2: | `:rage2:` | :rage3: | `:rage3:` | [top](#introduction) | +| [top](#github-custom-emoji) | :rage4: | `:rage4:` | :shipit: | `:shipit:` | [top](#introduction) | +| [top](#github-custom-emoji) | :suspect: | `:suspect:` | :trollface: | `:trollface:` | [top](#introduction) | diff --git a/content/en/content-management/formats.md b/content/en/content-management/formats.md index ccc45b943fc..85ca55ba776 100644 --- a/content/en/content-management/formats.md +++ b/content/en/content-management/formats.md @@ -24,7 +24,7 @@ The current list of content formats in Hugo: | Name | Markup identifiers | Comment | | ------------- | ------------- |-------------| -| Goldmark | md, markdown, goldmark |Note that you can set the default handler of `md` and `markdown` to something else, see [Configure Markup](/getting-started/configuration-markup/).| +| Goldmark | markdown, goldmark |Note that you can set the default handler of `md` and `markdown` to something else, see [Configure Markup](/getting-started/configuration-markup/).| |Emacs Org-Mode|org|See [go-org](https://github.com/niklasfasching/go-org).| |AsciiDoc|asciidocext, adoc, ad|Needs [Asciidoctor][ascii] installed.| |RST|rst|Needs [RST](https://docutils.sourceforge.io/rst.html) installed.| @@ -59,7 +59,7 @@ optional extensions like `asciidoctor-diagram` or `asciidoctor-html5s` are insta External `asciidoctor` command requires Hugo rendering to _disk_ to a specific destination directory. It is required to run Hugo with the command option `--destination`. {{% /note %}} -Some Asciidoctor parameters can be customized in Hugo. See [details]. +Some Asciidoctor parameters can be customized in Hugo. See [details]. [details]: /getting-started/configuration-markup/#asciidoc @@ -75,7 +75,6 @@ Markdown syntax is simple enough to learn in a single sitting. The following are [ascii]: https://asciidoctor.org/ [config]: /getting-started/configuration/ [developer tools]: /tools/ -[emojis]: https://www.webpagefx.com/tools/emoji-cheat-sheet/ [fireball]: https://daringfireball.net/projects/markdown/ [gfmtasks]: https://guides.github.com/features/mastering-markdown/#syntax [helperssource]: https://github.com/gohugoio/hugo/blob/77c60a3440806067109347d04eb5368b65ea0fe8/helpers/general.go#L65 diff --git a/content/en/content-management/front-matter.md b/content/en/content-management/front-matter.md index 10317e80566..38393e9ba5b 100644 --- a/content/en/content-management/front-matter.md +++ b/content/en/content-management/front-matter.md @@ -144,7 +144,7 @@ You can add fields to your front matter arbitrarily to meet your needs. These us The following fields can be accessed via `.Params.include_toc` and `.Params.show_comments`, respectively. The [Variables] section provides more information on using Hugo's page- and site-level variables in your templates. -{{< code-toggle copy=false >}} +{{< code-toggle >}} include_toc: true show_comments: false {{}} @@ -157,7 +157,7 @@ Any node or section can pass down to descendants a set of front matter values as The `cascade` block can be a slice with a optional `_target` keyword, allowing for multiple `cascade` values targeting different page sets. -{{< code-toggle copy=false >}} +{{< code-toggle >}} title ="Blog" [[cascade]] background = "yosemite.jpg" @@ -191,7 +191,7 @@ Any of the above can be omitted. In `content/blog/_index.md` -{{< code-toggle copy=false >}} +{{< code-toggle >}} title: Blog cascade: banner: images/typewriter.jpg diff --git a/content/en/content-management/image-processing/index.md b/content/en/content-management/image-processing/index.md index 9cce9070a37..9ac1d5c4357 100644 --- a/content/en/content-management/image-processing/index.md +++ b/content/en/content-management/image-processing/index.md @@ -10,6 +10,7 @@ menu: toc: true weight: 90 --- + ## Image resources To process an image you must access the file as a page resource, global resource, or remote resource. @@ -50,7 +51,7 @@ To access an image as a global resource: ### Remote resource -A remote resource is a file on a remote server, accessible via http or https. To access an image as a remote resource: +A remote resource is a file on a remote server, accessible via HTTP or HTTPS. To access an image as a remote resource: ```go-html-template {{ $image := resources.GetRemote "https://gohugo.io/img/hugo-logo.png" }} @@ -112,7 +113,7 @@ Metadata (EXIF, IPTC, XMP, etc.) is not preserved during image transformation. U {{< new-in "0.119.0" >}} {{% note %}} -The `Process` method is also available as a filter, which is more effective if need to apply multiple filters to an image. See [Process filter](/functions/images/#process). +The `Process` method is also available as a filter, which is more effective if you need to apply multiple filters to an image. See [Process filter](/functions/images/process). {{% /note %}} Process processes the image with the given specification. The specification can contain an optional action, one of `resize`, `crop`, `fit` or `fill`. This means that you can use this method instead of [`Resize`], [`Fit`], [`Fill`], or [`Crop`]. @@ -139,10 +140,9 @@ Some more examples: {{ $image := $image.Process "fill 600x400" }} ``` - ### Resize -Resize an image to the specified width and/or height. +Resize an image to the given width and/or height. If you specify both width and height, the resulting image will be disproportionally scaled unless the original image has the same aspect ratio. @@ -215,7 +215,6 @@ Sometimes it can be useful to create the filter chain once and then reuse it. This method is fast, but if you also scale down your images, it would be good for performance to extract the colors from the scaled down image. - ### EXIF Provides an [EXIF] object containing image metadata. @@ -266,7 +265,7 @@ You may also access EXIF fields individually, using the [`lang.FormatNumber`] fu ## Image processing options -The [`Resize`], [`Fit`], [`Fill`], and [`Crop`] methods accept a space-separated, case-insensitive list of options. The order of the options within the list is irrelevant. +The [`Resize`], [`Fit`], [`Fill`], and [`Crop`] methods accept a space-delimited, case-insensitive list of options. The order of the options within the list is irrelevant. ### Dimensions @@ -369,7 +368,7 @@ The default value is `photo`. You may override the default value in the [site co When converting an image from a format that supports transparency (e.g., PNG) to a format that does _not_ support transparency (e.g., JPEG), you may specify the background color of the resulting image. -Use either a 3-digit or a 6-digit hexadecimal color code (e.g., `#00f` or `#0000ff`). +Use either a 3-digit or 6-digit hexadecimal color code (e.g., `#00f` or `#0000ff`). The default value is `#ffffff` (white). You may override the default value in the [site configuration]. @@ -402,28 +401,26 @@ See [github.com/disintegration/imaging] for the complete list of resampling filt _The photo of the sunset used in the examples below is Copyright [Bjørn Erik Pedersen](https://commons.wikimedia.org/wiki/User:Bep) (Creative Commons Attribution-Share Alike 4.0 International license)_ -{{< imgproc sunset Resize "300x" />}} +{{< imgproc "sunset.jpg" "resize 300x" />}} -{{< imgproc sunset Fill "90x120 left" />}} +{{< imgproc "sunset.jpg" "fill 90x120 left" />}} -{{< imgproc sunset Fill "90x120 right" />}} +{{< imgproc "sunset.jpg" "fill 90x120 right" />}} -{{< imgproc sunset Fit "90x90" />}} +{{< imgproc "sunset.jpg" "fit 90x90" />}} -{{< imgproc sunset Crop "250x250 center" />}} +{{< imgproc "sunset.jpg" "crop 250x250 center" />}} -{{< imgproc sunset Resize "300x q10" />}} +{{< imgproc "sunset.jpg" "resize 300x q10" />}} This is the shortcode used to generate the examples above: -{{< code file="layouts/shortcodes/imgproc.html" >}} -{{< readfile file="layouts/shortcodes/imgproc.html" >}} -{{< /code >}} +{{< readfile file="layouts/shortcodes/imgproc.html" highlight="go-html-template" >}} Call the shortcode from your Markdown like this: ```go-html-template -{{}} +{{}} ``` {{% note %}} @@ -457,7 +454,7 @@ resampleFilter Define an `imaging.exif` section in your site configuration to control the availability of EXIF data. -{{< code-toggle file="hugo" copy=true >}} +{{< code-toggle file=hugo >}} [imaging.exif] includeFields = "" excludeFields = "" @@ -487,9 +484,9 @@ By default, Hugo uses the [Smartcrop] library when cropping images with the `Cro Examples using the sunset image from above: -{{< imgproc sunset Fill "200x200 smart" />}} +{{< imgproc "sunset.jpg" "fill 200x200 smart" />}} -{{< imgproc sunset Crop "200x200 smart" />}} +{{< imgproc "sunset.jpg" "crop 200x200 smart" />}} ## Image processing performance consideration @@ -497,7 +494,7 @@ Hugo caches processed images in the `resources` directory. If you include this d If you change image processing methods or options, or if you rename or remove images, the `resources` directory will contain unused images. To remove the unused images, perform garbage collection with: -```bash +```sh hugo --gc ``` diff --git a/content/en/content-management/menus.md b/content/en/content-management/menus.md index 07bf4166977..f7f7cec4ccd 100644 --- a/content/en/content-management/menus.md +++ b/content/en/content-management/menus.md @@ -36,7 +36,7 @@ Although you can use these methods in combination when defining a menu, the menu To automatically define menu entries for each top-level section of your site, enable the section pages menu in your site configuration. -{{< code-toggle file="hugo" copy=false >}} +{{< code-toggle file=hugo >}} sectionPagesMenu = "main" {{< /code-toggle >}} @@ -46,7 +46,7 @@ This creates a menu structure that you can access with `site.Menus.main` in your To add a page to the "main" menu: -{{< code-toggle file="content/about.md" copy=false fm=true >}} +{{< code-toggle file="content/about.md" fm=true >}} title = 'About' menu = 'main' {{< /code-toggle >}} @@ -55,7 +55,7 @@ Access the entry with `site.Menus.main` in your templates. See [menu templates] To add a page to the "main" and "footer" menus: -{{< code-toggle file="content/contact.md" copy=false fm=true >}} +{{< code-toggle file="content/contact.md" fm=true >}} title = 'Contact' menu = ['main','footer'] {{< /code-toggle >}} @@ -94,7 +94,7 @@ weight This front matter menu entry demonstrates some of the available properties: -{{< code-toggle file="content/products/software.md" copy=false fm=true >}} +{{< code-toggle file="content/products/software.md" fm=true >}} title = 'Software' [menu.main] parent = 'Products' @@ -106,12 +106,11 @@ class = 'center' Access the entry with `site.Menus.main` in your templates. See [menu templates] for details. - ## Define in site configuration To define entries for the "main" menu: -{{< code-toggle file="hugo" copy=false >}} +{{< code-toggle file=hugo >}} [[menu.main]] name = 'Home' pageRef = '/' @@ -132,7 +131,7 @@ This creates a menu structure that you can access with `site.Menus.main` in your To define entries for the "footer" menu: -{{< code-toggle file="hugo" copy=false >}} +{{< code-toggle file=hugo >}} [[menu.footer]] name = 'Terms' pageRef = '/terms' @@ -177,7 +176,7 @@ url This nested menu demonstrates some of the available properties: -{{< code-toggle file="hugo" copy=false >}} +{{< code-toggle file=hugo >}} [[menu.main]] name = 'Products' pageRef = '/products' diff --git a/content/en/content-management/multilingual.md b/content/en/content-management/multilingual.md index 6e787c526dd..910af91ce13 100644 --- a/content/en/content-management/multilingual.md +++ b/content/en/content-management/multilingual.md @@ -25,7 +25,7 @@ This is the default language configuration: This is an example of a site configuration for a multilingual project. Any key not defined in a `languages` object will fall back to the global value in the root of your site configuration. -{{< code-toggle file="hugo" >}} +{{< code-toggle file=hugo >}} defaultContentLanguage = 'de' defaultContentLanguageInSubdir = true @@ -104,7 +104,7 @@ In Hugo `v0.112.0` we consolidated all configuration options, and improved how t 1. `site.Language.Params` is deprecated. Use `site.Params` directly. 1. Adding custom parameters to the top level language configuration is deprecated. Define custom parameters within `languages.xx.params`. See `color` in the example below. -{{< code-toggle file=hugo copy=false >}} +{{< code-toggle file=hugo >}} title = "My blog" languageCode = "en-us" @@ -129,20 +129,20 @@ In the example above, all settings except `color` below `params` map to predefin To disable a language within a `languages` object in your site configuration: -{{< code-toggle file="hugo" copy=false >}} +{{< code-toggle file=hugo >}} [languages.es] disabled = true {{< /code-toggle >}} To disable one or more languages in the root of your site configuration: -{{< code-toggle file="hugo" copy=false >}} +{{< code-toggle file=hugo >}} disableLanguages = ["es", "fr"] {{< /code-toggle >}} To disable one or more languages using an environment variable: -```bash +```sh HUGO_DISABLELANGUAGES="es fr" hugo ``` @@ -160,7 +160,7 @@ If a `baseURL` is set on the `language` level, then all languages must have one Example: -{{< code-toggle file="hugo" >}} +{{< code-toggle file=hugo >}} [languages] [languages.fr] baseURL = "https://example.fr" @@ -169,7 +169,7 @@ weight = 1 title = "En Français" [languages.en] -baseURL = "https://example.com" +baseURL = "https://example.org/" languageName = "English" weight = 2 title = "In English" @@ -183,7 +183,7 @@ public └── fr ``` -**All URLs (i.e `.Permalink` etc.) will be generated from that root. So the English home page above will have its `.Permalink` set to `https://example.com/`.** +**All URLs (i.e `.Permalink` etc.) will be generated from that root. So the English home page above will have its `.Permalink` set to `https://example.org/`.** When you run `hugo server` we will start multiple HTTP servers. You will typically see something like this in the console: @@ -221,7 +221,7 @@ If a file has no language code, it will be assigned the default language. This system uses different content directories for each of the languages. Each language's content directory is set using the `contentDir` parameter. -{{< code-toggle file="hugo" >}} +{{< code-toggle file=hugo >}} languages: en: weight: 10 @@ -277,7 +277,7 @@ To localize URLs: For example, a French translation can have its own localized slug. -{{< code-toggle file="content/about.fr.md" fm=true copy=false >}} +{{< code-toggle file="content/about.fr.md" fm=true >}} title: A Propos slug: "a-propos" {{< /code-toggle >}} @@ -427,7 +427,7 @@ In case you need to pass a custom data: (`(dict "Count" numeric_value_only)` is The following localization examples assume your site's primary language is English, with translations to French and German. -{{< code-toggle file="hugo" >}} +{{< code-toggle file=hugo >}} defaultContentLanguage = 'en' [languages] @@ -530,7 +530,7 @@ Localization of menu entries depends on how you define them: - When you define menu entries [automatically] using the section pages menu, you must use translation tables to localize each entry. - When you define menu entries [in front matter], they are already localized based on the front matter itself. If the front matter values are insufficient, use translation tables to localize each entry. -- When you define menu entries [in site configuration], you must create language-specific menu entries under each language key. If the names of the menu entries are insufficent, use translation tables to localize each entry. +- When you define menu entries [in site configuration], you must create language-specific menu entries under each language key. If the names of the menu entries are insufficient, use translation tables to localize each entry. ### Create language-specific menu entries @@ -538,7 +538,7 @@ Localization of menu entries depends on how you define them: For a simple menu with a small number of entries, use a single configuration file. For example: -{{< code-toggle file="hugo" copy=false >}} +{{< code-toggle file=hugo >}} [languages.de] languageCode = 'de-DE' languageName = 'Deutsch' @@ -583,7 +583,7 @@ config/ └── hugo.toml ``` -{{< code-toggle file="config/_default/menus/menu.de" copy=false >}} +{{< code-toggle file="config/_default/menus/menu.de" >}} [[main]] name = 'Produkte' pageRef = '/products' @@ -594,7 +594,7 @@ pageRef = '/services' weight = 20 {{< /code-toggle >}} -{{< code-toggle file="config/_default/menus/menu.en" copy=false >}} +{{< code-toggle file="config/_default/menus/menu.en" >}} [[main]] name = 'Products' pageRef = '/products' @@ -624,7 +624,7 @@ The `identifier` depends on how you define menu entries: For example, if you define menu entries in site configuration: -{{< code-toggle file="hugo" copy=false >}} +{{< code-toggle file=hugo >}} [[menu.main]] identifier = 'products' name = 'Products' @@ -639,7 +639,7 @@ For example, if you define menu entries in site configuration: Create corresponding entries in the translation tables: -{{< code-toggle file="i18n/de" copy=false >}} +{{< code-toggle file="i18n/de" >}} products = 'Produkte' services = 'Leistungen' {{< / code-toggle >}} @@ -663,7 +663,7 @@ For merging of content from other languages (i.e. missing content translations), To track down missing translation strings, run Hugo with the `--printI18nWarnings` flag: -```bash +```sh hugo --printI18nWarnings | grep i18n i18n|MISSING_TRANSLATION|en|wordCount ``` @@ -677,19 +677,18 @@ To support Multilingual mode in your themes, some considerations must be taken f If there is more than one language defined, the `LanguagePrefix` variable will equal `/en` (or whatever your `CurrentLanguage` is). If not enabled, it will be an empty string (and is therefore harmless for single-language Hugo websites). - ## Generate multilingual content with `hugo new content` If you organize content with translations in the same directory: -```text +```sh hugo new content post/test.en.md hugo new content post/test.de.md ``` If you organize content with translations in different directories: -```text +```sh hugo new content content/en/post/test.md hugo new content content/de/post/test.md ``` diff --git a/content/en/content-management/organization/index.md b/content/en/content-management/organization/index.md index 2c0d2e60432..250e514600d 100644 --- a/content/en/content-management/organization/index.md +++ b/content/en/content-management/organization/index.md @@ -19,16 +19,14 @@ Hugo `0.32` announced page-relative images and other resources packaged into `Pa These terms are connected, and you also need to read about [Page Resources](/content-management/page-resources) and [Image Processing](/content-management/image-processing) to get the full picture. -{{< imgproc 1-featured Resize "300x" >}} +{{< imgproc "1-featured-content-bundles.png" "resize 300x" >}} The illustration shows three bundles. Note that the home page bundle cannot contain other content pages, although other files (images etc.) are allowed. {{< /imgproc >}} - {{% note %}} The bundle documentation is a **work in progress**. We will publish more comprehensive docs about this soon. {{% /note %}} - ## Organization of content source In Hugo, your content should be organized in a manner that reflects the rendered website. @@ -41,33 +39,31 @@ Without any additional configuration, the following will automatically work: . └── content └── about - | └── index.md // <- https://example.com/about/ + | └── index.md // <- https://example.org/about/ ├── posts - | ├── firstpost.md // <- https://example.com/posts/firstpost/ + | ├── firstpost.md // <- https://example.org/posts/firstpost/ | ├── happy - | | └── ness.md // <- https://example.com/posts/happy/ness/ - | └── secondpost.md // <- https://example.com/posts/secondpost/ + | | └── ness.md // <- https://example.org/posts/happy/ness/ + | └── secondpost.md // <- https://example.org/posts/secondpost/ └── quote - ├── first.md // <- https://example.com/quote/first/ - └── second.md // <- https://example.com/quote/second/ + ├── first.md // <- https://example.org/quote/first/ + └── second.md // <- https://example.org/quote/second/ ``` ## Path breakdown in Hugo - -The following demonstrates the relationships between your content organization and the output URL structure for your Hugo website when it renders. These examples assume you are [using pretty URLs][pretty], which is the default behavior for Hugo. The examples also assume a key-value of `baseURL = "https://example.com"` in your [site's configuration file][config]. +The following demonstrates the relationships between your content organization and the output URL structure for your Hugo website when it renders. These examples assume you are [using pretty URLs][pretty], which is the default behavior for Hugo. The examples also assume a key-value of `baseURL = "https://example.org"` in your [site's configuration file][config]. ### Index pages: `_index.md` `_index.md` has a special role in Hugo. It allows you to add front matter and content to your [list templates][lists]. These templates include those for [section templates], [taxonomy templates], [taxonomy terms templates], and your [homepage template]. {{% note %}} -**Tip:** You can get a reference to the content and metadata in `_index.md` using the [`.Site.GetPage` function](/functions/getpage/). +**Tip:** You can get a reference to the content and metadata in `_index.md` using the [`.Site.GetPage` function](/methods/page/getpage). {{% /note %}} You can create one `_index.md` for your homepage and one in each of your content sections, taxonomies, and taxonomy terms. The following shows typical placement of an `_index.md` that would contain content and front matter for a `posts` section list page on a Hugo website: - ```txt . url . ⊢--^-⊣ @@ -88,17 +84,15 @@ At build, this will output to the following destination with the associated valu ⊢--------^---------⊣⊢-^-⊣ permalink ⊢----------^-------------⊣ -https://example.com/posts/index.html +https://example.org/posts/index.html ``` The [sections] can be nested as deeply as you want. The important thing to understand is that to make the section tree fully navigational, at least the lower-most section must include a content file. (i.e. `_index.md`). - ### Single pages in sections Single content files in each of your sections will be rendered as [single page templates][singles]. Here is an example of a single `post` within `posts`: - ```txt path ("posts/my-first-hugo-post.md") . ⊢-----------^------------⊣ @@ -117,10 +111,9 @@ When Hugo builds your site, the content will be output to the following destinat ⊢--------^--------⊣⊢-^--⊣⊢-------^---------⊣ permalink ⊢--------------------^---------------------⊣ -https://example.com/posts/my-first-hugo-post/index.html +https://example.org/posts/my-first-hugo-post/index.html ``` - ## Paths explained The following concepts provide more insight into the relationship between your project's organization and the default Hugo behavior when building output for the website. @@ -147,7 +140,7 @@ The `url` is the entire URL path, defined by the file path and optionally overri [config]: /getting-started/configuration/ [formats]: /content-management/formats/ [front matter]: /content-management/front-matter/ -[getpage]: /functions/getpage/ +[getpage]: /methods/page/getpage [homepage template]: /templates/homepage/ [homepage]: /templates/homepage/ [lists]: /templates/lists/ diff --git a/content/en/content-management/page-bundles.md b/content/en/content-management/page-bundles.md index c4ce69f5f1b..91ed0d5df17 100644 --- a/content/en/content-management/page-bundles.md +++ b/content/en/content-management/page-bundles.md @@ -48,14 +48,14 @@ content/ │ │ ├── image2.png │ │ └── index.md │ └── my-other-post -│    └── index.md +│ └── index.md │ └── another-section ├── .. -    └── not-a-leaf-bundle + └── not-a-leaf-bundle ├── .. -    └── another-leaf-bundle -    └── index.md + └── another-leaf-bundle + └── index.md ``` In the above example `content/` directory, there are four leaf @@ -90,7 +90,6 @@ The hierarchy depth at which a leaf bundle is created does not matter, as long as it is not inside another **leaf** bundle. {{% /note %}} - ### Headless bundle A headless bundle is a bundle that is configured to not get published @@ -128,7 +127,7 @@ Explanation of the above example: A leaf bundle can be made headless by adding below in the front matter (in the `index.md`): -{{< code-toggle file="content/headless/index.md" fm=true copy=false >}} +{{< code-toggle file="content/headless/index.md" fm=true >}} headless = true {{< /code-toggle >}} @@ -149,17 +148,16 @@ Here `md` (markdown) is used just as an example. You can use any file type as a content resource as long as it is a content type recognized by Hugo. {{% /note %}} - ### Examples of branch bundle organization ```text content/ ├── branch-bundle-1 -│   ├── branch-content1.md -│   ├── branch-content2.md -│   ├── image1.jpg -│   ├── image2.png -│   └── _index.md +│ ├── branch-content1.md +│ ├── branch-content2.md +│ ├── image1.jpg +│ ├── image2.png +│ └── _index.md └── branch-bundle-2 ├── _index.md └── a-leaf-bundle diff --git a/content/en/content-management/page-resources.md b/content/en/content-management/page-resources.md index bbf28845996..bcff981189c 100644 --- a/content/en/content-management/page-resources.md +++ b/content/en/content-management/page-resources.md @@ -112,7 +112,6 @@ GetMatch .Resources.Match "*" 🚫 .Resources.Match "sunset.jpg" 🚫 .Resources.Match "*sunset.jpg" 🚫 - ``` ## Page resources metadata @@ -138,7 +137,7 @@ params ### Resources metadata example -{{< code-toggle copy=false >}} +{{< code-toggle >}} title: Application date : 2018-01-25 resources : @@ -184,7 +183,8 @@ The counter starts at 1 the first time they are used in either `name` or `title` For example, if a bundle has the resources `photo_specs.pdf`, `other_specs.pdf`, `guide.pdf` and `checklist.pdf`, and the front matter has specified the `resources` as: -{{< code-toggle copy=false >}} +{{< code-toggle file="content/inspections/engine/index.md" fm=true >}} +title = 'Engine inspections' [[resources]] src = "*specs.pdf" title = "Specification #:counter" diff --git a/content/en/content-management/related.md b/content/en/content-management/related.md index 410e183b734..601d1b7d5da 100644 --- a/content/en/content-management/related.md +++ b/content/en/content-management/related.md @@ -33,16 +33,19 @@ To list up to 5 related pages (which share the same _date_ or _keyword_ paramete The `Related` method takes one argument which may be a `Page` or a options map. The options map have these options: indices -: The indices to search in. +: (`slice`) The indices to search within. document -: The document to search for related content for. +: (`page`) The page for which to find related content. Required when specifying an options map. namedSlices -: The keywords to search for. +: (`slice`) The keywords to search for, expressed as a slice of `KeyValues` using the [`keyVals`] function. fragments -: Fragments holds a a list of special keywords that is used for indices configured as type "fragments". This will match the fragment identifiers of the documents. +: (`slice`) A list of special keywords that is used for indices configured as type "fragments". This will match the [fragment] identifiers of the documents. + +[fragment]: /getting-started/glossary/#fragment +[`keyVals`]: /functions/collections/keyvals/ A fictional example using all of the above options: @@ -66,7 +69,7 @@ We improved and simplified this feature in Hugo 0.111.0. Before this we had 3 di Hugo can index the headings in your content and use this to find related content. You can enable this by adding a index of type `fragments` to your `related` configuration: -{{< code-toggle file="hugo" copy=false >}} +{{< code-toggle file=hugo >}} [related] threshold = 20 includeNewer = true @@ -74,7 +77,7 @@ toLower = false [[related.indices]] name = "fragmentrefs" type = "fragments" -applyFilter = false +applyFilter = true weight = 80 {{< /code-toggle >}} @@ -146,7 +149,6 @@ applyFilter weight : An integer weight that indicates _how important_ this parameter is relative to the other parameters. It can be 0, which has the effect of turning this index off, or even negative. Test with different values to see what fits your content best. - cardinalityThreshold (default 0) : {{< new-in "0.111.0" >}}. A percentage (0-100) used to remove common keywords from the index. As an example, setting this to 50 will remove all keywords that are used in more than 50% of the documents in the index. diff --git a/content/en/content-management/sections.md b/content/en/content-management/sections.md index a3e4397f379..8ee5c0d526c 100644 --- a/content/en/content-management/sections.md +++ b/content/en/content-management/sections.md @@ -26,35 +26,35 @@ A typical site consists of one or more sections. For example: ```text content/ ├── articles/ <-- section (top-level directory) -│   ├── 2022/ -│   │   ├── article-1/ -│   │   │   ├── cover.jpg -│   │   │   └── index.md -│   │   └── article-2.md -│   └── 2023/ -│   ├── article-3.md -│   └── article-4.md +│ ├── 2022/ +│ │ ├── article-1/ +│ │ │ ├── cover.jpg +│ │ │ └── index.md +│ │ └── article-2.md +│ └── 2023/ +│ ├── article-3.md +│ └── article-4.md ├── products/ <-- section (top-level directory) -│   ├── product-1/ <-- section (has _index.md file) -│   │   ├── benefits/ <-- section (has _index.md file) -│   │   │   ├── _index.md -│   │   │   ├── benefit-1.md -│   │   │   └── benefit-2.md -│   │   ├── features/ <-- section (has _index.md file) -│   │   │   ├── _index.md -│   │   │   ├── feature-1.md -│   │   │   └── feature-2.md -│   │   └── _index.md -│   └── product-2/ <-- section (has _index.md file) -│   ├── benefits/ <-- section (has _index.md file) -│   │   ├── _index.md -│   │   ├── benefit-1.md -│   │   └── benefit-2.md -│   ├── features/ <-- section (has _index.md file) -│   │   ├── _index.md -│   │   ├── feature-1.md -│   │   └── feature-2.md -│   └── _index.md +│ ├── product-1/ <-- section (has _index.md file) +│ │ ├── benefits/ <-- section (has _index.md file) +│ │ │ ├── _index.md +│ │ │ ├── benefit-1.md +│ │ │ └── benefit-2.md +│ │ ├── features/ <-- section (has _index.md file) +│ │ │ ├── _index.md +│ │ │ ├── feature-1.md +│ │ │ └── feature-2.md +│ │ └── _index.md +│ └── product-2/ <-- section (has _index.md file) +│ ├── benefits/ <-- section (has _index.md file) +│ │ ├── _index.md +│ │ ├── benefit-1.md +│ │ └── benefit-2.md +│ ├── features/ <-- section (has _index.md file) +│ │ ├── _index.md +│ │ ├── feature-1.md +│ │ └── feature-2.md +│ └── _index.md ├── _index.md └── about.md ``` @@ -77,7 +77,7 @@ With the file structure from the [example above](#overview): 1. The articles/2022 and articles/2023 directories do not have list pages; they are not sections. -1. The list page for the products section, by default, includes product-1 and product-2, but not their descendant pages. To include descendant pages, use the `.RegularPagesRecursive` collection instead of the `.Pages` collection in the list template. See [details](/variables/page/#page-collections). +1. The list page for the products section, by default, includes product-1 and product-2, but not their descendant pages. To include descendant pages, use the `.RegularPagesRecursive` collection instead of the `.Pages` collection in the list template. See [details](/variables/page/#page-collections). 1. All directories in the products section have list pages; each directory is a section. @@ -108,7 +108,6 @@ If you need to use a different template for a subsection, specify `type` and/or A section has one or more ancestors (including the home page), and zero or more descendants. With the file structure from the [example above](#overview): - ```text content/products/product-1/benefits/benefit-1.md ``` @@ -122,11 +121,11 @@ For example, use the `.Ancestors` method to render breadcrumb navigation.
    {{ range .Ancestors.Reverse }}
  1. - {{ .LinkTitle }} + {{ .Title }}
  2. {{ end }}
  3. - {{ .LinkTitle }} + {{ .Title }}
@@ -154,7 +153,6 @@ Hugo renders this, where each breadcrumb is a link to the corresponding page: Home » Products » Product 1 » Benefits » Benefit 1 ``` - [archetype]: /content-management/archetypes/ [content type]: /content-management/types/ [directory structure]: /getting-started/directory-structure/ diff --git a/content/en/content-management/shortcodes.md b/content/en/content-management/shortcodes.md index b1e32d90206..29274486fe5 100644 --- a/content/en/content-management/shortcodes.md +++ b/content/en/content-management/shortcodes.md @@ -58,7 +58,6 @@ and a new line with a "quoted string".` */>}} Shortcodes using the `%` as the outer-most delimiter will be fully rendered when sent to the content renderer. This means that the rendered output from a shortcode can be part of the page's table of contents, footnotes, etc. - ### Shortcodes without markdown The `<` character indicates that the shortcode's inner content does *not* need further rendering. Often shortcodes without Markdown include internal HTML: @@ -172,7 +171,7 @@ To display a highlighted code sample: ```text {{}} {{ range .Pages }} -

{{ .LinkTitle }}

+

{{ .Title }}

{{ end }} {{}} ``` @@ -181,7 +180,7 @@ Rendered: {{< highlight go-html-template >}} {{ range .Pages }} -

{{ .LinkTitle }}

+

{{ .Title }}

{{ end }} {{< /highlight >}} @@ -192,7 +191,7 @@ To specify one or more [highlighting options], include a quotation-encapsulated, ```text {{}} {{ range .Pages }} -

{{ .LinkTitle }}

+

{{ .Title }}

{{ end }} {{}} ``` @@ -201,7 +200,7 @@ Rendered: {{< highlight go-html-template "lineNos=inline, lineNoStart=42" >}} {{ range .Pages }} -

{{ .LinkTitle }}

+

{{ .Title }}

{{ end }} {{< /highlight >}} @@ -220,14 +219,14 @@ You must obtain an Access Token to use the `instagram` shortcode. If your site configuration is private: -{{< code-toggle file="hugo" copy=false >}} +{{< code-toggle file=hugo >}} [services.instagram] accessToken = 'xxx' {{< /code-toggle >}} If your site configuration is _not_ private, set the Access Token with an environment variable: -```text +```sh HUGO_SERVICES_INSTAGRAM_ACCESSTOKEN=xxx hugo --gc --minify ``` @@ -251,7 +250,7 @@ Include this in your markdown: Gets a value from the current `Page's` parameters set in front matter, with a fallback to the site parameter value. It will log an `ERROR` if the parameter with the given key could not be found in either. -```bash +```sh {{}} ``` @@ -261,7 +260,7 @@ Since `testparam` is a parameter defined in front matter of this page with the v To access deeply nested parameters, use "dot syntax", e.g: -```bash +```sh {{}} ``` @@ -289,7 +288,7 @@ Read a more extensive description of `ref` and `relref` in the [cross references Assuming that standard Hugo pretty URLs are turned on. ```html -Neat +Neat Who ``` @@ -355,7 +354,6 @@ Copy the YouTube video ID that follows `v=` in the video's URL and pass it to th Furthermore, you can automatically start playback of the embedded video by setting the `autoplay` parameter to `true`. Remember that you can't mix named and unnamed parameters, so you'll need to assign the yet unnamed video ID to the parameter `id`: - {{< code file="example-youtube-input-with-autoplay.md" >}} {{}} {{< /code >}} @@ -398,7 +396,7 @@ To learn more about creating custom shortcodes, see the [shortcode template docu [partials]: /templates/partials/ [quickstart]: /getting-started/quick-start/ [sctemps]: /templates/shortcode-templates/ -[scvars]: /variables/shortcodes/ +[scvars]: /variables/shortcode/ [shortcode template documentation]: /templates/shortcode-templates/ [templatessection]: /templates/ [Vimeo]: https://vimeo.com/ diff --git a/content/en/content-management/static-files.md b/content/en/content-management/static-files.md index 78772af99fa..6eb4c0f8623 100644 --- a/content/en/content-management/static-files.md +++ b/content/en/content-management/static-files.md @@ -27,13 +27,13 @@ This union filesystem will be served from your site root. So a file Here's an example of setting `staticDir` and `staticDir2` for a multi-language site: -{{< code-toggle copy=false file="hugo" >}} +{{< code-toggle file=hugo >}} staticDir = ["static1", "static2"] [languages] [languages.en] staticDir2 = "static_en" -baseURL = "https://example.com" +baseURL = "https://example.org/" languageName = "English" weight = 2 title = "In English" diff --git a/content/en/content-management/summaries.md b/content/en/content-management/summaries.md index 74c5623cbf2..120a079956c 100644 --- a/content/en/content-management/summaries.md +++ b/content/en/content-management/summaries.md @@ -53,7 +53,7 @@ Pros Cons : Extra work for content authors, since they need to remember to type <!--more--> (or `# more` for [org content][org]) in each content file. This can be automated by adding the summary divider below the front matter of an [archetype](/content-management/archetypes/). -{{% warning "Be Precise with the Summary Divider" %}} +{{% note %}} Be careful to enter <!--more--> exactly; i.e., all lowercase and with no whitespace. {{% /note %}} @@ -75,7 +75,7 @@ Because there are multiple ways in which a summary can be specified it is useful 2. If there is a `summary` variable in the article front matter the value of the variable will be provided as per the front matter summary method 3. The text at the start of the article will be provided as per the automatic summary split method -{{% warning "Competing selections" %}} +{{% note %}} Hugo uses the _first_ of the above steps that returns text. So if, for example, your article has both `summary` variable in its front matter and a <!--more--> summary divider Hugo will use the manual summary split method. {{% /note %}} diff --git a/content/en/content-management/syntax-highlighting.md b/content/en/content-management/syntax-highlighting.md index 39fef0f9b56..e8dae3afaf0 100644 --- a/content/en/content-management/syntax-highlighting.md +++ b/content/en/content-management/syntax-highlighting.md @@ -24,7 +24,7 @@ If you run with `markup.highlight.noClasses=false` in your site configuration, y You can generate one with Hugo: -```bash +```sh hugo gen chromastyles --style=monokai > syntax.css ``` @@ -104,7 +104,6 @@ Highlighting in code fences is enabled by default. ``` ```` - Gives this: ```go {linenos=table,hl_lines=[8,"15-17"],linenostart=199} diff --git a/content/en/content-management/taxonomies.md b/content/en/content-management/taxonomies.md index af829787656..71dd18881d6 100644 --- a/content/en/content-management/taxonomies.md +++ b/content/en/content-management/taxonomies.md @@ -27,7 +27,6 @@ Term Value : a piece of content assigned to a term - ## Example taxonomy: movie website Let's assume you are making a website about movies. You may want to include the following taxonomies: @@ -86,11 +85,11 @@ Without adding a single line to your [site configuration] file, Hugo will automa If you do not want Hugo to create any taxonomies, set `disableKinds` in your [site configuration] to the following: -{{< code-toggle file="hugo" copy=false >}} +{{< code-toggle file=hugo >}} disableKinds = ["taxonomy","term"] {{}} -{{% page-kinds %}} +{{% include "content-management/_common/page-kinds.md" %}} ### Default destinations @@ -109,7 +108,7 @@ Custom taxonomies other than the [defaults](#default-taxonomies) must be defined While adding custom taxonomies, you need to put in the default taxonomies too, _if you want to keep them_. {{% /note %}} -{{< code-toggle file="hugo" copy=false >}} +{{< code-toggle file=hugo >}} [taxonomies] tag = "tags" category = "categories" @@ -120,7 +119,7 @@ While adding custom taxonomies, you need to put in the default taxonomies too, _ If you want to have just the default `tags` taxonomy, and remove the `categories` taxonomy for your site, you can do so by modifying the `taxonomies` value in your [site configuration]. -{{< code-toggle file="hugo" copy=false >}} +{{< code-toggle file=hugo >}} [taxonomies] tag = "tags" {{}} @@ -151,7 +150,7 @@ If you would like the ability to quickly generate content files with preconfigur ### Example: front matter with taxonomies -{{< code-toggle file="content/example.md" fm=true copy=false >}} +{{< code-toggle file="content/example.md" fm=true >}} title = "Hugo: A fast and flexible static site generator" tags = [ "Development", "Go", "fast", "Blogging" ] categories = [ "Development" ] @@ -168,7 +167,7 @@ The following show a piece of content that has a weight of 22, which can be used ### Example: taxonomic `weight` -{{< code-toggle copy=false >}} +{{< code-toggle >}} title = "foo" tags = [ "a", "b", "c" ] tags_weight = 22 @@ -186,7 +185,7 @@ Currently taxonomies only support the [default `weight => date` ordering of list If you need to add custom metadata to your taxonomy terms, you will need to create a page for that term at `/content///_index.md` and add your metadata in its front matter. Continuing with our 'Actors' example, let's say you want to add a Wikipedia page link to each actor. Your terms pages would be something like this: -{{< code-toggle file="content/actors/bruce-willis/_index.md" fm=true copy=false >}} +{{< code-toggle file="content/actors/bruce-willis/_index.md" fm=true >}} title: "Bruce Willis" wikipedia: "https://en.wikipedia.org/wiki/Bruce_Willis" {{< /code-toggle >}} diff --git a/content/en/content-management/toc.md b/content/en/content-management/toc.md index ca179c77dc1..1474d1c7a3d 100644 --- a/content/en/content-management/toc.md +++ b/content/en/content-management/toc.md @@ -37,7 +37,7 @@ He lay on his armour-like back, and if he lifted his head a little he could see ### My Subheading -A collection of textile samples lay spread out on the table - Samsa was a travelling salesman - and above it there hung a picture that he had recently cut out of an illustrated magazine and housed in a nice, gilded frame. It showed a lady fitted out with a fur hat and fur boa who sat upright, raising a heavy fur muff that covered the whole of her lower arm towards the viewer. Gregor then turned to look out the window at the dull weather. Drops +A collection of textile samples lay spread out on the table - Samsa was a traveling salesman - and above it there hung a picture that he had recently cut out of an illustrated magazine and housed in a nice, gilded frame. It showed a lady fitted out with a fur hat and fur boa who sat upright, raising a heavy fur muff that covered the whole of her lower arm towards the viewer. Gregor then turned to look out the window at the dull weather. Drops ``` Hugo will take this Markdown and create a table of contents from `## Introduction`, `## My Heading`, and `### My Subheading` and then store it in the [page variable][pagevars]`.TableOfContents`. @@ -105,8 +105,9 @@ He lay on his armour-like back, and if he lifted his head a little he could see === My Subheading -A collection of textile samples lay spread out on the table - Samsa was a travelling salesman - and above it there hung a picture that he had recently cut out of an illustrated magazine and housed in a nice, gilded frame. It showed a lady fitted out with a fur hat and fur boa who sat upright, raising a heavy fur muff that covered the whole of her lower arm towards the viewer. Gregor then turned to look out the window at the dull weather. Drops +A collection of textile samples lay spread out on the table - Samsa was a traveling salesman - and above it there hung a picture that he had recently cut out of an illustrated magazine and housed in a nice, gilded frame. It showed a lady fitted out with a fur hat and fur boa who sat upright, raising a heavy fur muff that covered the whole of her lower arm towards the viewer. Gregor then turned to look out the window at the dull weather. Drops ``` + Hugo will take this AsciiDoc and create a table of contents store it in the page variable `.TableOfContents`, in the same as described for Markdown. [conditionals]: /templates/introduction/#conditionals diff --git a/content/en/content-management/urls.md b/content/en/content-management/urls.md index 60f9615b864..f1c3af2a5c1 100644 --- a/content/en/content-management/urls.md +++ b/content/en/content-management/urls.md @@ -28,7 +28,7 @@ You can change the structure and appearance of URLs with front matter values and Set the `slug` in front matter to override the last segment of the path. The `slug` value does not affect section pages. -{{< code-toggle file="content/posts/post-1.md" copy=false fm=true >}} +{{< code-toggle file="content/posts/post-1.md" fm=true >}} title = 'My First Post' slug = 'my-first-post' {{< /code-toggle >}} @@ -45,7 +45,7 @@ Set the `url` in front matter to override the entire path. Use this with either With this front matter: -{{< code-toggle file="content/posts/post-1.md" copy=false fm=true >}} +{{< code-toggle file="content/posts/post-1.md" fm=true >}} title = 'My First Article' url = '/articles/my-first-article' {{< /code-toggle >}} @@ -58,7 +58,7 @@ https://example.org/articles/my-first-article/ If you include a file extension: -{{< code-toggle file="content/posts/post-1.md" copy=false fm=true >}} +{{< code-toggle file="content/posts/post-1.md" fm=true >}} title = 'My First Article' url = '/articles/my-first-article.html' {{< /code-toggle >}} @@ -112,7 +112,7 @@ content/ Render tutorials under "training", and render the posts under "articles" with a date-base hierarchy: -{{< code-toggle file="hugo" copy=false >}} +{{< code-toggle file=hugo >}} [permalinks.page] posts = '/articles/:year/:month/:slug/' tutorials = '/training/:slug/' @@ -145,14 +145,14 @@ public/ To create a date-based hierarchy for regular pages in the content root: -{{< code-toggle file="hugo" copy=false >}} +{{< code-toggle file=hugo >}} [permalinks.page] "/" = "/:year/:month/:slug/" {{< /code-toggle >}} Use the same approach with taxonomy terms. For example, to omit the taxonomy segment of the URL: -{{< code-toggle file="hugo" copy=false >}} +{{< code-toggle file=hugo >}} [permalinks.term] 'tags' = '/:slug/' {{< /code-toggle >}} @@ -179,7 +179,7 @@ content/ And this site configuration: -{{< code-toggle file="hugo" copy=false >}} +{{< code-toggle file=hugo >}} defaultContentLanguage = 'en' defaultContentLanguageInSubdir = true @@ -280,7 +280,7 @@ For time-related values, you can also use the layout string components defined i [time package]: https://pkg.go.dev/time#pkg-constants -{{< code-toggle file="hugo" copy=false >}} +{{< code-toggle file=hugo >}} permalinks: posts: /:06/:1/:2/:title/ {{< /code-toggle >}} @@ -296,7 +296,7 @@ pretty|content/about.md|`https://example.org/about/` By default, Hugo produces pretty URLs. To generate ugly URLs, change your site configuration: -{{< code-toggle file="hugo" copy=false >}} +{{< code-toggle file=hugo >}} uglyURLs = true {{< /code-toggle >}} @@ -314,7 +314,7 @@ This is a legacy configuration option, superseded by template functions and mark If enabled, Hugo performs a search and replace _after_ it renders the page. It searches for site-relative URLs (those with a leading slash) associated with `action`, `href`, `src`, `srcset`, and `url` attributes. It then prepends the `baseURL` to create absolute URLs. -```text +```html ``` @@ -323,7 +323,7 @@ This is an imperfect, brute force approach that can affect content as well as HT To enable: -{{< code-toggle file="hugo" copy=false >}} +{{< code-toggle file=hugo >}} canonifyURLs = true {{< /code-toggle >}} @@ -337,7 +337,7 @@ If enabled, Hugo performs a search and replace _after_ it renders the page. It s For example, when rendering `content/posts/post-1`: -```text +```html ``` @@ -346,7 +346,7 @@ This is an imperfect, brute force approach that can affect content as well as HT To enable: -{{< code-toggle file="hugo" copy=false >}} +{{< code-toggle file=hugo >}} relativeURLs = true {{< /code-toggle >}} @@ -361,7 +361,7 @@ Create redirects from old URLs to new URLs with aliases: Change the file name of an existing page, and create an alias from the previous URL to the new URL: -{{< code-toggle file="content/posts/new-file-name.md" copy=false >}} +{{< code-toggle file="content/posts/new-file-name.md" >}} aliases = ['/posts/previous-file-name'] {{< /code-toggle >}} @@ -373,13 +373,13 @@ Each of these directory-relative aliases is equivalent to the site-relative alia You can create more than one alias to the current page: -{{< code-toggle file="content/posts/new-file-name.md" copy=false >}} +{{< code-toggle file="content/posts/new-file-name.md" >}} aliases = ['previous-file-name','original-file-name'] {{< /code-toggle >}} In a multilingual site, use a directory-relative alias, or include the language prefix with a site-relative alias: -{{< code-toggle file="content/posts/new-file-name.de.md" copy=false >}} +{{< code-toggle file="content/posts/new-file-name.de.md" >}} aliases = ['/de/posts/previous-file-name'] {{< /code-toggle >}} @@ -400,7 +400,7 @@ public/ The alias from the previous URL to the new URL is a client-side redirect: -{{< code file="posts/previous-file-name/index.html" copy=false >}} +{{< code file="posts/previous-file-name/index.html" >}} diff --git a/content/en/contribute/development.md b/content/en/contribute/development.md index 7bc201288a8..2136b16923e 100644 --- a/content/en/contribute/development.md +++ b/content/en/contribute/development.md @@ -313,7 +313,7 @@ git commit --amend #### Modify multiple commits -{{% warning "Be Careful Modifying Multiple Commits"%}} +{{% note %}} Modifications such as those described in this section can have serious unintended consequences. Skip this section if you're not sure! {{% /note %}} diff --git a/content/en/contribute/documentation.md b/content/en/contribute/documentation.md index 03939f837dc..3c4d99cd491 100644 --- a/content/en/contribute/documentation.md +++ b/content/en/contribute/documentation.md @@ -24,7 +24,7 @@ Step 2 Step 3 : Create a new branch with a descriptive name. -```bash +```sh git checkout -b fix/typos-site-variables ``` @@ -34,7 +34,7 @@ Step 4 Step 5 : Commit your changes with a descriptive commit message, typically 50 characters or less. Included the "Closes" keyword if your change addresses one or more open [issues]. -```bash +```sh git commit -m "Fix typos on site variables page Closes #1234 @@ -128,7 +128,7 @@ fm #### Site configuration example ```text -{{}} +{{}} baseURL = 'https://example.org' languageCode = 'en-US' title = "Example Site" @@ -137,7 +137,7 @@ title = "Example Site" Rendered: -{{< code-toggle file="hugo" >}} +{{< code-toggle file=hugo >}} baseURL = 'https://example.org' languageCode = 'en-US' title = "Example Site" diff --git a/content/en/documentation.md b/content/en/documentation.md index cf864e81928..da7b3ef9b4c 100644 --- a/content/en/documentation.md +++ b/content/en/documentation.md @@ -9,6 +9,14 @@ weight: 1 layout: documentation-home --- -Hugo is the **world's fastest static website engine.** It's written in Go (aka Golang) and developed by [bep](https://github.com/bep), [spf13](https://github.com/spf13) and [friends](https://github.com/gohugoio/hugo/graphs/contributors). +A fast and flexible [static site generator] built with love by [bep], [spf13], and [friends] in [Go]. + +Hugo is optimized for speed and designed for flexibility. With its advanced templating system and fast asset pipelines, Hugo renders a complete site in seconds, often less. + +[bep]: https://github.com/bep +[spf13]: https://github.com/spf13 +[friends]: https://github.com/gohugoio/hugo/graphs/contributors +[go]: https://go.dev/ +[static site generator]: https://en.wikipedia.org/wiki/Static_site_generator Below you will find some of the most common and helpful pages from our documentation. diff --git a/content/en/functions/Get.md b/content/en/functions/Get.md deleted file mode 100644 index 142e9811b2f..00000000000 --- a/content/en/functions/Get.md +++ /dev/null @@ -1,26 +0,0 @@ ---- -title: .Get -description: Accesses positional and ordered parameters in shortcode declaration. -categories: [functions] -keywords: [] -menu: - docs: - parent: functions -function: - aliases: [] - returnType: any - signatures: - - .Get INDEX - - .Get KEY -relatedFunctions: [] ---- - -`.Get` is specifically used when creating your own [shortcode template][sc], to access the [positional and named](/templates/shortcode-templates/#positional-vs-named-parameters) parameters passed to it. When used with a numeric INDEX, it queries positional parameters (starting with 0). With a string KEY, it queries named parameters. - -When accessing named or positional parameters that do not exist, `.Get` returns an empty string instead of interrupting the build. This allows you to chain `.Get` with `if`, `with`, `default` or `cond` to check for parameter existence. For example: - -```go-html-template -{{ $quality := default "100" (.Get 1) }} -``` - -[sc]: /templates/shortcode-templates/ diff --git a/content/en/functions/GetPage.md b/content/en/functions/GetPage.md deleted file mode 100644 index 4afebaca7be..00000000000 --- a/content/en/functions/GetPage.md +++ /dev/null @@ -1,84 +0,0 @@ ---- -title: .GetPage -description: Gets a `Page` of a given `path`. -categories: [functions] -keywords: [] -menu: - docs: - parent: functions -function: - aliases: [] - returnType: - signatures: [.GetPage PATH] -relatedFunctions: [] ---- - -`.GetPage` returns a page of a given `path`. Both `Site` and `Page` implements this method. The `Page` variant will, if given a relative path -- i.e. a path without a leading `/` -- try look for the page relative to the current page. - -{{% note %}} -**Note:** We overhauled and simplified the `.GetPage` API in Hugo 0.45. Before that you needed to provide a `Kind` attribute in addition to the path, e.g. `{{ .Site.GetPage "section" "blog" }}`. This will still work, but is now superfluous. -{{% /note %}} - - -```go-html-template -{{ with .Site.GetPage "/blog" }}{{ .Title }}{{ end }} -``` - -This method will return `nil` when no page could be found, so the above will not print anything if the blog section is not found. - -To find a regular page in the blog section:: - -```go-html-template -{{ with .Site.GetPage "/blog/my-post.md" }}{{ .Title }}{{ end }} -``` - -And since `Page` also provides a `.GetPage` method, the above is the same as: - -```go-html-template -{{ with .Site.GetPage "/blog" }} -{{ with .GetPage "my-post.md" }}{{ .Title }}{{ end }} -{{ end }} -``` - -## .GetPage and multilingual sites - -The previous examples have used the full content file name to look up the post. Depending on how you have organized your content (whether you have the language code in the file name or not, e.g. `my-post.en.md`), you may want to do the lookup without extension. This will get you the current language's version of the page: - -```go-html-template -{{ with .Site.GetPage "/blog/my-post" }}{{ .Title }}{{ end }} -``` - -## .GetPage example - -This code snippet---in the form of a [partial template][partials]---allows you to do the following: - -1. Grab the index object of your `tags` [taxonomy]. -2. Assign this object to a variable, `$t` -3. Sort the terms associated with the taxonomy by popularity. -4. Grab the top two most popular terms in the taxonomy (i.e., the two most popular tags assigned to content. - -{{< code file="grab-top-two-tags.html" >}} - -{{< /code >}} - -## `.GetPage` on page bundles - -If the page retrieved by `.GetPage` is a [Leaf Bundle][leaf_bundle], and you -need to get the nested _**page** resources_ in that, you will need to use the -methods in `.Resources` as explained in the [Page Resources][page_resources] -section. - -See the [Headless Bundle][headless_bundle] documentation for an example. - - -[partials]: /templates/partials/ -[taxonomy]: /content-management/taxonomies/ -[page_kinds]: /templates/section-templates/#page-kinds -[leaf_bundle]: /content-management/page-bundles/#leaf-bundles -[headless_bundle]: /content-management/page-bundles/#headless-bundle -[page_resources]: /content-management/page-resources/ diff --git a/content/en/functions/Param.md b/content/en/functions/Param.md deleted file mode 100644 index a2826a3c8a4..00000000000 --- a/content/en/functions/Param.md +++ /dev/null @@ -1,50 +0,0 @@ ---- -title: .Param -description: Returns a page parameter, falling back to a site parameter if present. -categories: [functions] -keywords: [] -menu: - docs: - parent: functions -function: - aliases: [] - returnType: any - signatures: [.Param KEY] -relatedFunctions: [] ---- - -The `.Param` method on `.Page` looks for the given `KEY` in page parameters, and returns the corresponding value. If it cannot find the `KEY` in page parameters, it looks for the `KEY` in site parameters. If it cannot find the `KEY` in either location, the `.Param` method returns `nil`. - -Site and theme developers commonly set parameters at the site level, allowing content authors to override those parameters at the page level. - -For example, to show a table of contents on every page, but allow authors to hide the table of contents as needed: - -**Configuration** - -{{< code-toggle file="hugo" copy=false >}} -[params] -display_toc = true -{{< /code-toggle >}} - -**Content** - -{{< code-toggle file="content/example.md" fm=true copy=false >}} -title = 'Example' -date = 2023-01-01 -draft = false -display_toc = false -{{< /code-toggle >}} - -**Template** - -{{< code file="layouts/_default/single.html" copy=false >}} -{{ if .Param "display_toc" }} - {{ .TableOfContents }} -{{ end }} -{{< /code >}} - -The `.Param` method returns the value associated with the given `KEY`, regardless of whether the value is truthy or falsy. If you need to ignore falsy values, use this construct instead: - -{{< code file="layouts/_default/single.html" copy=false >}} -{{ or .Params.foo site.Params.foo }} -{{< /code >}} diff --git a/content/en/functions/Render.md b/content/en/functions/Render.md deleted file mode 100644 index 02567845f9c..00000000000 --- a/content/en/functions/Render.md +++ /dev/null @@ -1,28 +0,0 @@ ---- -title: .Render -description: Takes a view to apply when rendering content. -categories: [functions] -keywords: [] -menu: - docs: - parent: functions -function: - aliases: [] - returnType: template.HTML - signatures: [.Render LAYOUT] -relatedFunctions: [] ---- - -The view is an alternative layout and should be a file name that points to a template in one of the locations specified in the documentation for [Content Views](/templates/views). - -This function is only available when applied to a single piece of content within a [list context]. - -This example could render a piece of content using the content view located at `/layouts/_default/summary.html`: - -```go-html-template -{{ range .Pages }} - {{ .Render "summary" }} -{{ end }} -``` - -[list context]: /templates/lists/ diff --git a/content/en/functions/RenderString.md b/content/en/functions/RenderString.md deleted file mode 100644 index 91414d6a0a7..00000000000 --- a/content/en/functions/RenderString.md +++ /dev/null @@ -1,35 +0,0 @@ ---- -title: .RenderString -description: Renders markup to HTML. -categories: [functions] -keywords: [] -menu: - docs: - parent: functions -function: - aliases: [] - returnType: template.HTML - signatures: ['.RenderString MARKUP [OPTIONS]'] ---- - -`.RenderString` is a method on `Page` that renders some markup to HTML using the content renderer defined for that page (if not set in the options). - -The method takes an optional map argument with these options: - -display ("inline") -: `inline` or `block`. If `inline` (default), surrounding `

` on short snippets will be trimmed. - -markup (defaults to the Page's markup) -: See identifiers in [List of content formats](/content-management/formats/#list-of-content-formats). - -Some examples: - -```go-html-template -{{ $optBlock := dict "display" "block" }} -{{ $optOrg := dict "markup" "org" }} -{{ "**Bold Markdown**" | $p.RenderString }} -{{ "**Bold Block Markdown**" | $p.RenderString $optBlock }} -{{ "/italic org mode/" | $p.RenderString $optOrg }} -``` - -{{< new-in "0.93.0" >}} **Note**: [markdownify](/functions/transform/markdownify) uses this function in order to support [Render Hooks](/getting-started/configuration-markup/#markdown-render-hooks). diff --git a/content/en/functions/Scratch.md b/content/en/functions/Scratch.md deleted file mode 100644 index 0640d6bcc57..00000000000 --- a/content/en/functions/Scratch.md +++ /dev/null @@ -1,151 +0,0 @@ ---- -title: .Scratch -description: Acts as a "scratchpad" to store and manipulate data. -categories: [functions] -keywords: [] -menu: - docs: - parent: functions -function: - aliases: [] - returnType: - signatures: [] -relatedFunctions: - - .Store - - .Scratch -aliases: [/extras/scratch/,/doc/scratch/] ---- - -Scratch is a Hugo feature designed to conveniently manipulate data in a Go Template world. It is either a Page or Shortcode method for which the resulting data will be attached to the given context, or it can live as a unique instance stored in a variable. - -{{% note %}} -Note that Scratch was initially created as a workaround for a [Go template scoping limitation](https://github.com/golang/go/issues/10608) that affected Hugo versions prior to 0.48. For a detailed analysis of `.Scratch` and contextual use cases, see [this blog post](https://regisphilibert.com/blog/2017/04/hugo-scratch-explained-variable/). -{{% /note %}} - -### Contexted `.Scratch` vs. local `newScratch` - -Since Hugo 0.43, there are two different ways of using Scratch: - -#### The Page's `.Scratch` - -`.Scratch` is available as a Page method or a Shortcode method and attaches the "scratched" data to the given page. Either a Page or a Shortcode context is required to use `.Scratch`. - -```go-html-template -{{ .Scratch.Set "greeting" "bonjour" }} -{{ range .Pages }} - {{ .Scratch.Set "greeting" (print "bonjour" .Title) }} -{{ end }} -``` - -#### The local `newScratch` - -A Scratch instance can also be assigned to any variable using the `newScratch` function. In this case, no Page or Shortcode context is required and the scope of the scratch is only local. The methods detailed below are available from the variable the Scratch instance was assigned to. - -```go-html-template -{{ $data := newScratch }} -{{ $data.Set "greeting" "hola" }} -``` - -### Methods - -A Scratch has the following methods: - -{{% note %}} -Note that the following examples assume a [local Scratch instance](#the-local-newscratch) has been stored in `$scratch`. -{{% /note %}} - -#### .Set - -Set the value of a given key. - -```go-html-template -{{ $scratch.Set "greeting" "Hello" }} -``` - -#### .Get - -Get the value of a given key. - -```go-html-template -{{ $scratch.Set "greeting" "Hello" }} ----- -{{ $scratch.Get "greeting" }} > Hello -``` - -#### .Add - -Add a given value to existing value(s) of the given key. - -For single values, `Add` accepts values that support Go's `+` operator. If the first `Add` for a key is an array or slice, the following adds will be [appended](/functions/collections/append/) to that list. - -```go-html-template -{{ $scratch.Add "greetings" "Hello" }} -{{ $scratch.Add "greetings" "Welcome" }} ----- -{{ $scratch.Get "greetings" }} > HelloWelcome -``` - -```go-html-template -{{ $scratch.Add "total" 3 }} -{{ $scratch.Add "total" 7 }} ----- -{{ $scratch.Get "total" }} > 10 -``` - -```go-html-template -{{ $scratch.Add "greetings" (slice "Hello") }} -{{ $scratch.Add "greetings" (slice "Welcome" "Cheers") }} ----- -{{ $scratch.Get "greetings" }} > []interface {}{"Hello", "Welcome", "Cheers"} -``` - -#### .SetInMap - -Takes a `key`, `mapKey` and `value` and adds a map of `mapKey` and `value` to the given `key`. - -```go-html-template -{{ $scratch.SetInMap "greetings" "english" "Hello" }} -{{ $scratch.SetInMap "greetings" "french" "Bonjour" }} ----- -{{ $scratch.Get "greetings" }} > map[french:Bonjour english:Hello] -``` - -#### .DeleteInMap -Takes a `key` and `mapKey` and removes the map of `mapKey` from the given `key`. - -```go-html-template -{{ .Scratch.SetInMap "greetings" "english" "Hello" }} -{{ .Scratch.SetInMap "greetings" "french" "Bonjour" }} ----- -{{ .Scratch.DeleteInMap "greetings" "english" }} ----- -{{ .Scratch.Get "greetings" }} > map[french:Bonjour] -``` - -#### .GetSortedMapValues - -Return an array of values from `key` sorted by `mapKey`. - -```go-html-template -{{ $scratch.SetInMap "greetings" "english" "Hello" }} -{{ $scratch.SetInMap "greetings" "french" "Bonjour" }} ----- -{{ $scratch.GetSortedMapValues "greetings" }} > [Hello Bonjour] -``` - -#### .Delete - -Remove the given key. - -```go-html-template -{{ $scratch.Set "greeting" "Hello" }} ----- -{{ $scratch.Delete "greeting" }} -``` - -#### .Values - -Return the raw backing map. Note that you should only use this method on the locally scoped Scratch instances you obtain via [`newScratch`](#the-local-newscratch), not `.Page.Scratch` etc., as that will lead to concurrency issues. - - -[pagevars]: /variables/page/ diff --git a/content/en/functions/Store.md b/content/en/functions/Store.md deleted file mode 100644 index 49b38188b97..00000000000 --- a/content/en/functions/Store.md +++ /dev/null @@ -1,111 +0,0 @@ ---- -title: .Store -description: Returns a Scratch that is not reset on server rebuilds. -categories: [functions] -keywords: [] -menu: - docs: - parent: functions -function: - aliases: [] - returnType: - signatures: [] -relatedFunctions: - - .Store - - .Scratch ---- - -The `.Store` method on `.Page` returns a [Scratch] to store and manipulate data. In contrast to the `.Scratch` method, this Scratch is not reset on server rebuilds. - -[Scratch]: /functions/scratch/ - -### Methods - -#### .Set - -Sets the value of a given key. - -```go-html-template -{{ .Store.Set "greeting" "Hello" }} -``` - -#### .Get - -Gets the value of a given key. - -```go-html-template -{{ .Store.Set "greeting" "Hello" }} - -{{ .Store.Get "greeting" }} → Hello -``` - -#### .Add - -Adds a given value to existing value(s) of the given key. - -For single values, `Add` accepts values that support Go's `+` operator. If the first `Add` for a key is an array or slice, the following adds will be appended to that list. - -```go-html-template -{{ .Store.Add "greetings" "Hello" }} -{{ .Store.Add "greetings" "Welcome" }} - -{{ .Store.Get "greetings" }} → HelloWelcome -``` - -```go-html-template -{{ .Store.Add "total" 3 }} -{{ .Store.Add "total" 7 }} - -{{ .Store.Get "total" }} → 10 -``` - -```go-html-template -{{ .Store.Add "greetings" (slice "Hello") }} -{{ .Store.Add "greetings" (slice "Welcome" "Cheers") }} - -{{ .Store.Get "greetings" }} → []interface {}{"Hello", "Welcome", "Cheers"} -``` - -#### .SetInMap - -Takes a `key`, `mapKey` and `value` and adds a map of `mapKey` and `value` to the given `key`. - -```go-html-template -{{ .Store.SetInMap "greetings" "english" "Hello" }} -{{ .Store.SetInMap "greetings" "french" "Bonjour" }} - -{{ .Store.Get "greetings" }} → map[french:Bonjour english:Hello] -``` - -#### .DeleteInMap - -Takes a `key` and `mapKey` and removes the map of `mapKey` from the given `key`. - -```go-html-template -{{ .Store.SetInMap "greetings" "english" "Hello" }} -{{ .Store.SetInMap "greetings" "french" "Bonjour" }} -{{ .Store.DeleteInMap "greetings" "english" }} - -{{ .Store.Get "greetings" }} → map[french:Bonjour] -``` - -#### .GetSortedMapValues - -Returns an array of values from `key` sorted by `mapKey`. - -```go-html-template -{{ .Store.SetInMap "greetings" "english" "Hello" }} -{{ .Store.SetInMap "greetings" "french" "Bonjour" }} - -{{ .Store.GetSortedMapValues "greetings" }} → [Hello Bonjour] -``` - -#### .Delete - -Removes the given key. - -```go-html-template -{{ .Store.Set "greeting" "Hello" }} - -{{ .Store.Delete "greeting" }} -``` diff --git a/content/en/functions/Unix.md b/content/en/functions/Unix.md deleted file mode 100644 index ed49b72b5b7..00000000000 --- a/content/en/functions/Unix.md +++ /dev/null @@ -1,36 +0,0 @@ ---- -title: .Unix -description: Converts a time.Time value to the number of seconds elapsed since the Unix epoch, excluding leap seconds. The Unix epoch is 00:00:00 UTC on 1 January 1970. -categories: [functions] -menu: - docs: - parent: functions -function: - aliases: [] - returnType: int64 - signatures: - - .Unix - - .UnixMilli - - .UnixMicro - - .UnixNano -relatedFunctions: [] ---- - -The `Milli`, `Micro`, and `Nano` variants return the number of milliseconds, microseconds, and nanoseconds (respectively) elapsed since the Unix epoch. - -```go-html-template -.Date.Unix → 1637259694 -.ExpiryDate.Unix → 1672559999 -.Lastmod.Unix → 1637361786 -.PublishDate.Unix → 1637421261 - -("1970-01-01T00:00:00-00:00" | time.AsTime).Unix → 0 -("1970-01-01T00:00:42-00:00" | time.AsTime).Unix → 42 -("1970-04-11T01:48:29-08:00" | time.AsTime).Unix → 8675309 -("2026-05-02T20:09:31-07:00" | time.AsTime).Unix → 1777777771 - -now.Unix → 1637447841 -now.UnixMilli → 1637447841347 -now.UnixMicro → 1637447841347378 -now.UnixNano → 1637447841347378799 -``` diff --git a/content/en/functions/_common/_index.md b/content/en/functions/_common/_index.md new file mode 100644 index 00000000000..47d5812fba5 --- /dev/null +++ b/content/en/functions/_common/_index.md @@ -0,0 +1,13 @@ +--- +cascade: + _build: + list: never + publishResources: false + render: never +--- + + diff --git a/content/en/functions/_common/glob-patterns.md b/content/en/functions/_common/glob-patterns.md new file mode 100644 index 00000000000..3b0813f6fe9 --- /dev/null +++ b/content/en/functions/_common/glob-patterns.md @@ -0,0 +1,23 @@ +--- +# Do not remove front matter. +--- + +Path|Pattern|Match +:--|:--|:-- +`images/foo/a.jpg`|`images/foo/*.jpg`|`true` +`images/foo/a.jpg`|`images/foo/*.*`|`true` +`images/foo/a.jpg`|`images/foo/*`|`true` +`images/foo/a.jpg`|`images/*/*.jpg`|`true` +`images/foo/a.jpg`|`images/*/*.*`|`true` +`images/foo/a.jpg`|`images/*/*`|`true` +`images/foo/a.jpg`|`*/*/*.jpg`|`true` +`images/foo/a.jpg`|`*/*/*.*`|`true` +`images/foo/a.jpg`|`*/*/*`|`true` +`images/foo/a.jpg`|`**/*.jpg`|`true` +`images/foo/a.jpg`|`**/*.*`|`true` +`images/foo/a.jpg`|`**/*`|`true` +`images/foo/a.jpg`|`**`|`true` +`images/foo/a.jpg`|`*/*.jpg`|`false` +`images/foo/a.jpg`|`*.jpg`|`false` +`images/foo/a.jpg`|`*.*`|`false` +`images/foo/a.jpg`|`*`|`false` diff --git a/content/en/functions/_common/go-template-functions.md b/content/en/functions/_common/go-template-functions.md deleted file mode 100644 index b8722743e07..00000000000 --- a/content/en/functions/_common/go-template-functions.md +++ /dev/null @@ -1,3 +0,0 @@ -See Go's [text/template] documentation for more details. - -[text/template]: https://pkg.go.dev/text/template diff --git a/content/en/functions/_common/index.md b/content/en/functions/_common/index.md deleted file mode 100644 index cbb7365a66e..00000000000 --- a/content/en/functions/_common/index.md +++ /dev/null @@ -1,3 +0,0 @@ -+++ -headless = true -+++ diff --git a/content/en/functions/_common/locales.md b/content/en/functions/_common/locales.md index 259231cbff2..fd84157819b 100644 --- a/content/en/functions/_common/locales.md +++ b/content/en/functions/_common/locales.md @@ -1,3 +1,10 @@ +--- +# Do not remove front matter. +--- + +{{% note %}} + Localization of dates, currencies, numbers, and percentages is performed by the [gohugoio/locales] package. The language tag of the current site must match one of the listed locales. [gohugoio/locales]: https://github.com/gohugoio/locales +{{% /note %}} diff --git a/content/en/functions/_common/regular-expressions.md b/content/en/functions/_common/regular-expressions.md index 9da340849f6..48e020ac65a 100644 --- a/content/en/functions/_common/regular-expressions.md +++ b/content/en/functions/_common/regular-expressions.md @@ -1,3 +1,7 @@ +--- +# Do not remove front matter. +--- + When specifying the regular expression, use a raw [string literal] (backticks) instead of an interpreted string literal (double quotes) to simplify the syntax. With an interpreted string literal you must escape backslashes. Go's regular expression package implements the [RE2 syntax]. The RE2 syntax is a subset of that accepted by [PCRE], roughly speaking, and with various [caveats]. Note that the RE2 `\C` escape sequence is not supported. diff --git a/content/en/functions/_common/time-layout-string.md b/content/en/functions/_common/time-layout-string.md index 40932f8cfc2..827dc989465 100644 --- a/content/en/functions/_common/time-layout-string.md +++ b/content/en/functions/_common/time-layout-string.md @@ -1,12 +1,16 @@ +--- +# Do not remove front matter. +--- + Format a `time.Time` value based on [Go's reference time]: [Go's reference time]: https://pkg.go.dev/time#pkg-constants -```text {copy=false} +```text Mon Jan 2 15:04:05 MST 2006 ``` -Create a format string using these components: +Create a layout string using these components: Description|Valid components :--|:-- @@ -21,7 +25,7 @@ Second|`"5" "05"` AM/PM mark|`"PM"` Time zone offsets|`"-0700" "-07:00" "-07" "-070000" "-07:00:00"` -Replace the sign in the format string with a Z to print Z instead of an offset for the UTC zone. +Replace the sign in the layout string with a Z to print Z instead of an offset for the UTC zone. Description|Valid components :--|:-- diff --git a/content/en/functions/_index.md b/content/en/functions/_index.md index 4f8aa47ca00..b4b58eada08 100644 --- a/content/en/functions/_index.md +++ b/content/en/functions/_index.md @@ -1,7 +1,8 @@ --- title: Functions linkTitle: Overview -description: Comprehensive list of Hugo templating functions, including basic and advanced usage examples. +description: A list of Hugo template functions including examples. +categories: [] keywords: [] menu: docs: @@ -9,9 +10,8 @@ menu: parent: functions weight: 10 weight: 10 +showSectionMenu: true aliases: [/layout/functions/,/templates/functions] --- -Go templates are lightweight but extensible. Go itself supplies built-in functions, including comparison operators and other basic tools. These are listed in the [Go template documentation][gofuncs]. Hugo has added additional functions to the basic template logic. - -[gofuncs]: https://golang.org/pkg/text/template/#hdr-Functions +Use these functions within your templates and archetypes. diff --git a/content/en/functions/cast/ToFloat.md b/content/en/functions/cast/ToFloat.md index acf70bc43aa..51bc908b6f7 100644 --- a/content/en/functions/cast/ToFloat.md +++ b/content/en/functions/cast/ToFloat.md @@ -1,20 +1,15 @@ --- title: cast.ToFloat -linkTitle: float -description: Casts a value to a decimal (base 10) floating point value. -categories: [functions] +description: Converts a value to a decimal floating-point number (base 10). +categories: [] keywords: [] -menu: - docs: - parent: functions -function: +action: aliases: [float] + related: + - functions/cast/ToInt + - functions/cast/ToString returnType: float64 signatures: [cast.ToFloat INPUT] -relatedFunctions: - - cast.ToFloat - - cast.ToInt - - cast.ToString aliases: [/functions/float] --- diff --git a/content/en/functions/cast/ToInt.md b/content/en/functions/cast/ToInt.md index b4a37cb6ce7..283da1e4007 100644 --- a/content/en/functions/cast/ToInt.md +++ b/content/en/functions/cast/ToInt.md @@ -1,20 +1,14 @@ --- title: cast.ToInt -linkTitle: int -description: Casts a value to a decimal (base 10) integer. -categories: [functions] +description: Converts a value to a decimal integer (base 10). keywords: [] -menu: - docs: - parent: functions -function: +action: aliases: [int] + related: + - functions/cast/ToFloat + - functions/cast/ToString returnType: int - signatures: [cast.ToInt INPUT] -relatedFunctions: - - cast.ToFloat - - cast.ToInt - - cast.ToString + signatures: [cast/ToInt INPUT] aliases: [/functions/int] --- @@ -24,8 +18,8 @@ With a decimal (base 10) input: {{ int 11 }} → 11 (int) {{ int "11" }} → 11 (int) -{{ int 11.1 }} → 11 (int) -{{ int 11.9 }} → 11 (int) +{{ int 11/1 }} → 11 (int) +{{ int 11/9 }} → 11 (int) ``` With a binary (base 2) input: @@ -55,5 +49,5 @@ With a hexadecimal (base 16) input: {{% note %}} Values with a leading zero are octal (base 8). When casting a string representation of a decimal (base 10) number, remove leading zeros: -`{{ strings.TrimLeft "0" "0011" | int }} → 11` +`{{ strings/TrimLeft "0" "0011" | int }} → 11` {{% /note %}} diff --git a/content/en/functions/cast/ToString.md b/content/en/functions/cast/ToString.md index d677ecdbfd6..a701c942170 100644 --- a/content/en/functions/cast/ToString.md +++ b/content/en/functions/cast/ToString.md @@ -1,20 +1,15 @@ --- title: cast.ToString -linkTitle: string -description: Cast a value to a string. -categories: [functions] +description: Converts a value to a string. +categories: [] keywords: [] -menu: - docs: - parent: functions -function: +action: aliases: [string] + related: + - functions/cast/ToFloat + - functions/cast/ToInt returnType: string signatures: [cast.ToString INPUT] -relatedFunctions: - - cast.ToFloat - - cast.ToInt - - cast.ToString aliases: [/functions/string] --- diff --git a/content/en/functions/cast/_index.md b/content/en/functions/cast/_index.md new file mode 100644 index 00000000000..82389237a26 --- /dev/null +++ b/content/en/functions/cast/_index.md @@ -0,0 +1,12 @@ +--- +title: Cast functions +linkTitle: cast +description: Template functions to cast a value from one data type to another. +categories: [] +keywords: [] +menu: + docs: + parent: functions +--- + +Use these functions to cast a value from one data type to another. diff --git a/content/en/functions/collections/After.md b/content/en/functions/collections/After.md index e27c1507fdb..743e5a48115 100644 --- a/content/en/functions/collections/After.md +++ b/content/en/functions/collections/After.md @@ -1,20 +1,15 @@ --- title: collections.After -linkTitle: after description: Slices an array to the items after the Nth item. -categories: [functions] +categories: [] keywords: [] -menu: - docs: - parent: functions -function: +action: aliases: [after] + related: + - functions/collections/First + - functions/collections/Last returnType: any signatures: [collections.After INDEX COLLECTION] -relatedFunctions: - - collections.After - - collections.First - - collections.Last aliases: [/functions/after] --- @@ -37,26 +32,26 @@ You can use `after` in combination with the [`first`] function and Hugo's [power {{< code file="layouts/section/articles.html" >}} {{ define "main" }} -
-

Featured Article

- {{ range first 1 .Pages.ByPublishDate.Reverse }} -
+
+

Featured Article

+ {{ range first 1 .Pages.ByPublishDate.Reverse }} +

{{ .Title }}

-
-

{{ .Description }}

-{{ end }} -
-
-

Recent Articles

- {{ range first 3 (after 1 .Pages.ByPublishDate.Reverse) }} -
-
-

{{ .Title }}

-
-

{{ .Description }}

-
+
+

{{ .Description }}

{{ end }} - +
+
+

Recent Articles

+ {{ range first 3 (after 1 .Pages.ByPublishDate.Reverse) }} +
+
+

{{ .Title }}

+
+

{{ .Description }}

+
+ {{ end }} +
{{ end }} {{< /code >}} diff --git a/content/en/functions/collections/Append.md b/content/en/functions/collections/Append.md index 31657288fee..db98355b427 100644 --- a/content/en/functions/collections/Append.md +++ b/content/en/functions/collections/Append.md @@ -1,22 +1,17 @@ --- title: collections.Append -linkTitle: append description: Appends one or more elements to a slice and returns the resulting slice. -categories: [functions] +categories: [] keywords: [] -menu: - docs: - parent: functions -function: +action: aliases: [append] + related: + - functions/collections/Merge + - functions/collections/Slice returnType: any signatures: - - COLLECTION | collections.Append ELEMENT [ELEMENT]... + - COLLECTION | collections.Append ELEMENT [ELEMENT...] - COLLECTION | collections.Append COLLECTION -relatedFunctions: - - collections.Append - - collections.Merge - - collections.Slice aliases: [/functions/append] --- @@ -100,7 +95,7 @@ Although the elements in the examples above are strings, you can use the `append {{ with $p }} {{ end }} diff --git a/content/en/functions/collections/Apply.md b/content/en/functions/collections/Apply.md index 4d972b85358..d8a652d80a3 100644 --- a/content/en/functions/collections/Apply.md +++ b/content/en/functions/collections/Apply.md @@ -2,17 +2,13 @@ title: collections.Apply linkTitle: apply description: Returns a new collection with each element transformed by the given function. -categories: [functions] +categories: [] keywords: [] -menu: - docs: - parent: functions -function: +action: aliases: [apply] returnType: '[]any' signatures: [collections.Apply COLLECTION FUNCTION PARAM...] relatedFunctions: - - collections.Apply - collections.Delimit - collections.In - collections.Reverse @@ -25,7 +21,6 @@ The `apply` function takes three or more arguments, depending on the function be The first argument is the collection itself, the second argument is the function name, and the remaining arguments are passed to the function, with the string `"."` representing the collection element. - ```go-html-template {{ $s := slice "hello" "world" }} diff --git a/content/en/functions/collections/Complement.md b/content/en/functions/collections/Complement.md index 28b7ded3dc9..773323b3884 100644 --- a/content/en/functions/collections/Complement.md +++ b/content/en/functions/collections/Complement.md @@ -1,21 +1,16 @@ --- title: collections.Complement -linkTitle: complement description: Returns the elements of the last collection that are not in any of the others. -categories: [functions] +categories: [] keywords: [] -menu: - docs: - parent: functions -function: +action: aliases: [complement] + related: + - functions/collections/Intersect + - functions/collections/SymDiff + - functions/collections/Union returnType: any - signatures: ['collections.Complement COLLECTION [COLLECTION]...'] -relatedFunctions: - - collections.Complement - - collections.Intersect - - collections.SymDiff - - collections.Union + signatures: ['collections.Complement COLLECTION [COLLECTION...]'] aliases: [/functions/complement] --- @@ -35,7 +30,6 @@ Make your code simpler to understand by using a [chained pipeline]: [chained pipeline]: https://pkg.go.dev/text/template#hdr-Pipelines {{% /note %}} - ```go-html-template {{ $c3 | complement $c1 $c2 }} → [1 2] ``` @@ -57,7 +51,7 @@ To list everything except blog articles (`blog`) and frequently asked questions {{ $blog := where site.RegularPages "Type" "blog" }} {{ $faqs := where site.RegularPages "Type" "faqs" }} {{ range site.RegularPages | complement $blog $faqs }} - {{ .LinkTitle }} + {{ .Title }} {{ end }} ``` @@ -65,11 +59,11 @@ To list everything except blog articles (`blog`) and frequently asked questions Although the example above demonstrates the `complement` function, you could use the [`where`] function as well: [`where`]: /functions/collections/where -{{% /note %}} +{{% /note %}} ```go-html-template {{ range where site.RegularPages "Type" "not in" (slice "blog" "faqs") }} - {{ .LinkTitle }} + {{ .Title }} {{ end }} ``` diff --git a/content/en/functions/collections/Delimit.md b/content/en/functions/collections/Delimit.md index 0fc3ef537c2..6aea467ee17 100644 --- a/content/en/functions/collections/Delimit.md +++ b/content/en/functions/collections/Delimit.md @@ -1,24 +1,19 @@ --- title: collections.Delimit -linkTitle: delimit description: Loops through any array, slice, or map and returns a string of all the values separated by a delimiter. -categories: [functions] +categories: [] keywords: [] -menu: - docs: - parent: functions -function: +action: aliases: [delimit] - returnType: template.HTML + related: + - functions/collections/Apply + - functions/collections/In + - functions/collections/Reverse + - functions/collections/Seq + - functions/collections/Slice + - functions/strings/Split + returnType: string signatures: ['collections.Delimit COLLECTION DELIMITER [LAST]'] -relatedFunctions: - - collections.Apply - - collections.Delimit - - collections.In - - collections.Reverse - - collections.Seq - - collections.Slice - - strings.Split aliases: [/functions/delimit] --- @@ -26,8 +21,8 @@ Delimit a slice: ```go-html-template {{ $s := slice "b" "a" "c" }} -{{ delimit $s ", " }} → "b, a, c" -{{ delimit $s ", " " and "}} → "b, a and c" +{{ delimit $s ", " }} → b, a, c +{{ delimit $s ", " " and "}} → b, a and c ``` Delimit a map: @@ -38,6 +33,6 @@ The `delimit` function sorts maps by key, returning the values. ```go-html-template {{ $m := dict "b" 2 "a" 1 "c" 3 }} -{{ delimit $m ", " }} → "1, 2, 3" -{{ delimit $m ", " " and "}} → "1, 2 and 3" +{{ delimit $m ", " }} → 1, 2, 3 +{{ delimit $m ", " " and "}} → 1, 2 and 3 ``` diff --git a/content/en/functions/collections/Dictionary.md b/content/en/functions/collections/Dictionary.md index 28c38772611..a90c9e590c2 100644 --- a/content/en/functions/collections/Dictionary.md +++ b/content/en/functions/collections/Dictionary.md @@ -1,22 +1,17 @@ --- title: collections.Dictionary -linkTitle: dict description: Creates a map from a list of key and value pairs. -categories: [functions] +categories: [] keywords: [] -menu: - docs: - parent: functions -function: +action: aliases: [dict] + related: + - functions/collections/Group + - functions/collections/IndexFunction + - functions/collections/IsSet + - functions/collections/Where returnType: mapany - signatures: ['collections.Dictionary KEY VALUE [KEY VALUE]...'] -relatedFunctions: - - collections.Dictionary - - collections.Group - - collections.Index - - collections.IsSet - - collections.Where + signatures: ['collections.Dictionary KEY VALUE [VALUE...]'] aliases: [/functions/dict] --- @@ -24,10 +19,23 @@ aliases: [/functions/dict] Note that the `key` can be either a `string` or a `string slice`. The latter is useful to create a deeply nested structure, e.g.: -```go-text-template +```go-html-template {{ $m := dict (slice "a" "b" "c") "value" }} ``` +The above produces this data structure: + + +```json +{ + "a": { + "b": { + "c": "value" + } + } +} +``` + ## Example: using `dict` to pass multiple values to a `partial` The partial below creates an SVG and expects `fill`, `height` and `width` from the caller: diff --git a/content/en/functions/collections/EchoParam.md b/content/en/functions/collections/EchoParam.md deleted file mode 100644 index 7617eedd90d..00000000000 --- a/content/en/functions/collections/EchoParam.md +++ /dev/null @@ -1,40 +0,0 @@ ---- -title: collections.EchoParam -linkTitle: echoParam -description: Prints a parameter if it is set. -categories: [functions] -keywords: [] -menu: - docs: - parent: functions -function: - aliases: [echoParam] - returnType: any - signatures: [collections.EchoParam COLLECTION KEY] -relatedFunctions: [] -aliases: [/functions/echoparam] ---- - -For example, consider this site configuration: - -{{< code-toggle file=hugo copy=false >}} -[params.footer] -poweredBy = 'Hugo' -{{< /code-toggle >}} - -To print the value of `poweredBy`: - -```go-html-template -{{ echoParam site.Params.footer "poweredby" }} → Hugo -``` - -{{% note %}} -When using the `echoParam` function you must reference the key using lower case. See the previous example. - -The `echoParam` function will be deprecated in a future release. Instead, use either of the constructs below. -{{% /note %}} - -```go-html-template -{{ site.Params.footer.poweredBy }} → Hugo -{{ index site.Params.footer "poweredBy" }} → Hugo -``` diff --git a/content/en/functions/collections/First.md b/content/en/functions/collections/First.md index ddb04538250..ab7ea0b7290 100644 --- a/content/en/functions/collections/First.md +++ b/content/en/functions/collections/First.md @@ -1,34 +1,28 @@ --- title: collections.First -linkTitle: first description: Slices an array to the first N elements. -categories: [functions] +categories: [] keywords: [] -menu: - docs: - parent: functions -function: +action: aliases: [first] + related: + - functions/collections/After + - functions/collections/Last returnType: any signatures: [collections.First LIMIT COLLECTION] -relatedFunctions: - - collections.After - - collections.First - - collections.Last aliases: [/functions/first] --- -`first` works in a similar manner to the [`limit` keyword in -SQL][limitkeyword]. It reduces the array to only the `first N` -elements. It takes the array and number of elements as input. +`first` works in a similar manner to the [`limit` keyword in SQL][limitkeyword]. It reduces the array to only the `first N` elements. It takes the array and number of elements as input. `first` takes two arguments: + 1. `number of elements` 2. `array` *or* `slice of maps or structs` {{< code file="layout/_default/section.html" >}} {{ range first 10 .Pages }} - {{ .Render "summary" }} + {{ .Render "summary" }} {{ end }} {{< /code >}} @@ -43,11 +37,10 @@ ranges through only the first 5 posts in that list: {{< code file="first-and-where-together.html" >}} {{ range first 5 (where site.RegularPages "Type" "in" site.Params.mainSections).ByTitle }} - {{ .Content }} + {{ .Content }} {{ end }} {{< /code >}} - [limitkeyword]: https://www.techonthenet.com/sql/select_limit.php [`where`]: /functions/collections/where [main sections]: /functions/collections/where#mainsections diff --git a/content/en/functions/collections/Group.md b/content/en/functions/collections/Group.md index 29220f1f780..cc919ca2a40 100644 --- a/content/en/functions/collections/Group.md +++ b/content/en/functions/collections/Group.md @@ -1,26 +1,21 @@ --- title: collections.Group -linkTitle: group description: Groups a list of pages. -categories: [functions] +categories: [] keywords: [] -menu: - docs: - parent: functions -function: +action: aliases: [group] + related: + - functions/collections/Dictionary + - functions/collections/IndexFunction + - functions/collections/IsSet + - functions/collections/Where returnType: any signatures: [PAGES | collections.Group KEY] -relatedFunctions: - - collections.Dictionary - - collections.Group - - collections.Index - - collections.IsSet - - collections.Where aliases: [/functions/group] --- -{{< code file="layouts/partials/groups.html" >}} +```go-html-template {{ $new := .Site.RegularPages | first 10 | group "New" }} {{ $old := .Site.RegularPages | last 10 | group "Old" }} {{ $groups := slice $new $old }} @@ -35,6 +30,6 @@ aliases: [/functions/group] {{ end }} {{ end }} -{{< /code >}} +``` The page group you get from `group` is of the same type you get from the built-in [group methods](/templates/lists#group-content) in Hugo. The above example can be [paginated](/templates/pagination/#list-paginator-pages). diff --git a/content/en/functions/collections/In.md b/content/en/functions/collections/In.md index 57ffbd6539d..8a3d31d0e85 100644 --- a/content/en/functions/collections/In.md +++ b/content/en/functions/collections/In.md @@ -1,22 +1,22 @@ --- title: collections.In -linkTitle: in description: Reports whether an element is in an array or slice, or if a substring is in a string. +categories: [] keywords: [] -menu: - docs: - parent: functions -function: +action: aliases: [in] + related: + - functions/collections/Slice + - functions/strings/Contains + - functions/strings/ContainsAny + - functions/strings/ContainsNonSpace + - functions/strings/HasPrefix + - functions/strings/HasSuffix returnType: bool signatures: [collections.In SET ITEM] -relatedFunctions: - - collections.Slice aliases: [/functions/in] --- - - ```go-html-template {{ $s := slice "a" "b" "c" }} {{ in $s "b" }} → true diff --git a/content/en/functions/collections/IndexFunction.md b/content/en/functions/collections/IndexFunction.md index cd063f36e93..5bd189d9f7e 100644 --- a/content/en/functions/collections/IndexFunction.md +++ b/content/en/functions/collections/IndexFunction.md @@ -1,71 +1,66 @@ --- title: collections.Index -linkTitle: index description: Looks up the index(es) or key(s) of the data structure passed into it. -categories: [functions] +categories: [] keywords: [] -menu: - docs: - parent: functions -function: +action: aliases: [index] + related: + - functions/collections/Dictionary + - functions/collections/Group + - functions/collections/IsSet + - functions/collections/Where returnType: any signatures: - collections.Index COLLECTION INDEXES - collections.Index COLLECTION KEYS -relatedFunctions: - - collections.Dictionary - - collections.EchoParam - - collections.Group - - collections.Index - - collections.IsSet - - collections.Where aliases: [/functions/index,/functions/index-function] --- The `index` functions returns the result of indexing its first argument by the following arguments. Each indexed item must be a map or a slice, e.g.: -```go-text-template +```go-html-template {{ $slice := slice "a" "b" "c" }} -{{ index $slice 1 }} => b +{{ index $slice 0 }} → a +{{ index $slice 1 }} → b + {{ $map := dict "a" 100 "b" 200 }} -{{ index $map "b" }} => 200 +{{ index $map "b" }} → 200 ``` The function takes multiple indices as arguments, and this can be used to get nested values, e.g.: -```go-text-template +```go-html-template {{ $map := dict "a" 100 "b" 200 "c" (slice 10 20 30) }} -{{ index $map "c" 1 }} => 20 +{{ index $map "c" 1 }} → 20 {{ $map := dict "a" 100 "b" 200 "c" (dict "d" 10 "e" 20) }} -{{ index $map "c" "e" }} => 20 +{{ index $map "c" "e" }} → 20 ``` You may write multiple indices as a slice: -```go-text-template +```go-html-template {{ $map := dict "a" 100 "b" 200 "c" (dict "d" 10 "e" 20) }} {{ $slice := slice "c" "e" }} -{{ index $map $slice }} => 20 +{{ index $map $slice }} → 20 ``` ## Example: load data from a path based on front matter parameters Assume you want to add a `location = ""` field to your front matter for every article written in `content/vacations/`. You want to use this field to populate information about the location at the bottom of the article in your `single.html` template. You also have a directory in `data/locations/` that looks like the following: -``` -. -└── data - └── locations - ├── abilene.toml - ├── chicago.toml - ├── oslo.toml - └── provo.toml +```text +data/ + └── locations/ + ├── abilene.toml + ├── chicago.toml + ├── oslo.toml + └── provo.toml ``` Here is an example: -{{< code-toggle file="data/locations/oslo" copy=false >}} +{{< code-toggle file="data/locations/oslo" >}} website = "https://www.oslo.kommune.no" pop_city = 658390 pop_metro = 1717900 @@ -73,7 +68,7 @@ pop_metro = 1717900 The example we will use will be an article on Oslo, whose front matter should be set to exactly the same name as the corresponding file name in `data/locations/`: -{{< code-toggle file="content/articles/oslo.md" fm=true copy=false >}} +{{< code-toggle file="content/articles/oslo.md" fm=true >}} title = "My Norwegian Vacation" location = "oslo" {{< /code-toggle >}} diff --git a/content/en/functions/collections/Intersect.md b/content/en/functions/collections/Intersect.md index 6a2c131b413..efcbbf4708e 100644 --- a/content/en/functions/collections/Intersect.md +++ b/content/en/functions/collections/Intersect.md @@ -1,21 +1,16 @@ --- title: collections.Intersect -linkTitle: intersect description: Returns the common elements of two arrays or slices, in the same order as the first array. -categories: [functions] +categories: [] keywords: [] -menu: - docs: - parent: functions -function: +action: aliases: [intersect] + related: + - functions/collections/Complement + - functions/collections/SymDiff + - functions/collections/Union returnType: any signatures: [collections.Intersect SET1 SET2] -relatedFunctions: - - collections.Complement - - collections.Intersect - - collections.SymDiff - - collections.Union aliases: [/functions/intersect] --- A useful example is to use it as `AND` filters when combined with where: @@ -32,6 +27,5 @@ The above fetches regular pages not of `page` or `about` type unless they are pi See [union](/functions/collections/union) for `OR`. - [partials]: /templates/partials/ [single]: /templates/single-page-templates/ diff --git a/content/en/functions/collections/IsSet.md b/content/en/functions/collections/IsSet.md index 93fb9f8f618..76b336ae377 100644 --- a/content/en/functions/collections/IsSet.md +++ b/content/en/functions/collections/IsSet.md @@ -1,28 +1,25 @@ --- title: collections.IsSet -linkTitle: isset description: Reports whether the key exists within the collection. -categories: [functions] +categories: [] keywords: [] -menu: - docs: - parent: functions -function: +action: aliases: [isset] + related: + - functions/collections/Dictionary + - functions/collections/Group + - functions/collections/IndexFunction + - functions/collections/Where + - functions/go-template/if + - functions/go-template/with returnType: bool signatures: [collections.IsSet COLLECTION KEY] -relatedFunctions: - - collections.Dictionary - - collections.Group - - collections.Index - - collections.IsSet - - collections.Where aliases: [/functions/isset] --- For example, consider this site configuration: -{{< code-toggle file=hugo copy=false >}} +{{< code-toggle file=hugo >}} [params] showHeroImage = false {{< /code-toggle >}} diff --git a/content/en/functions/collections/KeyVals.md b/content/en/functions/collections/KeyVals.md index f3e0c559d36..6019ede5121 100644 --- a/content/en/functions/collections/KeyVals.md +++ b/content/en/functions/collections/KeyVals.md @@ -1,21 +1,20 @@ --- title: collections.KeyVals -linkTitle: keyVals description: Returns a KeyVals struct. -categories: [functions] +categories: [] keywords: [] -menu: - docs: - parent: functions -function: +action: aliases: [keyVals] - returnType: KeyValues + related: + - methods/pages/Related + returnType: types.KeyValues signatures: [collections.KeyVals KEY VALUES...] -relatedFunctions: [] aliases: [/functions/keyvals] --- -The primary application for this function is the definition of the `namedSlices` parameter in the options map passed to the `.Related` method on the `Page` object. +The primary application for this function is the definition of the `namedSlices` parameter in the options map passed to the [`Related`] method on the `Pages` object. + +[`Related`]: /methods/pages/related See [related content](/content-management/related). diff --git a/content/en/functions/collections/Last.md b/content/en/functions/collections/Last.md index 3f84963542d..4eda570ff34 100644 --- a/content/en/functions/collections/Last.md +++ b/content/en/functions/collections/Last.md @@ -1,20 +1,15 @@ --- title: collections.Last -linkTitle: last description: Slices an array to the last N elements. -categories: [functions] +categories: [] keywords: [] -menu: - docs: - parent: functions -function: +action: aliases: [last] + related: + - functions/collections/After + - functions/collections/First returnType: any signatures: [collections.Last INDEX COLLECTION] -relatedFunctions: - - collections.After - - collections.First - - collections.Last aliases: [/functions/last] --- diff --git a/content/en/functions/collections/Merge.md b/content/en/functions/collections/Merge.md index 908f1738a76..3f5208cfc06 100644 --- a/content/en/functions/collections/Merge.md +++ b/content/en/functions/collections/Merge.md @@ -1,19 +1,14 @@ --- title: collections.Merge -linkTitle: merge description: Returns the result of merging two or more maps. -categories: [functions] +categories: [] keywords: [] -menu: - docs: - parent: functions -function: +action: aliases: [merge] + related: + - functions/collections/Append returnType: any signatures: [collections.Merge MAP MAP...] -relatedFunctions: - - collections.Append - - collections.Merge aliases: [/functions/merge] --- diff --git a/content/en/functions/collections/NewScratch.md b/content/en/functions/collections/NewScratch.md index 0df90bb96a3..1cfc2dc4b7b 100644 --- a/content/en/functions/collections/NewScratch.md +++ b/content/en/functions/collections/NewScratch.md @@ -1,22 +1,107 @@ --- title: collections.NewScratch -linkTitle: newScratch -description: Creates a new Scratch which can be used to store values in a thread safe way. -categories: [functions] +description: Returns a locally scoped "scratch pad" to store and manipulate data. +categories: [] keywords: [] -menu: - docs: - parent: functions -function: +action: aliases: [newScratch] - returnType: Scratch + related: + - methods/page/scratch + - methods/page/store + returnType: maps.Scratch signatures: [collections.NewScratch ] -relatedFunctions: [] --- +The `collections.NewScratch` function creates a locally scoped [scratch pad] to store and manipulate data. To create a scratch pad that is attached to a `Page` object, use the [`Scratch`] or [`Store`] method. + +[`Scratch`]: /methods/page/scratch +[`Store`]: /methods/page/store +[scratch pad]: /getting-started/glossary/#scratch-pad + +## Methods + +Set +: Sets the value of a given key. + +```go-html-template +{{ $s := newScratch }} +{{ $s.Set "greeting" "Hello" }} +``` + +Get +: Gets the value of a given key. + +```go-html-template +{{ $s := newScratch }} +{{ $s.Set "greeting" "Hello" }} +{{ $s.Get "greeting" }} → Hello +``` + +Add +: Adds a given value to existing value(s) of the given key. + +: For single values, `Add` accepts values that support Go's `+` operator. If the first `Add` for a key is an array or slice, the following adds will be appended to that list. + +```go-html-template +{{ $s := newScratch }} +{{ $s.Set "greeting" "Hello" }} +{{ $s.Add "greeting" "Welcome" }} +{{ $s.Get "greeting" }} → HelloWelcome +``` + +```go-html-template +{{ $s := newScratch }} +{{ $s.Set "total" 3 }} +{{ $s.Add "total" 7 }} +{{ $s.Get "total" }} → 10 +``` + +```go-html-template +{{ $s := newScratch }} +{{ $s.Set "greetings" (slice "Hello") }} +{{ $s.Add "greetings" (slice "Welcome" "Cheers") }} +{{ $s.Get "greetings" }} → [Hello Welcome Cheers] +``` + +SetInMap +: Takes a `key`, `mapKey` and `value` and adds a map of `mapKey` and `value` to the given `key`. + +```go-html-template +{{ $s := newScratch }} +{{ $s.SetInMap "greetings" "english" "Hello" }} +{{ $s.SetInMap "greetings" "french" "Bonjour" }} +{{ $s.Get "greetings" }} → map[english:Hello french:Bonjour] +``` + +DeleteInMap +: Takes a `key` and `mapKey` and removes the map of `mapKey` from the given `key`. + ```go-html-template -{{ $scratch := newScratch }} -{{ $scratch.Add "b" 2 }} -{{ $scratch.Add "b" 2 }} -{{ $scratch.Get "b" }} → 4 +{{ $s := newScratch }} +{{ $s.SetInMap "greetings" "english" "Hello" }} +{{ $s.SetInMap "greetings" "french" "Bonjour" }} +{{ $s.DeleteInMap "greetings" "english" }} +{{ $s.Get "greetings" }} → map[french:Bonjour] ``` + +GetSortedMapValues +: Returns an array of values from `key` sorted by `mapKey`. + +```go-html-template +{{ $s := newScratch }} +{{ $s.SetInMap "greetings" "english" "Hello" }} +{{ $s.SetInMap "greetings" "french" "Bonjour" }} +{{ $s.GetSortedMapValues "greetings" }} → [Hello Bonjour] +``` + +Delete +: Removes the given key. + +```go-html-template +{{ $s := newScratch }} +{{ $s.Set "greeting" "Hello" }} +{{ $s.Delete "greeting" }} +``` + +Values +: Returns the raw backing map. Do not use with `Scratch` or `Store` methods on a `Page` object due to concurrency issues. diff --git a/content/en/functions/collections/Querify.md b/content/en/functions/collections/Querify.md index c94d5113374..e195c417ff7 100644 --- a/content/en/functions/collections/Querify.md +++ b/content/en/functions/collections/Querify.md @@ -1,19 +1,15 @@ --- title: collections.Querify -linkTitle: querify description: Takes a set or slice of key-value pairs and returns a query string to be appended to URLs. -categories: [functions] +categories: [] keywords: [] -menu: - docs: - parent: functions -function: +action: aliases: [querify] returnType: string signatures: - - collections.Querify KEY VALUE [KEY VALUE]... + - collections.Querify VALUE [VALUE...] - collections.Querify COLLECTION -relatedFunctions: +related: - collections.Querify - urlquery aliases: [/functions/querify] diff --git a/content/en/functions/collections/Reverse.md b/content/en/functions/collections/Reverse.md index 521adc6f255..12c964c76e4 100644 --- a/content/en/functions/collections/Reverse.md +++ b/content/en/functions/collections/Reverse.md @@ -1,16 +1,13 @@ --- title: collections.Reverse description: Reverses the order of a collection. -categories: [functions] +categories: [] keywords: [] -menu: - docs: - parent: functions -function: +action: aliases: [] returnType: any signatures: [collections.Reverse COLLECTION] -relatedFunctions: +related: - collections.Apply - collections.Delimit - collections.In @@ -20,7 +17,6 @@ relatedFunctions: aliases: [/functions/collections.reverse] --- - ```go-html-template {{ slice 2 1 3 | collections.Reverse }} → [3 1 2] ``` diff --git a/content/en/functions/collections/Seq.md b/content/en/functions/collections/Seq.md index 65ff1432f70..fca3e92d7df 100644 --- a/content/en/functions/collections/Seq.md +++ b/content/en/functions/collections/Seq.md @@ -1,20 +1,16 @@ --- title: collections.Seq -linkTitle: seq description: Returns a slice of integers. -categories: [functions] +categories: [] keywords: [] -menu: - docs: - parent: functions -function: +action: aliases: [seq] returnType: '[]int' signatures: - collections.Seq LAST - collections.Seq FIRST LAST - collections.Seq FIRST INCREMENT LAST -relatedFunctions: +related: - collections.Apply - collections.Delimit - collections.In diff --git a/content/en/functions/collections/Shuffle.md b/content/en/functions/collections/Shuffle.md index 8388d533284..18b8cc664cd 100644 --- a/content/en/functions/collections/Shuffle.md +++ b/content/en/functions/collections/Shuffle.md @@ -1,18 +1,13 @@ --- title: collections.Shuffle -linkTitle: shuffle description: Returns a random permutation of a given array or slice. -keywords: [ordering] -categories: [functions] +categories: [] keywords: [] -menu: - docs: - parent: functions -function: +action: aliases: [shuffle] returnType: any signatures: [collections.Shuffle COLLECTION] -relatedFunctions: +related: - collections.Reverse - collections.Shuffle - collections.Sort diff --git a/content/en/functions/collections/Slice.md b/content/en/functions/collections/Slice.md index a30800ed359..1e372ce44f3 100644 --- a/content/en/functions/collections/Slice.md +++ b/content/en/functions/collections/Slice.md @@ -1,17 +1,13 @@ --- title: collections.Slice -linkTitle: slice description: Creates a slice (array) of all passed arguments. -categories: [functions] +categories: [] keywords: [] -menu: - docs: - parent: functions -function: +action: aliases: [slice] returnType: any signatures: [collections.Slice ITEM...] -relatedFunctions: +related: - collections.Append - collections.Apply - collections.Delimit @@ -22,8 +18,6 @@ relatedFunctions: aliases: [/functions/slice] --- -One use case is the concatenation of elements in combination with the [`delimit` function]: - ```go-html-template {{ $s := slice "a" "b" "c" }} {{ $s }} → [a b c] diff --git a/content/en/functions/collections/Sort.md b/content/en/functions/collections/Sort.md index bb0f82cded0..b67720b3824 100644 --- a/content/en/functions/collections/Sort.md +++ b/content/en/functions/collections/Sort.md @@ -1,17 +1,13 @@ --- title: collections.Sort -linkTitle: sort description: Sorts slices, maps, and page collections. -categories: [functions] +categories: [] keywords: [] -menu: - docs: - parent: functions -function: +action: aliases: [sort] returnType: any signatures: ['collections.Sort COLLECTION [KEY] [ORDER]'] -relatedFunctions: +related: - collections.Reverse - collections.Shuffle - collections.Sort @@ -27,7 +23,7 @@ The `ORDER` may be either `asc` (ascending) or `desc` (descending). The default The examples below assume this site configuration: -{{< code-toggle file="hugo" copy=false >}} +{{< code-toggle file=hugo >}} [params] grades = ['b','a','c'] {{< /code-toggle >}} @@ -36,10 +32,10 @@ grades = ['b','a','c'] Sort slice elements in ascending order using either of these constructs: -{{< code file="layouts/_default/single.html" copy=false >}} +```go-html-template {{ sort site.Params.grades }} → [a b c] {{ sort site.Params.grades "value" "asc" }} → [a b c] -{{< /code >}} +``` In the examples above, `value` is the `KEY` representing the value of the slice element. @@ -47,9 +43,9 @@ In the examples above, `value` is the `KEY` representing the value of the slice Sort slice elements in descending order: -{{< code file="layouts/_default/single.html" copy=false >}} +```go-html-template {{ sort site.Params.grades "value" "desc" }} → [c b a] -{{< /code >}} +``` In the example above, `value` is the `KEY` representing the value of the slice element. @@ -57,7 +53,7 @@ In the example above, `value` is the `KEY` representing the value of the slice e The examples below assume this site configuration: -{{< code-toggle file="hugo" copy=false >}} +{{< code-toggle file=hugo >}} [params.authors.a] firstName = "Marius" lastName = "Pontmercy" @@ -77,7 +73,7 @@ When sorting maps, the `KEY` argument must be lowercase. Sort map objects in ascending order using either of these constructs: -{{< code file="layouts/_default/single.html" copy=false >}} +```go-html-template {{ range sort site.Params.authors "firstname" }} {{ .firstName }} {{ end }} @@ -85,7 +81,7 @@ Sort map objects in ascending order using either of these constructs: {{ range sort site.Params.authors "firstname" "asc" }} {{ .firstName }} {{ end }} -{{< /code >}} +``` These produce: @@ -97,11 +93,11 @@ Jean Marius Victor Sort map objects in descending order: -{{< code file="layouts/_default/single.html" copy=false >}} +```go-html-template {{ range sort site.Params.authors "firstname" "desc" }} {{ .firstName }} {{ end }} -{{< /code >}} +``` This produces: @@ -125,11 +121,10 @@ Although you can use the `sort` function to sort a page collection, Hugo provide In this contrived example, sort the site's regular pages by `.Type` in descending order: -{{< code file="layouts/_default/home.html" copy=false >}} +```go-html-template {{ range sort site.RegularPages "Type" "desc" }}

{{ .Title }}

{{ end }} -{{< /code >}} - +``` [built-in methods for sorting page collections]: /templates/lists/#order-content diff --git a/content/en/functions/collections/SymDiff.md b/content/en/functions/collections/SymDiff.md index ea9f201239e..828c10ce532 100644 --- a/content/en/functions/collections/SymDiff.md +++ b/content/en/functions/collections/SymDiff.md @@ -1,17 +1,13 @@ --- title: collections.SymDiff -linkTitle: symdiff description: Returns the symmetric difference of two collections. -categories: [functions] +categories: [] keywords: [] -menu: - docs: - parent: functions -function: +action: aliases: [symdiff] returnType: any signatures: [COLLECTION | collections.SymDiff COLLECTION] -relatedFunctions: +related: - collections.Complement - collections.Intersect - collections.SymDiff @@ -25,4 +21,4 @@ Example: {{ slice 1 2 3 | symdiff (slice 3 4) }} → [1 2 4] ``` -Also see https://en.wikipedia.org/wiki/Symmetric_difference +Also see . diff --git a/content/en/functions/collections/Union.md b/content/en/functions/collections/Union.md index 119da6fb458..bdc7c223718 100644 --- a/content/en/functions/collections/Union.md +++ b/content/en/functions/collections/Union.md @@ -1,17 +1,13 @@ --- title: collections.Union -linkTitle: union description: Given two arrays or slices, returns a new array that contains the elements or objects that belong to either or both arrays/slices. -categories: [functions] +categories: [] keywords: [] -menu: - docs: - parent: functions -function: +action: aliases: [union] returnType: any signatures: [collections.Union SET1 SET2] -relatedFunctions: +related: - collections.Complement - collections.Intersect - collections.SymDiff diff --git a/content/en/functions/collections/Uniq.md b/content/en/functions/collections/Uniq.md index 1b0a8f8f4d8..bda23956897 100644 --- a/content/en/functions/collections/Uniq.md +++ b/content/en/functions/collections/Uniq.md @@ -1,17 +1,13 @@ --- title: collections.Uniq -linkTitle: uniq description: Takes in a slice or array and returns a slice with duplicate elements removed. -categories: [functions] +categories: [] keywords: [] -menu: - docs: - parent: functions -function: +action: aliases: [uniq] returnType: any signatures: [collections.Uniq COLLECTION] -relatedFunctions: +related: - collections.Reverse - collections.Shuffle - collections.Sort @@ -19,7 +15,6 @@ relatedFunctions: aliases: [/functions/uniq] --- - ```go-html-template {{ slice 1 3 2 1 | uniq }} → [1 3 2] ``` diff --git a/content/en/functions/collections/Where.md b/content/en/functions/collections/Where.md index df6cec89f0b..3b8bbdd3758 100644 --- a/content/en/functions/collections/Where.md +++ b/content/en/functions/collections/Where.md @@ -1,17 +1,13 @@ --- title: collections.Where -linkTitle: where description: Filters an array to only the elements containing a matching value for a given field. -categories: [functions] +categories: [] keywords: [] -menu: - docs: - parent: functions -function: +action: aliases: [where] returnType: any signatures: ['collections.Where COLLECTION KEY [OPERATOR] MATCH'] -relatedFunctions: +related: - collections.Dictionary - collections.Group - collections.Index @@ -35,7 +31,7 @@ SQL][wherekeyword]. It can be used by dot-chaining the second argument to refer to a nested element of a value. -{{< code-toggle file="content/example.md" fm=true copy=false >}} +{{< code-toggle file="content/example.md" fm=true >}} title: Example series: golang {{< /code-toggle >}} @@ -106,13 +102,13 @@ When using booleans you should not put quotation marks. You can also put the returned value of the `where` clauses into a variable: -{{< code file="where-intersect-variables.html" >}} +```go-html-template {{ $v1 := where .Site.Pages "Params.a" "v1" }} {{ $v2 := where .Site.Pages "Params.b" "v2" }} {{ $filtered := $v1 | intersect $v2 }} {{ range $filtered }} {{ end }} -{{< /code >}} +``` ## Use `where` with `like` @@ -120,11 +116,11 @@ This example matches pages where the "foo" parameter begins with "ab": ```go-html-template {{ range where site.RegularPages "Params.foo" "like" `^ab` }} -

{{ .LinkTitle }}

+

{{ .Title }}

{{ end }} ``` -{{% readfile file="/functions/_common/regular-expressions.md" %}} +{{% include "functions/_common/regular-expressions.md" %}} ## Use `where` with `first` @@ -134,11 +130,11 @@ sections**](#mainsections), sorts it using the [default ordering](/templates/lists/) for lists (i.e., `weight => date`), and then ranges through only the first 5 posts in that list: -{{< code file="first-and-where-together.html" >}} +```go-html-template {{ range first 5 (where site.RegularPages "Type" "in" site.Params.mainSections) }} {{ .Content }} {{ end }} -{{< /code >}} +``` ## Nest `where` clauses @@ -181,7 +177,7 @@ If the user has not set this configuration parameter in their site configuration The user can override the default: -{{< code-toggle file="hugo" >}} +{{< code-toggle file=hugo >}} [params] mainSections = ["blog", "docs"] {{< /code-toggle >}} diff --git a/content/en/functions/collections/_index.md b/content/en/functions/collections/_index.md new file mode 100644 index 00000000000..51981f79b72 --- /dev/null +++ b/content/en/functions/collections/_index.md @@ -0,0 +1,12 @@ +--- +title: Collections functions +linkTitle: collections +description: Template functions to work with arrays, slices, maps, and page collections. +categories: [] +keywords: [] +menu: + docs: + parent: functions +--- + +Use these functions to work with arrays, slices, maps, and page collections. diff --git a/content/en/functions/compare/Cond.md b/content/en/functions/compare/Conditional.md similarity index 79% rename from content/en/functions/compare/Cond.md rename to content/en/functions/compare/Conditional.md index 4b92a893c10..6d693770d85 100644 --- a/content/en/functions/compare/Cond.md +++ b/content/en/functions/compare/Conditional.md @@ -1,19 +1,14 @@ --- title: compare.Conditional -linkTitle: cond description: Returns one of two arguments depending on the value of the control argument. -categories: [functions] +categories: [] keywords: [] -menu: - docs: - parent: functions -function: +action: aliases: [cond] + related: + - functions/compare/Default returnType: any signatures: [compare.Conditional CONTROL ARG1 ARG2] -relatedFunctions: - - compare.Conditional - - compare.Default aliases: [/functions/cond] --- @@ -21,14 +16,14 @@ The CONTROL argument is a boolean value that indicates whether the function shou ```go-html-template {{ $qty := 42 }} -{{ cond (le $qty 3) "few" "many" }} → "many" +{{ cond (le $qty 3) "few" "many" }} → many ``` The CONTROL argument must be either `true` or `false`. To cast a non-boolean value to boolean, pass it through the `not` operator twice. ```go-html-template -{{ cond (42 | not | not) "truthy" "falsy" }} → "truthy" -{{ cond ("" | not | not) "truthy" "falsy" }} → "falsy" +{{ cond (42 | not | not) "truthy" "falsy" }} → truthy +{{ cond ("" | not | not) "truthy" "falsy" }} → falsy ``` {{% note %}} @@ -38,7 +33,6 @@ Unlike [ternary operators] in other languages, the `cond` function does not perf [ternary operators]: https://en.wikipedia.org/wiki/Ternary_conditional_operator {{% /note %}} - Due to the absence of short-circuit evaluation, these examples throw an error: ```go-html-template diff --git a/content/en/functions/compare/Default.md b/content/en/functions/compare/Default.md index 24ad37ef2de..1e6bd7968ad 100644 --- a/content/en/functions/compare/Default.md +++ b/content/en/functions/compare/Default.md @@ -1,88 +1,48 @@ --- title: compare.Default -linkTitle: default -description: Allows setting a default value that can be returned if a first value is not set. -categories: [functions] +description: Returns the second argument if set, else the first argument. keywords: [] -menu: - docs: - parent: functions -function: +action: aliases: [default] + related: + - functions/compare/Conditional + - functions/go-template/Or returnType: any signatures: [compare.Default DEFAULT INPUT] -relatedFunctions: - - compare.Conditional - - compare.Default aliases: [/functions/default] --- -`default` checks whether a given value is set and returns a default value if it is not. *Set* in this context means different things depending on the data type: +The `default` function returns the second argument if set, else the first argument. -* non-zero for numeric types and times -* non-zero length for strings, arrays, slices, and maps -* any boolean or struct value -* non-nil for any other types +{{% note %}} +When the second argument is the boolean `false` value, the `default` function returns `false`. All _other_ falsy values are considered unset. -`default` function examples reference the following content page: +{{% include "functions/go-template/_common/truthy-falsy.md" %}} -{{< code file="content/posts/default-function-example.md" >}} ---- -title: Sane Defaults -seo_title: -date: 2017-02-18 -font: -oldparam: The default function helps make your templating DRYer. -newparam: ---- -{{< /code >}} - -`default` can be written in more than one way: - -```go-html-template -{{ .Params.font | default "Roboto" }} -{{ default "Roboto" .Params.font }} -``` - -Both of the above `default` function calls return `Roboto`. - -A `default` value, however, does not need to be hard coded like the previous example. The `default` value can be a variable or pulled directly from the front matter using dot notation: - -```go-html-template -{{ $old := .Params.oldparam }} -

{{ .Params.newparam | default $old }}

-``` - -Which would return: - -```html -

The default function helps make your templating DRYer.

-``` - -And then using dot notation - -```go-html-template -{{ .Params.seo_title | default .Title }} -``` - -Which would return - -```html -Sane Defaults -``` +To set a default value based on truthiness, use the [`or`] operator instead. -The following have equivalent return values but are far less terse. This demonstrates the utility of `default`: +[`or`]: /functions/go-template/or +{{% /note %}} -Using `if`: +The `default` function returns the second argument if set: ```go-html-template -{{ if .Params.seo_title }}{{ .Params.seo_title }}{{ else }}{{ .Title }}{{ end }} -=> Sane Defaults +{{ default 42 1 }} → 1 +{{ default 42 "foo" }} → foo +{{ default 42 (dict "k" "v") }} → map[k:v] +{{ default 42 (slice "a" "b") }} → [a b] +{{ default 42 true }} → true + + +{{ default 42 false }} → false ``` -Using `with`: +The `default` function returns the first argument if the second argument is not set: ```go-html-template -{{ with .Params.seo_title }}{{ . }}{{ else }}{{ .Title }}{{ end }} -=> Sane Defaults +{{ default 42 0 }} → 42 +{{ default 42 "" }} → 42 +{{ default 42 dict }} → 42 +{{ default 42 slice }} → 42 +{{ default 42 }} → 42 ``` diff --git a/content/en/functions/compare/Eq.md b/content/en/functions/compare/Eq.md index 010fc51b349..49350e676cb 100644 --- a/content/en/functions/compare/Eq.md +++ b/content/en/functions/compare/Eq.md @@ -1,23 +1,18 @@ --- title: compare.Eq -linkTitle: eq description: Returns the boolean truth of arg1 == arg2 || arg1 == arg3. -categories: [functions] +categories: [] keywords: [] -menu: - docs: - parent: functions -function: +action: aliases: [eq] + related: + - functions/compare/Ge + - functions/compare/Gt + - functions/compare/Le + - functions/compare/Lt + - functions/compare/Ne returnType: bool signatures: ['compare.Eq ARG1 ARG2 [ARG...]'] -relatedFunctions: - - compare.Eq - - compare.Ge - - compare.Gt - - compare.Le - - compare.Lt - - compare.Ne aliases: [/functions/eq] --- diff --git a/content/en/functions/compare/Ge.md b/content/en/functions/compare/Ge.md index 6bb48dd0094..479ecf9905b 100644 --- a/content/en/functions/compare/Ge.md +++ b/content/en/functions/compare/Ge.md @@ -1,23 +1,18 @@ --- title: compare.Ge -linkTitle: ge description: Returns the boolean truth of arg1 >= arg2 && arg1 >= arg3. -categories: [functions] +categories: [] keywords: [] -menu: - docs: - parent: functions -function: +action: aliases: [ge] + related: + - functions/compare/Eq + - functions/compare/Gt + - functions/compare/Le + - functions/compare/Lt + - functions/compare/Ne returnType: bool signatures: ['compare.Ge ARG1 ARG2 [ARG...]'] -relatedFunctions: - - compare.Eq - - compare.Ge - - compare.Gt - - compare.Le - - compare.Lt - - compare.Ne aliases: [/functions/ge] --- diff --git a/content/en/functions/compare/Gt.md b/content/en/functions/compare/Gt.md index 4691718ef6c..0af289ce20d 100644 --- a/content/en/functions/compare/Gt.md +++ b/content/en/functions/compare/Gt.md @@ -1,23 +1,18 @@ --- title: compare.Gt -linkTitle: gt description: Returns the boolean truth of arg1 > arg2 && arg1 > arg3. -categories: [functions] +categories: [] keywords: [] -menu: - docs: - parent: functions -function: +action: aliases: [gt] + related: + - functions/compare/Eq + - functions/compare/Ge + - functions/compare/Le + - functions/compare/Lt + - functions/compare/Ne returnType: bool signatures: ['compare.Gt ARG1 ARG2 [ARG...]'] -relatedFunctions: - - compare.Eq - - compare.Ge - - compare.Gt - - compare.Le - - compare.Lt - - compare.Ne aliases: [/functions/gt] --- diff --git a/content/en/functions/compare/Le.md b/content/en/functions/compare/Le.md index 792ea6ce6c1..319d376f609 100644 --- a/content/en/functions/compare/Le.md +++ b/content/en/functions/compare/Le.md @@ -1,23 +1,18 @@ --- title: compare.Le -linkTitle: le description: Returns the boolean truth of arg1 <= arg2 && arg1 <= arg3. -categories: [functions] +categories: [] keywords: [] -menu: - docs: - parent: functions -function: +action: aliases: [le] + related: + - functions/compare/Eq + - functions/compare/Ge + - functions/compare/Gt + - functions/compare/Lt + - functions/compare/Ne returnType: bool signatures: ['compare.Le ARG1 ARG2 [ARG...]'] -relatedFunctions: - - compare.Eq - - compare.Ge - - compare.Gt - - compare.Le - - compare.Lt - - compare.Ne aliases: [/functions/le] --- diff --git a/content/en/functions/compare/Lt.md b/content/en/functions/compare/Lt.md index 537c23b6ffd..3fe8f1d2c38 100644 --- a/content/en/functions/compare/Lt.md +++ b/content/en/functions/compare/Lt.md @@ -1,23 +1,18 @@ --- title: compare.Lt -linkTitle: lt description: Returns the boolean truth of arg1 < arg2 && arg1 < arg3. -categories: [functions] +categories: [] keywords: [] -menu: - docs: - parent: functions -function: +action: aliases: [lt] + related: + - functions/compare/Eq + - functions/compare/Ge + - functions/compare/Gt + - functions/compare/Le + - functions/compare/Ne returnType: bool signatures: ['compare.Lt ARG1 ARG2 [ARG...]'] -relatedFunctions: - - compare.Eq - - compare.Ge - - compare.Gt - - compare.Le - - compare.Lt - - compare.Ne aliases: [/functions/lt] --- diff --git a/content/en/functions/compare/Ne.md b/content/en/functions/compare/Ne.md index 412f43d4989..2d9f826fc73 100644 --- a/content/en/functions/compare/Ne.md +++ b/content/en/functions/compare/Ne.md @@ -1,23 +1,18 @@ --- title: compare.Ne -linkTitle: ne description: Returns the boolean truth of arg1 != arg2 && arg1 != arg3. -categories: [functions] +categories: [] keywords: [] -menu: - docs: - parent: functions -function: +action: aliases: [ne] + related: + - functions/compare/Eq + - functions/compare/Ge + - functions/compare/Gt + - functions/compare/Le + - functions/compare/Lt returnType: bool signatures: ['compare.Ne ARG1 ARG2 [ARG...]'] -relatedFunctions: - - compare.Eq - - compare.Ge - - compare.Gt - - compare.Le - - compare.Lt - - compare.Ne aliases: [/functions/ne] --- diff --git a/content/en/functions/compare/_index.md b/content/en/functions/compare/_index.md new file mode 100644 index 00000000000..a9b3a7b27c5 --- /dev/null +++ b/content/en/functions/compare/_index.md @@ -0,0 +1,12 @@ +--- +title: Compare functions +linkTitle: compare +description: Template functions to compare two or more values. +categories: [] +keywords: [] +menu: + docs: + parent: functions +--- + +Use these functions to compare two or more values. diff --git a/content/en/functions/crypto/FNV32a.md b/content/en/functions/crypto/FNV32a.md index 7a7fe303edc..5c091ebee13 100644 --- a/content/en/functions/crypto/FNV32a.md +++ b/content/en/functions/crypto/FNV32a.md @@ -1,21 +1,17 @@ --- title: crypto.FNV32a description: Returns the FNV (Fowler–Noll–Vo) 32 bit hash of a given string. -categories: [functions] +categories: [] keywords: [] -menu: - docs: - parent: functions -function: +action: aliases: [] + related: + - functions/crypto/HMAC + - functions/crypto/MD5 + - functions/crypto/SHA1 + - functions/crypto/SHA256 returnType: int signatures: [crypto.FNV32a STRING] -relatedFunctions: - - crypto.FNV32a - - crypto.HMAC - - crypto.MD5 - - crypto.SHA1 - - crypto.SHA256 aliases: [/functions/crypto.fnv32a] --- diff --git a/content/en/functions/crypto/HMAC.md b/content/en/functions/crypto/HMAC.md index e58619b3895..1906689a2b3 100644 --- a/content/en/functions/crypto/HMAC.md +++ b/content/en/functions/crypto/HMAC.md @@ -1,22 +1,17 @@ --- title: crypto.HMAC -linkTitle: hmac description: Returns a cryptographic hash that uses a key to sign a message. -categories: [functions] +categories: [] keywords: [] -menu: - docs: - parent: functions -function: +action: aliases: [hmac] + related: + - functions/crypto/FNV32a + - functions/crypto/MD5 + - functions/crypto/SHA1 + - functions/crypto/SHA256 returnType: string signatures: ['crypto.HMAC HASH_TYPE KEY MESSAGE [ENCODING]'] -relatedFunctions: - - crypto.FNV32a - - crypto.HMAC - - crypto.MD5 - - crypto.SHA1 - - crypto.SHA256 aliases: [/functions/hmac] --- diff --git a/content/en/functions/crypto/MD5.md b/content/en/functions/crypto/MD5.md index 9415e015cec..6c78ae55f62 100644 --- a/content/en/functions/crypto/MD5.md +++ b/content/en/functions/crypto/MD5.md @@ -1,28 +1,22 @@ --- title: crypto.MD5 -linkTitle: md5 -description: hashes the given input and returns its MD5 checksum. -categories: [functions] +description: Hashes the given input and returns its MD5 checksum. +categories: [] keywords: [] -menu: - docs: - parent: functions -function: +action: aliases: [md5] + related: + - functions/crypto/FNV32a + - functions/crypto/HMAC + - functions/crypto/SHA1 + - functions/crypto/SHA256 returnType: string signatures: [crypto.MD5 INPUT] -relatedFunctions: - - crypto.FNV32a - - crypto.HMAC - - crypto.MD5 - - crypto.SHA1 - - crypto.SHA256 aliases: [/functions/md5] --- ```go-html-template {{ md5 "Hello world" }} → 3e25960a79dbc69b674cd4ec67a72c62 - ``` This can be useful if you want to use [Gravatar](https://en.gravatar.com/) for generating a unique avatar: diff --git a/content/en/functions/crypto/SHA1.md b/content/en/functions/crypto/SHA1.md index 6269efe3814..247c9842a68 100644 --- a/content/en/functions/crypto/SHA1.md +++ b/content/en/functions/crypto/SHA1.md @@ -1,22 +1,17 @@ --- title: crypto.SHA1 -linkTitle: sha1 description: Hashes the given input and returns its SHA1 checksum. -categories: [functions] +categories: [] keywords: [] -menu: - docs: - parent: functions -function: +action: aliases: [sha1] + related: + - functions/crypto/FNV32a + - functions/crypto/HMAC + - functions/crypto/MD5 + - functions/crypto/SHA256 returnType: string signatures: [crypto.SHA1 INPUT] -relatedFunctions: - - crypto.FNV32a - - crypto.HMAC - - crypto.MD5 - - crypto.SHA1 - - crypto.SHA256 aliases: [/functions/sha,/functions/sha1] --- diff --git a/content/en/functions/crypto/SHA256.md b/content/en/functions/crypto/SHA256.md index 3019432d2a1..279cec35ce3 100644 --- a/content/en/functions/crypto/SHA256.md +++ b/content/en/functions/crypto/SHA256.md @@ -1,22 +1,17 @@ --- title: crypto.SHA256 -linkTitle: sha256 description: Hashes the given input and returns its SHA256 checksum. -categories: [functions] +categories: [] keywords: [] -menu: - docs: - parent: functions -function: +action: aliases: [sha256] + related: + - functions/crypto/FNV32a + - functions/crypto/HMAC + - functions/crypto/MD5 + - functions/crypto/SHA1 returnType: string signatures: [crypto.SHA256 INPUT] -relatedFunctions: - - crypto.FNV32a - - crypto.HMAC - - crypto.MD5 - - crypto.SHA1 - - crypto.SHA256 aliases: [/functions/sha256] --- diff --git a/content/en/functions/crypto/_index.md b/content/en/functions/crypto/_index.md new file mode 100644 index 00000000000..5c95aab6e45 --- /dev/null +++ b/content/en/functions/crypto/_index.md @@ -0,0 +1,12 @@ +--- +title: Crypto functions +linkTitle: crypto +description: Template functions to create cryptographic hashes. +categories: [] +keywords: [] +menu: + docs: + parent: functions +--- + +Use these functions to create cryptographic hashes. diff --git a/content/en/functions/data/GetCSV.md b/content/en/functions/data/GetCSV.md index e02c1588c31..b49f190e334 100644 --- a/content/en/functions/data/GetCSV.md +++ b/content/en/functions/data/GetCSV.md @@ -1,19 +1,17 @@ --- title: data.GetCSV -linkTitle: getCSV description: Returns an array of arrays from a local or remote CSV file, or an error if the file does not exist. -categories: [functions] +categories: [] keywords: [] -menu: - docs: - parent: functions -function: +action: aliases: [getCSV] - returnType: '[]string' - signatures: [data.GetCSV SEPARATOR PATHPART...] -relatedFunctions: - - data.GetCSV - - data.GetJSON + related: + - functions/data/GetJSON + - functions/resources/Get + - functions/resources/GetRemote + - methods/page/Resources + returnType: '[][]string' + signatures: ['data.GetCSV SEPARATOR INPUT... [OPTIONS]'] toc: true --- @@ -32,6 +30,12 @@ Access the data with either of the following: {{ $data := getCSV "," "other-files/" "pets.csv" }} ``` +{{% note %}} +When working with local data, the filepath is relative to the working directory. + +You must not place CSV files in the project's data directory. +{{% /note %}} + Access remote data with either of the following: ```go-html-template @@ -49,9 +53,25 @@ The resulting data structure is an array of arrays: ] ``` +## Options + +Add headers to the request by providing an options map: + +```go-html-template +{{ $opts := dict "Authorization" "Bearer abcd" }} +{{ $data := getCSV "," "https://example.org/pets.csv" $opts }} +``` + +Add multiple headers using a slice: + +```go-html-template +{{ $opts := dict "X-List" (slice "a" "b" "c") }} +{{ $data := getCSV "," "https://example.org/pets.csv" $opts }} +``` + ## Global resource alternative -Consider using `resources.Get` with [`transform.Unmarshal`] when accessing a global resource. +Consider using the [`resources.Get`] function with [`transform.Unmarshal`] when accessing a global resource. ```text my-project/ @@ -73,7 +93,9 @@ my-project/ ## Page resource alternative -Consider using `.Resources.Get` with [`transform.Unmarshal`] when accessing a page resource. +Consider using the [`Resources.Get`] method with [`transform.Unmarshal`] when accessing a page resource. + + ```text my-project/ @@ -97,7 +119,7 @@ my-project/ ## Remote resource alternative -Consider using `resources.GetRemote` with [`transform.Unmarshal`] for improved error handling when accessing a remote resource. +Consider using the [`resources.GetRemote`] function with [`transform.Unmarshal`] when accessing a remote resource to improve error handling and cache control. ```go-html-template {{ $data := "" }} @@ -114,4 +136,7 @@ Consider using `resources.GetRemote` with [`transform.Unmarshal`] for improved e {{ end }} ``` +[`Resources.Get`]: methods/page/Resources +[`resources.GetRemote`]: /functions/resources/getremote +[`resources.Get`]: /functions/resources/get [`transform.Unmarshal`]: /functions/transform/unmarshal diff --git a/content/en/functions/data/GetJSON.md b/content/en/functions/data/GetJSON.md index 37ee8e9a159..96812e7c063 100644 --- a/content/en/functions/data/GetJSON.md +++ b/content/en/functions/data/GetJSON.md @@ -1,19 +1,17 @@ --- title: data.GetJSON -linkTitle: getJSON description: Returns a JSON object from a local or remote JSON file, or an error if the file does not exist. -categories: [functions] +categories: [] keywords: [] -menu: - docs: - parent: functions -function: +action: aliases: [getJSON] + related: + - functions/data/GetCSV + - functions/resources/Get + - functions/resources/GetRemote + - methods/page/Resources returnType: any - signatures: [data.GetJSON PATHPART...] -relatedFunctions: - - data.GetCSV - - data.GetJSON + signatures: ['data.GetJSON INPUT... [OPTIONS]'] toc: true --- @@ -28,15 +26,19 @@ my-project/ Access the data with either of the following: ```go-html-template -{{ $data := getCSV "," "other-files/books.json" }} -{{ $data := getCSV "," "other-files/" "books.json" }} +{{ $data := getJSON "other-files/books.json" }} +{{ $data := getJSON "other-files/" "books.json" }} ``` +{{% note %}} +When working with local data, the filepath is relative to the working directory. +{{% /note %}} + Access remote data with either of the following: ```go-html-template -{{ $data := getCSV "," "https://example.org/books.json" }} -{{ $data := getCSV "," "https://example.org/" "books.json" }} +{{ $data := getJSON "https://example.org/books.json" }} +{{ $data := getJSON "https://example.org/" "books.json" }} ``` The resulting data structure is a JSON object: @@ -56,9 +58,25 @@ The resulting data structure is a JSON object: ] ``` +## Options + +Add headers to the request by providing an options map: + +```go-html-template +{{ $opts := dict "Authorization" "Bearer abcd" }} +{{ $data := getJSON "https://example.org/books.json" $opts }} +``` + +Add multiple headers using a slice: + +```go-html-template +{{ $opts := dict "X-List" (slice "a" "b" "c") }} +{{ $data := getJSON "https://example.org/books.json" $opts }} +``` + ## Global resource alternative -Consider using `resources.Get` with [`transform.Unmarshal`] when accessing a global resource. +Consider using the [`resources.Get`] function with [`transform.Unmarshal`] when accessing a global resource. ```text my-project/ @@ -80,7 +98,7 @@ my-project/ ## Page resource alternative -Consider using `.Resources.Get` with [`transform.Unmarshal`] when accessing a page resource. +Consider using the [`Resources.Get`] method with [`transform.Unmarshal`] when accessing a page resource. ```text my-project/ @@ -104,7 +122,7 @@ my-project/ ## Remote resource alternative -Consider using `resources.GetRemote` with [`transform.Unmarshal`] for improved error handling when accessing a remote resource. +Consider using the [`resources.GetRemote`] function with [`transform.Unmarshal`] when accessing a remote resource to improve error handling and cache control. ```go-html-template {{ $data := "" }} @@ -121,4 +139,7 @@ Consider using `resources.GetRemote` with [`transform.Unmarshal`] for improved e {{ end }} ``` +[`Resources.Get`]: methods/page/Resources +[`resources.GetRemote`]: /functions/resources/getremote +[`resources.Get`]: /functions/resources/get [`transform.Unmarshal`]: /functions/transform/unmarshal diff --git a/content/en/functions/data/_index.md b/content/en/functions/data/_index.md new file mode 100644 index 00000000000..142d6b52826 --- /dev/null +++ b/content/en/functions/data/_index.md @@ -0,0 +1,12 @@ +--- +title: Data functions +linkTitle: data +description: Template functions to read local or remote data files. +categories: [] +keywords: [] +menu: + docs: + parent: functions +--- + +Use these functions to read local or remote data files. diff --git a/content/en/functions/debug/Dump.md b/content/en/functions/debug/Dump.md index ff505a76b7a..d3161605fb9 100644 --- a/content/en/functions/debug/Dump.md +++ b/content/en/functions/debug/Dump.md @@ -1,16 +1,13 @@ --- title: debug.Dump description: Returns an object dump as a string. -categories: [functions] +categories: [] keywords: [] -menu: - docs: - parent: functions -function: +action: aliases: [] + related: [] returnType: string signatures: [debug.Dump VALUE] -relatedFunctions: [] --- ```go-html-template @@ -43,8 +40,6 @@ relatedFunctions: [] } ``` - - {{% note %}} Output from this function may change from one release to the next. Use for debugging only. {{% /note %}} diff --git a/content/en/functions/debug/Timer.md b/content/en/functions/debug/Timer.md index cfa2ad6dc83..3e3af0a0c91 100644 --- a/content/en/functions/debug/Timer.md +++ b/content/en/functions/debug/Timer.md @@ -1,21 +1,18 @@ --- title: debug.Timer description: Creates a named timer that reports elapsed time to the console. -categories: [functions] +categories: [] keywords: [] -menu: - docs: - parent: functions -function: +action: aliases: [] + related: [] returnType: debug.Timer signatures: [debug.Timer NAME] -relatedFunctions: [] --- {{< new-in "0.120.0" >}} -Use the `debug.Timer` function to determine execution time for a block of code, useful for finding performance bottlenecks in templates. +Use the `debug.Timer` function to determine execution time for a block of code, useful for finding performance bottle necks in templates. The timer starts when you instantiate it, and stops when you call its `Stop` method. diff --git a/content/en/functions/debug/_index.md b/content/en/functions/debug/_index.md new file mode 100644 index 00000000000..4188285157d --- /dev/null +++ b/content/en/functions/debug/_index.md @@ -0,0 +1,12 @@ +--- +title: Debug functions +linkTitle: debug +description: Template functions to debug your templates. +categories: [] +keywords: [] +menu: + docs: + parent: functions +--- + +Use these functions to debug your templates. diff --git a/content/en/functions/encoding/Base64Decode.md b/content/en/functions/encoding/Base64Decode.md index 8bd554c83ba..821ca805ae6 100644 --- a/content/en/functions/encoding/Base64Decode.md +++ b/content/en/functions/encoding/Base64Decode.md @@ -1,24 +1,19 @@ --- title: encoding.Base64Decode -linkTitle: base64Decode description: Returns the base64 decoding of the given content. -categories: [functions] +categories: [] keywords: [] -menu: - docs: - parent: functions -function: +action: aliases: [base64Decode] + related: + - functions/encoding/Base64Encode returnType: string signatures: [encoding.Base64Decode INPUT] -signatures: - - - - base64Decode INPUT aliases: [/functions/base64Decode] --- ```go-html-template -{{ "SHVnbw==" | base64Decode }} → "Hugo" +{{ "SHVnbw==" | base64Decode }} → Hugo ``` Use the `base64Decode` function to decode responses from APIs. For example, the result of this call to GitHub's API contains the base64-encoded representation of the repository's README file: diff --git a/content/en/functions/encoding/Base64Encode.md b/content/en/functions/encoding/Base64Encode.md index d548aca8e1e..14f67a132f1 100644 --- a/content/en/functions/encoding/Base64Encode.md +++ b/content/en/functions/encoding/Base64Encode.md @@ -1,22 +1,17 @@ --- title: encoding.Base64Encode -linkTitle: base64Encode description: Returns the base64 decoding of the given content. -categories: [functions] +categories: [] keywords: [] -menu: - docs: - parent: functions -function: +action: aliases: [base64Encode] + related: + - functions/encoding/Base64Decode returnType: string signatures: [encoding.Base64Encode INPUT] -relatedFunctions: - - encoding.Base64Decode - - encoding.Base64Encode aliases: [/functions/base64, /functions/base64Encode] --- ```go-html-template -{{ "Hugo" | base64Encode }} → "SHVnbw==" +{{ "Hugo" | base64Encode }} → SHVnbw== ``` diff --git a/content/en/functions/encoding/Jsonify.md b/content/en/functions/encoding/Jsonify.md index 0b9cb2e7414..09b181e311a 100644 --- a/content/en/functions/encoding/Jsonify.md +++ b/content/en/functions/encoding/Jsonify.md @@ -1,31 +1,25 @@ --- title: encoding.Jsonify -linkTitle: jsonify description: Encodes a given object to JSON. -categories: [functions] +categories: [] keywords: [] -menu: - docs: - parent: functions -function: +action: aliases: [jsonify] returnType: template.HTML + related: + - functions/transform/Remarshal + - functions/transform/Unmarshal signatures: - encoding.Jsonify INPUT - encoding.Jsonify OPTIONS INPUT -relatedFunctions: - - encoding.Jsonify - - transform.Remarshal - - transform.Unmarshal aliases: [/functions/jsonify] --- -To customize the printing of the JSON, pass a map of options as the first +To customize the printing of the JSON, pass an options map as the first argument. Supported options are "prefix" and "indent". Each JSON element in the output will begin on a new line beginning with *prefix* followed by one or more copies of *indent* according to the indentation nesting. - ```go-html-template {{ dict "title" .Title "content" .Plain | jsonify }} {{ dict "title" .Title "content" .Plain | jsonify (dict "indent" " ") }} @@ -34,15 +28,15 @@ more copies of *indent* according to the indentation nesting. ## Options -indent ("") -: Indentation to use. +indent +: (`string`) Indentation to use. Default is "". -prefix ("") -: Indentation prefix. +prefix +: (`string`) Indentation prefix. Default is "". noHTMLEscape (false) -: Disable escaping of problematic HTML characters inside JSON quoted strings. The default behavior is to escape &, <, and > to \u0026, \u003c, and \u003e to avoid certain safety problems that can arise when embedding JSON in HTML. +: (`bool`) Disable escaping of problematic HTML characters inside JSON quoted strings. The default behavior is to escape `&`, `<`, and `>` to `\u0026`, `\u003c`, and `\u003e` to avoid certain safety problems that can arise when embedding JSON in HTML. Default is `false`. -See also the `.PlainWords`, `.Plain`, and `.RawContent` [page variables][pagevars]. +See also the `.PlainWords`, `.Plain`, and `.RawContent` [page variables]. -[pagevars]: /variables/page/ +[page variables]: /variables/page/ diff --git a/content/en/functions/encoding/_index.md b/content/en/functions/encoding/_index.md new file mode 100644 index 00000000000..3c4c4519ece --- /dev/null +++ b/content/en/functions/encoding/_index.md @@ -0,0 +1,12 @@ +--- +title: Encoding functions +linkTitle: encoding +description: Template functions to encode and decode data. +categories: [] +keywords: [] +menu: + docs: + parent: functions +--- + +Use these functions to encode and decode data. diff --git a/content/en/functions/fmt/Errorf.md b/content/en/functions/fmt/Errorf.md index da9845073d1..c23d27372d1 100644 --- a/content/en/functions/fmt/Errorf.md +++ b/content/en/functions/fmt/Errorf.md @@ -1,20 +1,15 @@ --- title: fmt.Errorf -linkTitle: errorf description: Log an ERROR from a template. -categories: [functions] +categories: [] keywords: [] -menu: - docs: - parent: functions -function: +action: aliases: [errorf] + related: + - functions/fmt/Erroridf + - functions/fmt/Warnf returnType: string signatures: ['fmt.Errorf FORMAT [INPUT]'] -relatedFunctions: - - fmt.Errorf - - fmt.Erroridf - - fmt.Warnf aliases: [/functions/errorf] --- diff --git a/content/en/functions/fmt/Erroridf.md b/content/en/functions/fmt/Erroridf.md index 98681043697..a84671e3e6b 100644 --- a/content/en/functions/fmt/Erroridf.md +++ b/content/en/functions/fmt/Erroridf.md @@ -1,20 +1,15 @@ --- title: fmt.Erroridf -linkTitle: erroridf description: Log a suppressable ERROR from a template. -categories: [functions] +categories: [] keywords: [] -menu: - docs: - parent: functions -function: +action: aliases: [erroridf] + related: + - functions/fmt/Errorf + - functions/fmt/Warnf returnType: string signatures: ['fmt.Erroridf ID FORMAT [INPUT]'] -relatedFunctions: - - fmt.Errorf - - fmt.Erroridf - - fmt.Warnf aliases: [/functions/erroridf] --- @@ -40,7 +35,7 @@ ignoreErrors = ["error-42"] To suppress this message: -{{< code-toggle file=hugo copy=false >}} +{{< code-toggle file=hugo >}} ignoreErrors = ["error-42"] {{< /code-toggle >}} diff --git a/content/en/functions/fmt/Print.md b/content/en/functions/fmt/Print.md index f9ff885f8ce..6f3128e5f01 100644 --- a/content/en/functions/fmt/Print.md +++ b/content/en/functions/fmt/Print.md @@ -1,25 +1,20 @@ --- title: fmt.Print -linkTitle: print description: Prints the default representation of the given arguments using the standard `fmt.Print` function. -categories: [functions] +categories: [] keywords: [] -menu: - docs: - parent: functions -function: +action: aliases: [print] + related: + - functions/fmt/Printf + - functions/fmt/Println returnType: string signatures: [fmt.Print INPUT] -relatedFunctions: - - fmt.Print - - fmt.Printf - - fmt.Println aliases: [/functions/print] --- ```go-html-template -{{ print "foo" }} → "foo" -{{ print "foo" "bar" }} → "foobar" +{{ print "foo" }} → foo +{{ print "foo" "bar" }} → foobar {{ print (slice 1 2 3) }} → [1 2 3] ``` diff --git a/content/en/functions/fmt/Printf.md b/content/en/functions/fmt/Printf.md index 06b7222e992..af2fd398dd5 100644 --- a/content/en/functions/fmt/Printf.md +++ b/content/en/functions/fmt/Printf.md @@ -1,20 +1,15 @@ --- title: fmt.Printf -linkTitle: printf description: Formats a string using the standard `fmt.Sprintf` function. -categories: [functions] +categories: [] keywords: [] -menu: - docs: - parent: functions -function: +action: aliases: [printf] + related: + - functions/fmt/Print + - functions/fmt/Println returnType: string signatures: ['fmt.Printf FORMAT [INPUT]'] -relatedFunctions: - - fmt.Print - - fmt.Printf - - fmt.Println aliases: [/functions/printf] --- diff --git a/content/en/functions/fmt/Println.md b/content/en/functions/fmt/Println.md index 358b5f8ac37..a4db56ffb7b 100644 --- a/content/en/functions/fmt/Println.md +++ b/content/en/functions/fmt/Println.md @@ -1,23 +1,18 @@ --- title: fmt.Println -linkTitle: println -description: Prints the default representation of the given argument using the standard `fmt.Print` function and enforces a linebreak. -categories: [functions] +description: Prints the default representation of the given argument using the standard `fmt.Print` function and enforces a line break. +categories: [] keywords: [] -menu: - docs: - parent: functions -function: +action: aliases: [println] + related: + - functions/fmt/Print + - functions/fmt/Printf returnType: string signatures: [fmt.Println INPUT] -relatedFunctions: - - fmt.Print - - fmt.Printf - - fmt.Println aliases: [/functions/println] --- ```go-html-template -{{ println "foo" }} → "foo\n" +{{ println "foo" }} → foo\n ``` diff --git a/content/en/functions/fmt/Warnf.md b/content/en/functions/fmt/Warnf.md index be579a21687..02dc0b9c183 100644 --- a/content/en/functions/fmt/Warnf.md +++ b/content/en/functions/fmt/Warnf.md @@ -1,20 +1,15 @@ --- title: fmt.Warnf -linkTitle: warnf description: Log a WARNING from a template. -categories: [functions] +categories: [] keywords: [] -menu: - docs: - parent: functions -function: +action: aliases: [warnf] + related: + - functions/fmt/Errorf + - functions/fmt/Erroridf returnType: string signatures: ['fmt.Warnf FORMAT [INPUT]'] -relatedFunctions: - - fmt.Errorf - - fmt.Erroridf - - fmt.Warnf aliases: [/functions/warnf] --- diff --git a/content/en/functions/fmt/_index.md b/content/en/functions/fmt/_index.md new file mode 100644 index 00000000000..51ef847ca73 --- /dev/null +++ b/content/en/functions/fmt/_index.md @@ -0,0 +1,12 @@ +--- +title: Fmt functions +linkTitle: fmt +description: Template functions to print strings within a template or to print messages to the terminal +categories: [] +keywords: [] +menu: + docs: + parent: functions +--- + +Use these functions to print strings within a template or to print messages to the terminal. diff --git a/content/en/functions/global/_index.md b/content/en/functions/global/_index.md new file mode 100644 index 00000000000..f8bc9457737 --- /dev/null +++ b/content/en/functions/global/_index.md @@ -0,0 +1,11 @@ +--- +title: Global functions +linkTitle: global +description: Template functions to access site, page, and application data. +categories: [] +menu: + docs: + parent: functions +--- + +Use these global functions to access site, page, and application data. diff --git a/content/en/functions/hugo/index.md b/content/en/functions/global/hugo.md similarity index 78% rename from content/en/functions/hugo/index.md rename to content/en/functions/global/hugo.md index 64ea0db221a..9407eb3ee0d 100644 --- a/content/en/functions/hugo/index.md +++ b/content/en/functions/global/hugo.md @@ -1,59 +1,55 @@ --- title: hugo description: Provides global access to Hugo-related data. -categories: [functions] +categories: [] keywords: [] -menu: - docs: - parent: functions -function: +action: aliases: [] + related: + - functions/global/page + - functions/global/site returnType: signatures: [hugo] -relatedFunctions: - - hugo - - page - - site aliases: [/functions/hugo] --- `hugo` returns an instance that contains the following functions: -`hugo.BuildDate` +hugo.BuildDate : (`string`) The compile date of the current Hugo binary formatted per [RFC 3339](https://datatracker.ietf.org/doc/html/rfc3339) (e.g., `2023-05-23T08:14:20Z`). -`hugo.CommitHash` +hugo.CommitHash : (`string`) The Git commit hash of the Hugo binary (e.g., `0a95d6704a8ac8d41cc5ca8fffaad8c5c7a3754a`). -`hugo.Deps` +hugo.Deps : (`[]*hugo.Dependency`) See [hugo.Deps](#hugodeps). -`hugo.Environment` +hugo.Environment : (`string`) The current running environment as defined through the `--environment` CLI flag (e.g., `development`, `production`). -`hugo.Generator` +hugo.Generator : (`template.HTML`) Renders an HTML `meta` element identifying the software that generated the site (e.g., ``). -`hugo.GoVersion` {{< new-in "0.101.0" >}} -: (`string`) The Go version used to compile the Hugo binary (e.g., `go1.20.4`). +hugo.GoVersion {{< new-in "0.101.0" >}} +: (`string`) The Go version used to compile the Hugo binary (e.g., `go1.20.4`). -`hugo.IsDevelopment` {{< new-in "0.120.0" >}} -: (`bool`) Returns `true` if `hugo.Environment` is "development". +hugo.IsDevelopment {{< new-in "0.120.0" >}} +: (`bool`) Reports whether the current running environment is "development". -`hugo.IsExtended` -: (`bool`) Returns `true` if the Hugo binary is the extended version. +hugo.IsExtended +: (`bool`) Reports whether the Hugo binary is the extended version. -`hugo.IsProduction` -: (`bool`) Returns `true` if `hugo.Environment` is "production". +hugo.IsProduction +: (`bool`) Reports whether the current running environment is "production". -`hugo.IsServer` {{< new-in "0.120.0" >}} -: (`bool`) Returns `true` if the site is being served with Hugo's built-in server. +hugo.IsServer {{< new-in "0.120.0" >}} +: (`bool`) Reports whether the built-in server is running. -`hugo.Version` +hugo.Version : (`hugo.VersionString`) The current version of the Hugo binary (e.g., `0.112.1`). -`hugo.WorkingDir` {{< new-in "0.112.0" >}} -: (`string`) The project working directory (e.g., `/home/user/projects/my-hugo-site`). +hugo.WorkingDir {{< new-in "0.112.0" >}} +: (`string`) The project working directory (e.g., `/home/user/projects/my-hugo-site`). ## hugo.Deps diff --git a/content/en/functions/page/index.md b/content/en/functions/global/page.md similarity index 91% rename from content/en/functions/page/index.md rename to content/en/functions/global/page.md index 01f01407893..094fc7c9fbd 100644 --- a/content/en/functions/page/index.md +++ b/content/en/functions/global/page.md @@ -1,19 +1,15 @@ --- title: page -description: Provides global access to the .Page object. -categories: [functions] +description: Provides global access to a Page object. +categories: [] keywords: [] -menu: - docs: - parent: functions -function: +action: aliases: [] + related: + - functions/global/hugo + - functions/global/site returnType: signatures: [page] -relatedFunctions: - - hugo - - page - - site aliases: [/functions/page] --- @@ -47,7 +43,7 @@ Use the global `page` function to access the `Page` object from anywhere in any ### Be aware of top-level context -The global `page` function accesses the `Page` object passed into the top-level template. +The global `page` function accesses the `Page` object passed into the top-level template. With this content structure: @@ -94,7 +90,7 @@ Consider this section template: ```go-html-template {{ range .Pages }} -

{{ .LinkTitle }}

+

{{ .Title }}

{{ .Summary }} {{ end }} ``` diff --git a/content/en/functions/site/index.md b/content/en/functions/global/site.md similarity index 75% rename from content/en/functions/site/index.md rename to content/en/functions/global/site.md index 3341bff9844..cba2359f4f8 100644 --- a/content/en/functions/site/index.md +++ b/content/en/functions/global/site.md @@ -1,19 +1,15 @@ --- title: site -description: Provides global access to the .Site object. -categories: [functions] +description: Provides global access to the Site object. +categories: [] keywords: [] -menu: - docs: - parent: functions -function: +action: aliases: [] + related: + - functions/global/hugo + - functions/global/page returnType: signatures: [site] -relatedFunctions: - - hugo - - page - - site aliases: [/functions/site] --- diff --git a/content/en/functions/go-template/_common/_index.md b/content/en/functions/go-template/_common/_index.md new file mode 100644 index 00000000000..47d5812fba5 --- /dev/null +++ b/content/en/functions/go-template/_common/_index.md @@ -0,0 +1,13 @@ +--- +cascade: + _build: + list: never + publishResources: false + render: never +--- + + diff --git a/content/en/functions/go-template/_common/text-template.md b/content/en/functions/go-template/_common/text-template.md new file mode 100644 index 00000000000..71718c3fd36 --- /dev/null +++ b/content/en/functions/go-template/_common/text-template.md @@ -0,0 +1,7 @@ +--- +# Do not remove front matter. +--- + +See Go's [text/template] documentation for more information. + +[text/template]: https://pkg.go.dev/text/template diff --git a/content/en/functions/go-template/_common/truthy-falsy.md b/content/en/functions/go-template/_common/truthy-falsy.md new file mode 100644 index 00000000000..c8fc9e09e4d --- /dev/null +++ b/content/en/functions/go-template/_common/truthy-falsy.md @@ -0,0 +1,5 @@ +--- +# Do not remove front matter. +--- + +In Go templates, the falsy values are `false`, `0`, any nil pointer or interface value, and any array, slice, map, or string of length zero. Everything else is truthy. diff --git a/content/en/functions/go-template/_index.md b/content/en/functions/go-template/_index.md new file mode 100644 index 00000000000..9d7f89e08d6 --- /dev/null +++ b/content/en/functions/go-template/_index.md @@ -0,0 +1,14 @@ +--- +title: Go template functions +linkTitle: go template +description: Template functions and statements provided by Go's text/template package. +categories: [] +keywords: [] +menu: + docs: + parent: functions +--- + +These are the statements, operators, and functions provided by Go's [text/template] package. + +[text/template]: https://pkg.go.dev/text/template diff --git a/content/en/functions/go-template/and.md b/content/en/functions/go-template/and.md new file mode 100644 index 00000000000..6bc8c60a577 --- /dev/null +++ b/content/en/functions/go-template/and.md @@ -0,0 +1,26 @@ +--- +title: and +description: Returns the first falsy argument. If all arguments are truthy, returns the last argument. +categories: [] +keywords: [] +action: + aliases: [] + related: + - functions/go-template/not + - functions/go-template/or + returnType: any + signatures: [and VALUE...] +--- + +{{% include "functions/go-template/_common/truthy-falsy.md" %}} + +```go-html-template +{{ and 1 0 "" }} → 0 (int) +{{ and 1 false 0 }} → false (bool) + +{{ and 1 2 3 }} → 3 (int) +{{ and "a" "b" "c" }} → c (string) +{{ and "a" 1 true }} → true (bool) +``` + +{{% include "functions/go-template/_common/text-template.md" %}} diff --git a/content/en/functions/go-template/block.md b/content/en/functions/go-template/block.md new file mode 100644 index 00000000000..f5c7644df71 --- /dev/null +++ b/content/en/functions/go-template/block.md @@ -0,0 +1,55 @@ +--- +title: block +description: Defines a template and executes it in place. +categories: [] +keywords: [] +action: + aliases: [] + related: + - functions/go-template/define + - functions/go-template/end + returnType: + signatures: [block NAME CONTEXT] +--- + +A block is shorthand for defining a template: + +```go-html-template +{{ define "name" }} T1 {{ end }} +``` + +and then executing it in place: + +```go-html-template +{{ template "name" pipeline }} +``` +The typical use is to define a set of root templates that are then customized by redefining the block templates within. + +{{< code file="layouts/_default/baseof.html" >}} + +
+ {{ block "main" . }} + {{ print "default value if 'main' template is empty" }} + {{ end }} +
+ +{{< /code >}} + +{{< code file="layouts/_default/single.html" >}} +{{ define "main" }} +

{{ .Title }}

+ {{ .Content }} +{{ end }} +{{< /code >}} + +{{< code file="layouts/_default/list.html" >}} +{{ define "main" }} +

{{ .Title }}

+ {{ .Content }} + {{ range .Pages }} +

{{ .Title }}

+ {{ end }} +{{ end }} +{{< /code >}} + +{{% include "functions/go-template/_common/text-template.md" %}} diff --git a/content/en/functions/go-template/break.md b/content/en/functions/go-template/break.md new file mode 100644 index 00000000000..14074d7c08c --- /dev/null +++ b/content/en/functions/go-template/break.md @@ -0,0 +1,33 @@ +--- +title: break +description: Used with the range statement, stops the innermost iteration and bypasses all remaining iterations. +categories: [] +keywords: [] +action: + aliases: [] + related: + - functions/go-template/continue + - functions/go-template/range + returnType: + signatures: [break] +--- + +This template code: + +```go-html-template +{{ $s := slice "foo" "bar" "baz" }} +{{ range $s }} + {{ if eq . "bar" }} + {{ break }} + {{ end }} +

{{ . }}

+{{ end }} +``` + +Is rendered to: + +```html +

foo

+``` + +{{% include "functions/go-template/_common/text-template.md" %}} diff --git a/content/en/functions/go-template/continue.md b/content/en/functions/go-template/continue.md new file mode 100644 index 00000000000..c8030b8b7e9 --- /dev/null +++ b/content/en/functions/go-template/continue.md @@ -0,0 +1,34 @@ +--- +title: continue +description: Used with the range statement, stops the innermost iteration and continues to the next iteration. +categories: [] +keywords: [] +action: + aliases: [] + related: + - functions/go-template/break + - functions/go-template/range + returnType: + signatures: [continue] +--- + +This template code: + +```go-html-template +{{ $s := slice "foo" "bar" "baz" }} +{{ range $s }} + {{ if eq . "bar" }} + {{ continue }} + {{ end }} +

{{ . }}

+{{ end }} +``` + +Is rendered to: + +```html +

foo

+

baz

+``` + +{{% include "functions/go-template/_common/text-template.md" %}} diff --git a/content/en/functions/go-template/define.md b/content/en/functions/go-template/define.md new file mode 100644 index 00000000000..4d09c14f3ee --- /dev/null +++ b/content/en/functions/go-template/define.md @@ -0,0 +1,55 @@ +--- +title: define +description: Defines a template. +categories: [] +keywords: [] +action: + aliases: [] + related: + - functions/go-template/block + - functions/go-template/end + - functions/go-template/template + - functions/partials/Include + - functions/partials/IncludeCached + returnType: + signatures: [define NAME] +--- + +Use with the [`block`] statement: + +```go-html-template +{{ block "main" . }} + {{ print "default value if 'main' template is empty" }} +{{ end }} + +{{ define "main" }} +

{{ .Title }}

+ {{ .Content }} +{{ end }} +``` + +Use with the [`partial`] function: + +```go-html-template +{{ partial "inline/foo.html" (dict "answer" 42) }} + +{{ define "partials/inline/foo.html" }} + {{ printf "The answer is %v." .answer }} +{{ end }} +``` + +Use with the [`template`] function: + +```go-html-template +{{ template "foo" (dict "answer" 42) }} + +{{ define "foo" }} + {{ printf "The answer is %v." .answer }} +{{ end }} +``` + +[`block`]: /functions/go-template/block +[`template`]: /functions/go-template/block +[`partial`]: /functions/partials/include/ + +{{% include "functions/go-template/_common/text-template.md" %}} diff --git a/content/en/functions/go-template/else.md b/content/en/functions/go-template/else.md new file mode 100644 index 00000000000..ccb8b722cb3 --- /dev/null +++ b/content/en/functions/go-template/else.md @@ -0,0 +1,69 @@ +--- +title: else +description: Begins an alternate block for if, with, and range statements. +categories: [] +keywords: [] +action: + aliases: [] + related: + - functions/go-template/if + - functions/go-template/range + - functions/go-template/with + - functions/go-template/end + returnType: + signatures: [else VALUE] +--- + +Use with the [`if`] statement: + +```go-html-template +{{ $var := "foo" }} +{{ if $var }} + {{ $var }} → foo +{{ else }} + {{ print "var is falsy" }} +{{ end }} +``` + +Use with the [`with`] statement: + +```go-html-template +{{ $var := "foo" }} +{{ with $var }} + {{ . }} → foo +{{ else }} + {{ print "var is falsy" }} +{{ end }} +``` + +Use with the [`range`] statement: + +```go-html-template +{{ $var := slice 1 2 3 }} +{{ range $var }} + {{ . }} → 1 2 3 +{{ else }} + {{ print "var is falsy" }} +{{ end }} +``` + +Use `else if` to check multiple conditions. + +```go-html-template +{{ $var := 12 }} +{{ if eq $var 6 }} + {{ print "var is 6" }} +{{ else if eq $var 7 }} + {{ print "var is 7" }} +{{ else if eq $var 42 }} + {{ print "var is 42" }} +{{ else }} + {{ print "var is something else" }} +{{ end }} +``` + +{{% include "functions/go-template/_common/text-template.md" %}} + +[`if`]: /functions/go-template/if +[`with`]: /functions/go-template/with +[`range`]: /functions/go-template/range diff --git a/content/en/functions/go-template/end.md b/content/en/functions/go-template/end.md new file mode 100644 index 00000000000..07d004de510 --- /dev/null +++ b/content/en/functions/go-template/end.md @@ -0,0 +1,65 @@ +--- +title: end +description: Terminates if, with, range, block, and define statements. +categories: [] +keywords: [] +action: + aliases: [] + related: + - functions/go-template/block + - functions/go-template/define + - functions/go-template/if + - functions/go-template/range + - functions/go-template/with + returnType: + signatures: [end] +--- + +Use with the [`if`] statement: + +```go-html-template +{{ $var := "foo" }} +{{ if $var }} + {{ $var }} → foo +{{ end }} +``` + +Use with the [`with`] statement: + +```go-html-template +{{ $var := "foo" }} +{{ with $var }} + {{ . }} → foo +{{ end }} +``` + +Use with the [`range`] statement: + +```go-html-template +{{ $var := slice 1 2 3 }} +{{ range $var }} + {{ . }} → 1 2 3 +{{ end }} +``` + +Use with the [`block`] statement: + +```go-html-template +{{ block "main" . }}{{ end }} +``` + +Use with the [`define`] statement: + +```go-html-template +{{ define "main" }} + {{ print "this is the main section" }} +{{ end }} +``` + +{{% include "functions/go-template/_common/text-template.md" %}} + +[`block`]: /functions/go-template/block +[`define`]: /functions/go-template/define +[`if`]: /functions/go-template/if +[`range`]: /functions/go-template/range +[`with`]: /functions/go-template/with diff --git a/content/en/functions/go-template/if.md b/content/en/functions/go-template/if.md new file mode 100644 index 00000000000..e63c382e126 --- /dev/null +++ b/content/en/functions/go-template/if.md @@ -0,0 +1,54 @@ +--- +title: if +description: Executes the block if the expression is truthy. +categories: [] +keywords: [] +action: + aliases: [] + related: + - functions/go-template/with + - functions/go-template/else + - functions/go-template/end + - functions/collections/IsSet + returnType: + signatures: [if EXPR] +--- + +{{% include "functions/go-template/_common/truthy-falsy.md" %}} + +```go-html-template +{{ $var := "foo" }} +{{ if $var }} + {{ $var }} → foo +{{ end }} +``` + +Use with the [`else`] statement: + +```go-html-template +{{ $var := "foo" }} +{{ if $var }} + {{ $var }} → foo +{{ else }} + {{ print "var is falsy" }} +{{ end }} +``` + +Use `else if` to check multiple conditions. + +```go-html-template +{{ $var := 12 }} +{{ if eq $var 6 }} + {{ print "var is 6" }} +{{ else if eq $var 7 }} + {{ print "var is 7" }} +{{ else if eq $var 42 }} + {{ print "var is 42" }} +{{ else }} + {{ print "var is something else" }} +{{ end }} +``` + +{{% include "functions/go-template/_common/text-template.md" %}} + +[`else`]: /functions/go-template/else diff --git a/content/en/functions/go-template/len.md b/content/en/functions/go-template/len.md index b8be621e8bd..43f150a5ff3 100644 --- a/content/en/functions/go-template/len.md +++ b/content/en/functions/go-template/len.md @@ -1,26 +1,20 @@ --- title: len description: Returns the length of a string, slice, map, or collection. -categories: [functions] +categories: [] keywords: [] -menu: - docs: - parent: functions -function: +action: aliases: [] + related: + - functions/strings/Count + - functions/strings/CountRunes + - functions/strings/CountWords + - functions/strings/RuneCount returnType: int - signatures: [len INPUT] -relatedFunctions: - - len - - strings.Count - - strings.CountRunes - - strings.CountWords - - strings.RuneCount + signatures: [len VALUE] aliases: [/functions/len] --- -{{% readfile file="/functions/_common/go-template-functions.md" %}} - With a string: ```go-html-template @@ -53,3 +47,5 @@ You may also determine the number of pages in a collection with: ```go-html-template {{ site.RegularPages.Len }} → 42 ``` + +{{% include "functions/go-template/_common/text-template.md" %}} diff --git a/content/en/functions/go-template/not.md b/content/en/functions/go-template/not.md new file mode 100644 index 00000000000..4c7747c7bb7 --- /dev/null +++ b/content/en/functions/go-template/not.md @@ -0,0 +1,35 @@ +--- +title: not +description: Returns the boolean negation of its single argument. +categories: [] +keywords: [] +action: + aliases: [] + related: + - functions/go-template/and + - functions/go-template/or + returnType: bool + signatures: [not VALUE] +--- + +Unlike the `and` and `or` operators, the `not` operator always returns a boolean value. + +```go-html-template +{{ not true }} → false +{{ not false }} → true + +{{ not 1 }} → false +{{ not 0 }} → true + +{{ not "x" }} → false +{{ not "" }} → true +``` + +Use the `not` operator, twice in succession, to cast any value to a boolean value. For example: + +```go-html-template +{{ 42 | not | not }} → true +{{ "" | not | not }} → false +``` + +{{% include "functions/go-template/_common/text-template.md" %}} diff --git a/content/en/functions/go-template/or.md b/content/en/functions/go-template/or.md new file mode 100644 index 00000000000..0d861960869 --- /dev/null +++ b/content/en/functions/go-template/or.md @@ -0,0 +1,26 @@ +--- +title: or +description: Returns the first truthy argument. If all arguments are falsy, returns the last argument. +categories: [] +keywords: [] +action: + aliases: [] + related: + - functions/go-template/and + - functions/go-template/not + returnType: any + signatures: [or VALUE...] +--- + +{{% include "functions/go-template/_common/truthy-falsy.md" %}} + +```go-html-template +{{ or 0 1 2 }} → 1 +{{ or false "a" 1 }} → a +{{ or 0 true "a" }} → true + +{{ or false "" 0 }} → 0 +{{ or 0 "" false }} → false +``` + +{{% include "functions/go-template/_common/text-template.md" %}} diff --git a/content/en/functions/go-template/range.md b/content/en/functions/go-template/range.md index cf01633b4d5..4c2c2e1c4e1 100644 --- a/content/en/functions/go-template/range.md +++ b/content/en/functions/go-template/range.md @@ -1,36 +1,95 @@ --- title: range -description: Iterates over slices, maps, and page collections. -categories: [functions] +description: Iterates over a non-empty collection, binds context (the dot) to successive elements, and executes the block. +categories: [] keywords: [] -menu: - docs: - parent: functions -function: +action: aliases: [] + related: + - functions/go-template/break + - functions/go-template/continue + - functions/go-template/else + - functions/go-template/end returnType: signatures: [range COLLECTION] -relatedFunctions: - - with - - range aliases: [/functions/range] toc: true --- -{{% readfile file="/functions/_common/go-template-functions.md" %}} +{{% include "functions/go-template/_common/truthy-falsy.md" %}} -## Slices +```go-html-template +{{ $s := slice "foo" "bar" "baz" }} +{{ range $var }} + {{ . }} → foo "bar" "baz" +{{ end }} +``` -Template: +Use with the [`else`] statement: ```go-html-template {{ $s := slice "foo" "bar" "baz" }} {{ range $s }}

{{ . }}

+{{ else }} +

The collection is empty

+{{ end }} +``` + +Within a range block: + +- Use the [`continue`] statement to stop the innermost iteration and continue to the next iteration +- Use the [`break`] statement to stop the innermost iteration and bypass all remaining iterations + +## Understanding context + +At the top of a page template, the [context] (the dot) is a `Page` object. Within the `range` block, the context is bound to each successive element. + +With this contrived example that uses the [`seq`] function to generate a slice of integers: + +```go-html-template +{{ range seq 3 }} + {{ .Title }} +{{ end }} +``` + +Hugo will throw an error: + + can't evaluate field Title in type int + +The error occurs because we are trying to use the `.Title` method on an integer instead of a `Page` object. Within the `range` block, if we want to render the page title, we need to get the context passed into the template. + +{{% note %}} +Use the `$` to get the context passed into the template. +{{% /note %}} + +This template will render the page title three times: + +```go-html-template +{{ range seq 3 }} + {{ $.Title }} {{ end }} ``` -Result: +{{% note %}} +Gaining a thorough understanding of context is critical for anyone writing template code. +{{% /note %}} + +[`seq`]: functions/collections/seq/ +[context]: /getting-started/glossary/#context + +## Array or slice of scalars + +This template code: + +```go-html-template +{{ $s := slice "foo" "bar" "baz" }} +{{ range $s }} +

{{ . }}

+{{ end }} +``` + +Is rendered to: ```html

foo

@@ -38,7 +97,7 @@ Result:

baz

``` -Template: +This template code: ```go-html-template {{ $s := slice "foo" "bar" "baz" }} @@ -47,7 +106,7 @@ Template: {{ end }} ``` -Result: +Is rendered to: ```html

foo

@@ -55,7 +114,7 @@ Result:

baz

``` -Template: +This template code: ```go-html-template {{ $s := slice "foo" "bar" "baz" }} @@ -64,7 +123,7 @@ Template: {{ end }} ``` -Result: +Is rendered to: ```html

0: foo

@@ -72,9 +131,9 @@ Result:

2: baz

``` -## Maps +## Array or slice of maps -Template: +This template code: ```go-html-template {{ $m := slice @@ -87,7 +146,7 @@ Template: {{ end }} ``` -Result: +Is rendered to: ```html

John is 30

@@ -95,17 +154,17 @@ Result:

Joey is 24

``` -## Page collections +## Array or slice of pages -Template: +This template code: ```go-html-template {{ range where site.RegularPages "Type" "articles" }} -

{{ .LinkTitle }}

+

{{ .Title }}

{{ end }} ``` -Result: +Is rendered to: ```html

Article 3

@@ -113,47 +172,28 @@ Result:

Article 1

``` -## Break - -Use the `break` statement to stop the innermost iteration and bypass all remaining iterations. +## Maps -Template: +This template code: ```go-html-template -{{ $s := slice "foo" "bar" "baz" }} -{{ range $s }} - {{ if eq . "bar" }} - {{ break }} - {{ end }} -

{{ . }}

+{{ $m := dict "name" "John" "age" 30 }} +{{ range $k, $v := $m }} +

key = {{ $k }} value = {{ $v }}

{{ end }} ``` -Result: - -```html -

foo

-``` - -## Continue - -Use the `continue` statement to stop the innermost iteration and continue to the next iteration. - -Template: +Is rendered to: ```go-html-template -{{ $s := slice "foo" "bar" "baz" }} -{{ range $s }} - {{ if eq . "bar" }} - {{ continue }} - {{ end }} -

{{ . }}

-{{ end }} +

key = age value = 30

+

key = name value = John

``` -Result: +Unlike ranging over an array or slice, Hugo sorts by key when ranging over a map. -```html -

foo

-

baz

-``` +{{% include "functions/go-template/_common/text-template.md" %}} + +[`else`]: /functions/go-template/else +[`break`]: /functions/go-template/break +[`continue`]: /functions/go-template/continue diff --git a/content/en/functions/go-template/template.md b/content/en/functions/go-template/template.md new file mode 100644 index 00000000000..c089010cac0 --- /dev/null +++ b/content/en/functions/go-template/template.md @@ -0,0 +1,49 @@ +--- +title: template +description: Executes the given template, optionally passing context. +categories: [] +keywords: [] +action: + aliases: [] + related: + - functions/go-template/define + - functions/partials/Include + - functions/partials/IncludeCached + returnType: + signatures: ['template NAME [CONTEXT]'] +--- + +Use the `template` function to execute [internal templates]. For example: + +```go-html-template +{{ range (.Paginate .Pages).Pages }} +

{{ .Title }}

+{{ end }} +{{ template "_internal/pagination.html" . }} +``` + +You can also use the `template` function to execute a defined template: + +```go-html-template +{{ template "foo" (dict "answer" 42) }} + +{{ define "foo" }} + {{ printf "The answer is %v." .answer }} +{{ end }} +``` + +The example above can be rewritten using an [inline partial] template: + +```go-html-template +{{ partial "inline/foo.html" (dict "answer" 42) }} + +{{ define "partials/inline/foo.html" }} + {{ printf "The answer is %v." .answer }} +{{ end }} +``` + +{{% include "functions/go-template/_common/text-template.md" %}} + +[`partial`]: /functions/partials/include/ +[inline partial]: /templates/partials/#inline-partials +[internal templates]: /templates/internal diff --git a/content/en/functions/go-template/urlquery.md b/content/en/functions/go-template/urlquery.md index cbbfdfa7d3e..946828f5637 100644 --- a/content/en/functions/go-template/urlquery.md +++ b/content/en/functions/go-template/urlquery.md @@ -1,23 +1,17 @@ --- title: urlquery description: Returns the escaped value of the textual representation of its arguments in a form suitable for embedding in a URL query. -categories: [functions] +categories: [] keywords: [] -menu: - docs: - parent: functions -function: +action: aliases: [] + related: + - functions/collections/Querify returnType: string - signatures: ['urlquery INPUT [INPUT]...'] -relatedFunctions: - - collections.Querify - - urlquery + signatures: ['urlquery VALUE [VALUE...]'] aliases: [/functions/urlquery] --- -{{% readfile file="/functions/_common/go-template-functions.md" %}} - This template code: ```go-html-template @@ -30,3 +24,5 @@ Is rendered to: ```html Link ``` + +{{% include "functions/go-template/_common/text-template.md" %}} diff --git a/content/en/functions/go-template/with.md b/content/en/functions/go-template/with.md index 06ca38150f7..9245e563f18 100644 --- a/content/en/functions/go-template/with.md +++ b/content/en/functions/go-template/with.md @@ -1,35 +1,77 @@ --- title: with -description: Rebinds the context (`.`) within its scope and skips the block if the variable is absent or empty. -categories: [functions] +description: Binds context (the dot) to the expression and executes the block if expression is truthy. +categories: [] keywords: [] -menu: - docs: - parent: functions -function: +action: aliases: [] - returnType: any - signatures: [with PIPELINE] -relatedFunctions: - - with - - range + related: + - functions/go-template/if + - functions/go-template/else + - functions/go-template/end + - functions/collections/IsSet + returnType: + signatures: [with EXPR] aliases: [/functions/with] +toc: true --- -{{% readfile file="/functions/_common/go-template-functions.md" %}} +{{% include "functions/go-template/_common/truthy-falsy.md" %}} -An alternative way of writing an `if` statement and then referencing the same value is to use `with` instead. `with` rebinds the context (`.`) within its scope and skips the block if the variable is absent, unset or empty. +```go-html-template +{{ $var := "foo" }} +{{ with $var }} + {{ . }} → foo +{{ end }} +``` -The set of *empty* values is defined by [the Go templates package](https://golang.org/pkg/text/template/). Empty values include `false`, the number zero, and the empty string. +Use with the [`else`] statement: -If you want to render a block if an index or key is present in a slice, array, channel or map, regardless of whether the value is empty, you should use [`isset`](/functions/collections/isset) instead. +```go-html-template +{{ $var := "foo" }} +{{ with $var }} + {{ . }} → foo +{{ else }} + {{ print "var is falsy" }} +{{ end }} +``` -The following example checks for a [user-defined site variable](/variables/site/) called `twitteruser`. If the key-value is not set, the following will render nothing: +## Understanding context -{{< code file="layouts/partials/twitter.html" >}} -{{ with .Site.Params.twitteruser }}{{ end }} -{{< /code >}} +At the top of a page template, the [context] (the dot) is a `Page` object. Inside of the `with` block, the context is bound to the value passed to the `with` statement. + +With this contrived example: + +```go-html-template +{{ with 42 }} + {{ .Title }} +{{ end }} +``` + +Hugo will throw an error: + + can't evaluate field Title in type int + +The error occurs because we are trying to use the `.Title` method on an integer instead of a `Page` object. Inside of the `with` block, if we want to render the page title, we need to get the context passed into the template. + +{{% note %}} +Use the `$` to get the context passed into the template. +{{% /note %}} + +This template will render the page title as desired: + +```go-html-template +{{ with 42 }} + {{ $.Title }} +{{ end }} +``` + +{{% note %}} +Gaining a thorough understanding of context is critical for anyone writing template code. +{{% /note %}} + +[context]: /getting-started/glossary/#context + +{{% include "functions/go-template/_common/text-template.md" %}} + +[`else`]: /functions/go-template/else diff --git a/content/en/functions/images/Brightness.md b/content/en/functions/images/Brightness.md new file mode 100644 index 00000000000..0001bcba8b0 --- /dev/null +++ b/content/en/functions/images/Brightness.md @@ -0,0 +1,36 @@ +--- +title: images.Brightness +description: Returns an image filter that changes the brightness of an image. +categories: [] +keywords: [] +action: + aliases: [] + related: + - functions/images/Filter + - methods/resource/Filter + returnType: images.filter + signatures: [images.Brightness PERCENTAGE] +toc: true +--- + +The percentage must be in the range [-100, 100] where 0 has no effect. A value of `-100` produces a solid black image, and a value of `100` produces a solid white image. + +## Usage + +Create the image filter: + +```go-html-template +{{ $filter := images.Brightness 12 }} +``` + +{{% include "functions/images/_common/apply-image-filter.md" %}} + +## Example + +{{< img + src="images/examples/zion-national-park.jpg" + alt="Zion National Park" + filter="Brightness" + filterArgs="12" + example=true +>}} diff --git a/content/en/functions/images/ColorBalance.md b/content/en/functions/images/ColorBalance.md new file mode 100644 index 00000000000..29829f9e653 --- /dev/null +++ b/content/en/functions/images/ColorBalance.md @@ -0,0 +1,36 @@ +--- +title: images.ColorBalance +description: Returns an image filter that changes the color balance of an image. +categories: [] +keywords: [] +action: + aliases: [] + related: + - functions/images/Filter + - methods/resource/Filter + returnType: images.filter + signatures: [images.ColorBalance PCTRED PCTGREEN PCTBLUE] +toc: true +--- + +The percentage for each channel (red, green, blue) must be in the range [-100, 500]. + +## Usage + +Create the filter: + +```go-html-template +{{ $filter := images.ColorBalance -10 10 50 }} +``` + +{{% include "functions/images/_common/apply-image-filter.md" %}} + +## Example + +{{< img + src="images/examples/zion-national-park.jpg" + alt="Zion National Park" + filter="ColorBalance" + filterArgs="-10,10,50" + example=true +>}} diff --git a/content/en/functions/images/Colorize.md b/content/en/functions/images/Colorize.md new file mode 100644 index 00000000000..c974103b9fc --- /dev/null +++ b/content/en/functions/images/Colorize.md @@ -0,0 +1,40 @@ +--- +title: images.Colorize +description: Returns an image filter that produces a colorized version of an image. +categories: [] +keywords: [] +action: + aliases: [] + related: + - functions/images/Filter + - methods/resource/Filter + returnType: images.filter + signatures: [images.Colorize HUE SATURATION PERCENTAGE] +toc: true +--- + +The hue is the angle on the color wheel, typically in the range [0, 360]. + +The saturation must be in the range [0, 100]. + +The percentage specifies the strength of the effect, and must be in the range [0, 100]. + +## Usage + +Create the filter: + +```go-html-template +{{ $filter := images.Colorize 180 50 20 }} +``` + +{{% include "functions/images/_common/apply-image-filter.md" %}} + +## Example + +{{< img + src="images/examples/zion-national-park.jpg" + alt="Zion National Park" + filter="Colorize" + filterArgs="180,50,20" + example=true +>}} diff --git a/content/en/functions/images/Config.md b/content/en/functions/images/Config.md new file mode 100644 index 00000000000..fafb5b02401 --- /dev/null +++ b/content/en/functions/images/Config.md @@ -0,0 +1,31 @@ +--- +title: images.Config +description: Returns an image.Config structure from the image at the specified path, relative to the working directory. +categories: [] +keywords: [] +action: + aliases: [] + related: [] + returnType: image.Config + signatures: [images.Config PATH] +aliases: [/functions/imageconfig] +--- + +See [image processing] for an overview of Hugo's image pipeline. + +[image processing]: /content-management/image-processing/ + +```go-html-template +{{ $ic := images.Config "/static/images/a.jpg" }} + +{{ $ic.Width }} → 600 (int) +{{ $ic.Height }} → 400 (int) +``` + +Supported image formats include GIF, JPEG, PNG, TIFF, and WebP. + +{{% note %}} +This is a legacy function, superseded by the `.Width` and `.Height` methods for global, page, and remote resources. See the [image processing] section for details. + +[image processing]: /content-management/image-processing +{{% /note %}} diff --git a/content/en/functions/images/Contrast.md b/content/en/functions/images/Contrast.md new file mode 100644 index 00000000000..532ae8c9cbb --- /dev/null +++ b/content/en/functions/images/Contrast.md @@ -0,0 +1,36 @@ +--- +title: images.Contrast +description: Returns an image filter that changes the contrast of an image. +categories: [] +keywords: [] +action: + aliases: [] + related: + - functions/images/Filter + - methods/resource/Filter + returnType: images.filter + signatures: [images.Contrast PERCENTAGE] +toc: true +--- + +The percentage must be in the range [-100, 100] where 0 has no effect. A value of `-100` produces a solid grey image, and a value of `100` produces an over-contrasted image. + +## Usage + +Create the filter: + +```go-html-template +{{ $filter := images.Contrast -20 }} +``` + +{{% include "functions/images/_common/apply-image-filter.md" %}} + +## Example + +{{< img + src="images/examples/zion-national-park.jpg" + alt="Zion National Park" + filter="Contrast" + filterArgs="-20" + example=true +>}} diff --git a/content/en/functions/images/Filter.md b/content/en/functions/images/Filter.md new file mode 100644 index 00000000000..450a6481421 --- /dev/null +++ b/content/en/functions/images/Filter.md @@ -0,0 +1,67 @@ +--- +title: images.Filter +description: Applies one or more image filters to the given image resource. +categories: [] +keywords: [] +action: + aliases: [] + related: + - methods/resource/Filter + returnType: images.ImageResource + signatures: [images.Filter FILTERS... IMAGE] +toc: true +--- + +Apply one or more [image filters](#image-filters) to the given image. + +To apply a single filter: + +```go-html-template +{{ with resources.Get "images/original.jpg" }} + {{ with images.Filter images.Grayscale . }} + + {{ end }} +{{ end }} +``` + +To apply two or more filters, executing from left to right: + +```go-html-template +{{ $filters := slice + images.Grayscale + (images.GaussianBlur 8) +}} +{{ with resources.Get "images/original.jpg" }} + {{ with images.Filter $filters . }} + + {{ end }} +{{ end }} +``` + +You can also apply image filters using the [`Filter`] method on a `Resource` object. + +[`Filter`]: /methods/resource/filter + +## Example + +```go-html-template +{{ with resources.Get "images/original.jpg" }} + {{ with images.Filter images.Grayscale . }} + + {{ end }} +{{ end }} +``` + +{{< img + src="images/examples/zion-national-park.jpg" + alt="Zion National Park" + filter="Grayscale" + filterArgs="" + example=true +>}} + +## Image filters + +Use any of these filters with the `images.Filter` function, or with the `Filter` method on a `Resource` object. + +{{< list-pages-in-section path=/functions/images filter=functions_images_no_filters filterType=exclude >}} diff --git a/content/en/functions/images/Gamma.md b/content/en/functions/images/Gamma.md new file mode 100644 index 00000000000..affbdcfa8fa --- /dev/null +++ b/content/en/functions/images/Gamma.md @@ -0,0 +1,36 @@ +--- +title: images.Gamma +description: Returns an image filter that performs gamma correction on an image. +categories: [] +keywords: [] +action: + aliases: [] + related: + - functions/images/Filter + - methods/resource/Filter + returnType: images.filter + signatures: [images.Gamma GAMMA] +toc: true +--- + +The gamma value must be positive. A value greater than 1 lightens the image, while a value less than 1 darkens the image. The filter has no effect when the gamma value is 1. + +## Usage + +Create the filter: + +```go-html-template +{{ $filter := images.Gamma 1.667 }} +``` + +{{% include "functions/images/_common/apply-image-filter.md" %}} + +## Example + +{{< img + src="images/examples/zion-national-park.jpg" + alt="Zion National Park" + filter="Gamma" + filterArgs="1.667" + example=true +>}} diff --git a/content/en/functions/images/GaussianBlur.md b/content/en/functions/images/GaussianBlur.md new file mode 100644 index 00000000000..e2f49a847a1 --- /dev/null +++ b/content/en/functions/images/GaussianBlur.md @@ -0,0 +1,36 @@ +--- +title: images.GaussianBlur +description: Returns an image filter that applies a gaussian blur to an image. +categories: [] +keywords: [] +action: + aliases: [] + related: + - functions/images/Filter + - methods/resource/Filter + returnType: images.filter + signatures: [images.GaussianBlur SIGMA] +toc: true +--- + +The sigma value must be positive, and indicates how much the image will be blurred. The blur-affected radius is approximately 3 times the sigma value. + +## Usage + +Create the filter: + +```go-html-template +{{ $filter := images.GaussianBlur 5 }} +``` + +{{% include "functions/images/_common/apply-image-filter.md" %}} + +## Example + +{{< img + src="images/examples/zion-national-park.jpg" + alt="Zion National Park" + filter="GaussianBlur" + filterArgs="5" + example=true +>}} diff --git a/content/en/functions/images/Grayscale.md b/content/en/functions/images/Grayscale.md new file mode 100644 index 00000000000..d8a89b7f265 --- /dev/null +++ b/content/en/functions/images/Grayscale.md @@ -0,0 +1,34 @@ +--- +title: images.Grayscale +description: Returns an image filter that produces a grayscale version of an image. +categories: [] +keywords: [] +action: + aliases: [] + related: + - functions/images/Filter + - methods/resource/Filter + returnType: images.filter + signatures: [images.Grayscale] +toc: true +--- + +## Usage + +Create the filter: + +```go-html-template +{{ $filter := images.Grayscale }} +``` + +{{% include "functions/images/_common/apply-image-filter.md" %}} + +## Example + +{{< img + src="images/examples/zion-national-park.jpg" + alt="Zion National Park" + filter="Grayscale" + filterArgs="" + example=true +>}} diff --git a/content/en/functions/images/Hue.md b/content/en/functions/images/Hue.md new file mode 100644 index 00000000000..6eafac43788 --- /dev/null +++ b/content/en/functions/images/Hue.md @@ -0,0 +1,36 @@ +--- +title: images.Hue +description: Returns an image filter that rotates the hue of an image. +categories: [] +keywords: [] +action: + aliases: [] + related: + - functions/images/Filter + - methods/resource/Filter + returnType: images.filter + signatures: [images.Hue SHIFT] +toc: true +--- + +The hue angle shift is typically in the range [-180, 180] where 0 has no effect. + +## Usage + +Create the filter: + +```go-html-template +{{ $filter := images.Hue -15 }} +``` + +{{% include "functions/images/_common/apply-image-filter.md" %}} + +## Example + +{{< img + src="images/examples/zion-national-park.jpg" + alt="Zion National Park" + filter="Hue" + filterArgs="-15" + example=true +>}} diff --git a/content/en/functions/images/Invert.md b/content/en/functions/images/Invert.md new file mode 100644 index 00000000000..1ee85e5145c --- /dev/null +++ b/content/en/functions/images/Invert.md @@ -0,0 +1,34 @@ +--- +title: images.Invert +description: Returns an image filter that negates the colors of an image. +categories: [] +keywords: [] +action: + aliases: [] + related: + - functions/images/Filter + - methods/resource/Filter + returnType: images.filter + signatures: [images.Invert] +toc: true +--- + +## Usage + +Create the filter: + +```go-html-template +{{ $filter := images.Invert }} +``` + +{{% include "functions/images/_common/apply-image-filter.md" %}} + +## Example + +{{< img + src="images/examples/zion-national-park.jpg" + alt="Zion National Park" + filter="Invert" + filterArgs="" + example=true +>}} diff --git a/content/en/functions/images/Opacity.md b/content/en/functions/images/Opacity.md new file mode 100644 index 00000000000..c3b09efc42c --- /dev/null +++ b/content/en/functions/images/Opacity.md @@ -0,0 +1,50 @@ +--- +title: images.Opacity +description: Returns an image filter that changes the opacity of an image. +categories: [] +keywords: [] +action: + aliases: [] + related: + - functions/images/Filter + - methods/resource/Filter + returnType: images.filter + signatures: [images.Opacity OPACITY] +toc: true +--- + +The opacity value must be in the range [0, 1]. A value of `0` produces a transparent image, and a value of `1` produces an opaque image (no transparency). + +## Usage + +Create the filter: + +```go-html-template +{{ $filter := images.Opacity 0.65 }} +``` + +{{% include "functions/images/_common/apply-image-filter.md" %}} + +The `images.Opacity` filter is most useful for target formats such as PNG and WebP that support transparency. If the source image does not support transparency, combine this filter with the `images.Process` filter: + +```go-html-template +{{ with resources.Get "images/original.jpg" }} + {{ $filters := slice + (images.Opacity 0.65) + (images.Process "png") + }} + {{ with . | images.Filter $filters }} + + {{ end }} +{{ end }} +``` + +## Example + +{{< img + src="images/examples/zion-national-park.jpg" + alt="Zion National Park" + filter="Opacity" + filterArgs="0.65" + example=true +>}} diff --git a/content/en/functions/images/Overlay.md b/content/en/functions/images/Overlay.md new file mode 100644 index 00000000000..f04e3668b7c --- /dev/null +++ b/content/en/functions/images/Overlay.md @@ -0,0 +1,54 @@ +--- +title: images.Overlay +description: Returns an image filter that overlays the source image at the given coordinates. +categories: [] +keywords: [] +action: + aliases: [] + related: + - functions/images/Filter + - methods/resource/Filter + returnType: images.filter + signatures: [images.Overlay RESOURCE X Y] +toc: true +--- + +The coordinates are relative to the upper left corner. + +## Usage + +Capture the overlay image as a resource: + +```go-html-template +{{ $overlay := "" }} +{{ $path := "images/logo.png" }} +{{ with resources.Get $path }} + {{ $overlay = . }} +{{ else }} + {{ errorf "Unable to get resource %q" $path }} +{{ end }} +``` + +The overlay image can be a [global resource], a [page resource], or a [remote resource]. + +[global resource]: /getting-started/glossary/#global-resource +[page resource]: /getting-started/glossary/#page-resource +[remote resource]: /getting-started/glossary/#remote-resource + +Create the filter: + +```go-html-template +{{ $filter := images.Overlay $overlay 20 20 }} +``` + +{{% include "functions/images/_common/apply-image-filter.md" %}} + +## Example + +{{< img + src="images/examples/zion-national-park.jpg" + alt="Zion National Park" + filter="Overlay" + filterArgs="images/logos/logo-64x64.png,20,20" + example=true +>}} diff --git a/content/en/functions/images/Padding.md b/content/en/functions/images/Padding.md new file mode 100644 index 00000000000..1362bc78eb0 --- /dev/null +++ b/content/en/functions/images/Padding.md @@ -0,0 +1,73 @@ +--- +title: images.Padding +description: Returns an image filter that resizes the image canvas without resizing the image. +categories: [] +keywords: [] +action: + aliases: [] + related: + - functions/images/Filter + - methods/resource/Filter + returnType: images.filter + signatures: ['images.Padding V1 [V2] [V3] [V4] [COLOR]'] +toc: true +--- + +The last argument is the canvas color, expressed as an RGB or RGBA [hexadecimal color]. The default value is `ffffffff` (opaque white). The preceding arguments are the padding values, in pixels, using the CSS [shorthand property] syntax. Negative padding values will crop the image. + +[hexadecimal color]: https://developer.mozilla.org/en-US/docs/Web/CSS/hex-color +[shorthand property]: https://developer.mozilla.org/en-US/docs/Web/CSS/Shorthand_properties#edges_of_a_box + +## Usage + +Create the filter: + +```go-html-template +{{ $filter := images.Padding 20 40 "#976941" }} +``` + +{{% include "functions/images/_common/apply-image-filter.md" %}} + +Combine with the [`Colors`] method to create a border with one of the image's most dominant colors: + +[`Colors`]: /methods/resource/colors + +```go-html-template +{{ with resources.Get "images/original.jpg" }} + {{ $filter := images.Padding 20 40 (index .Colors 2) }} + {{ with . | images.Filter $filter }} + + {{ end }} +{{ end }} +``` + +## Example + +{{< img + src="images/examples/zion-national-park.jpg" + alt="Zion National Park" + filter="Padding" + filterArgs="20,40,20,40,#976941" + example=true +>}} + +## Other recipes + +This example resizes an image to 300px wide, converts it to the WebP format, adds 20px vertical padding and 50px horizontal padding, then sets the canvas color to dark green with 33% opacity. + +Conversion to WebP is required to support transparency. PNG and WebP images have an alpha channel; JPEG and GIF do not. + +```go-html-template +{{ $img := resources.Get "images/a.jpg" }} +{{ $filters := slice + (images.Process "resize 300x webp") + (images.Padding 20 50 "#0705") +}} +{{ $img = $img.Filter $filters }} +``` + +To add a 2px gray border to an image: + +```go-html-template +{{ $img = $img.Filter (images.Padding 2 "#777") }} +``` diff --git a/content/en/functions/images/Pixelate.md b/content/en/functions/images/Pixelate.md new file mode 100644 index 00000000000..2016877ed85 --- /dev/null +++ b/content/en/functions/images/Pixelate.md @@ -0,0 +1,34 @@ +--- +title: images.Pixelate +description: Returns an image filter that applies a pixelation effect to an image. +categories: [] +keywords: [] +action: + aliases: [] + related: + - functions/images/Filter + - methods/resource/Filter + returnType: images.filter + signatures: [images.Pixelate SIZE] +toc: true +--- + +## Usage + +Create the filter: + +```go-html-template +{{ $filter := images.Pixelate 4 }} +``` + +{{% include "functions/images/_common/apply-image-filter.md" %}} + +## Example + +{{< img + src="images/examples/zion-national-park.jpg" + alt="Zion National Park" + filter="Pixelate" + filterArgs="4" + example=true +>}} diff --git a/content/en/functions/images/Process.md b/content/en/functions/images/Process.md new file mode 100644 index 00000000000..f3cf494be67 --- /dev/null +++ b/content/en/functions/images/Process.md @@ -0,0 +1,110 @@ +--- +title: images.Process +description: Returns an image filter that processes the given image using the given specification. +categories: [] +keywords: [] +action: + aliases: [] + related: + - functions/images/Filter + - methods/resource/Filter + - methods/resource/Process + returnType: images.filter + signatures: [images.Process SPEC] +toc: true +--- + +This filter has the same options as the [`Process`] method on a `Resource` object, but using it as a filter may be more effective if you need to apply multiple filters to an image. + +[`Process`]: /methods/resource/process + +The process specification is a space-delimited, case-insensitive list of one or more of the following in any sequence: + +action +: Specify zero or one of `resize`, `fit`, `fill`, or `crop`. If you specify an action you must also provide dimensions. See [details](content-management/image-processing/#image-processing-methods). + +```go-html-template +{{ $filter := images.Process "resize 300x" }} +``` + +dimensions +: Required if you specify an action. Provide width _or_ height when using `resize`, else provide both width _and_ height. See [details](/content-management/image-processing/#dimensions). + +```go-html-template +{{ $filter := images.Process "crop 200x200" }} +``` + +anchor +: Use with the `crop` or `fill` action. Specify zero or one of `TopLeft`, `Top`, `TopRight`, `Left`, `Center`, `Right`, `BottomLeft`, `Bottom`, `BottomRight`, or `Smart`. Default is `Smart`. See [details](/content-management/image-processing/#anchor). + +```go-html-template +{{ $filter := images.Process "crop 200x200 center" }} +``` + +rotation +: Typically specify zero or one of `r90`, `r180`, or `r270`. Also supports arbitrary rotation angles. See [details](/content-management/image-processing/#rotation). + +```go-html-template +{{ $filter := images.Process "r90" }} +{{ $filter := images.Process "crop 200x200 center r90" }} +``` + +target format +: Specify zero or one of `gif`, `jpeg`, `png`, `tiff`, or `webp`. See [details](/content-management/image-processing/#target-format). + +```go-html-template +{{ $filter := images.Process "webp" }} +{{ $filter := images.Process "crop 200x200 center r90 webp" }} +``` + +quality +: Applicable to JPEG and WebP images. Optionally specify `qN` where `N` is an integer in the range [0, 100]. Default is `75`. See [details](/content-management/image-processing/#quality). + +```go-html-template +{{ $filter := images.Process "q50" }} +{{ $filter := images.Process "crop 200x200 center r90 webp q50" }} +``` + +hint +: Applicable to WebP images. Specify zero or one of `drawing`, `icon`, `photo`, `picture`, or `text`. Default is `photo`. See [details](/content-management/image-processing/#hint). + +```go-html-template +{{ $filter := images.Process "webp" "icon" }} +{{ $filter := images.Process "crop 200x200 center r90 webp q50 icon" }} +``` + +background color +: When converting a PNG or WebP with transparency to a format that does not support transparency, optionally specify a background color using a 3-digit or a 6-digit hexadecimal color code. Default is `#ffffff` (white). See [details](/content-management/image-processing/#background-color). + +```go-html-template +{{ $filter := images.Process "jpeg #000" }} +{{ $filter := images.Process "crop 200x200 center r90 q50 jpeg #000" }} +``` + +resampling filter +: Typically specify zero or one of `Box`, `Lanczos`, `CatmullRom`, `MitchellNetravali`, `Linear`, or `NearestNeighbor`. Other resampling filters are available. See [details](/content-management/image-processing/#resampling-filter). + +```go-html-template +{{ $filter := images.Process "resize 300x lanczos" }} +{{ $filter := images.Process "resize 300x r90 q50 jpeg #000 lanczos" }} +``` + +## Usage + +Create a filter: + +```go-html-template +{{ $filter := images.Process "resize 256x q40 webp" }} +``` + +{{% include "functions/images/_common/apply-image-filter.md" %}} + +## Example + +{{< img + src="images/examples/zion-national-park.jpg" + alt="Zion National Park" + filter="Process" + filterArgs="resize 256x q40 webp" + example=true +>}} diff --git a/content/en/functions/images/Saturation.md b/content/en/functions/images/Saturation.md new file mode 100644 index 00000000000..118bd021303 --- /dev/null +++ b/content/en/functions/images/Saturation.md @@ -0,0 +1,36 @@ +--- +title: images.Saturation +description: Returns an image filter that changes the saturation of an image. +categories: [] +keywords: [] +action: + aliases: [] + related: + - functions/images/Filter + - methods/resource/Filter + returnType: images.filter + signatures: [images.Saturation PERCENTAGE] +toc: true +--- + +The percentage must be in the range [-100, 500] where 0 has no effect. + +## Usage + +Create the filter: + +```go-html-template +{{ $filter := images.Saturation 65 }} +``` + +{{% include "functions/images/_common/apply-image-filter.md" %}} + +## Example + +{{< img + src="images/examples/zion-national-park.jpg" + alt="Zion National Park" + filter="Saturation" + filterArgs="65" + example=true +>}} diff --git a/content/en/functions/images/Sepia.md b/content/en/functions/images/Sepia.md new file mode 100644 index 00000000000..9f0b7adfb94 --- /dev/null +++ b/content/en/functions/images/Sepia.md @@ -0,0 +1,36 @@ +--- +title: images.Sepia +description: Returns an image filter that produces a sepia-toned version of an image. +categories: [] +keywords: [] +action: + aliases: [] + related: + - functions/images/Filter + - methods/resource/Filter + returnType: images.filter + signatures: [images.Sepia PERCENTAGE] +toc: true +--- + +The percentage must be in the range [0, 100] where 0 has no effect. + +## Usage + +Create the filter: + +```go-html-template +{{ $filter := images.Sepia 75 }} +``` + +{{% include "functions/images/_common/apply-image-filter.md" %}} + +## Example + +{{< img + src="images/examples/zion-national-park.jpg" + alt="Zion National Park" + filter="Sepia" + filterArgs="75" + example=true +>}} diff --git a/content/en/functions/images/Sigmoid.md b/content/en/functions/images/Sigmoid.md new file mode 100644 index 00000000000..32765f9238c --- /dev/null +++ b/content/en/functions/images/Sigmoid.md @@ -0,0 +1,40 @@ +--- +title: images.Sigmoid +description: Returns an image filter that changes the contrast of an image using a sigmoidal function. +categories: [] +keywords: [] +action: + aliases: [] + related: + - functions/images/Filter + - methods/resource/Filter + returnType: images.filter + signatures: [images.Sigmoid MIDPOINT FACTOR] +toc: true +--- + +This is a non-linear contrast change useful for photo adjustments; it preserves highlight and shadow detail. + +The midpoint is the midpoint of contrast. It must be in the range [0, 1], typically 0.5. + +The factor indicates how much to increase or decrease the contrast, typically in the range [-10, 10] where 0 has no effect. A positive value increases contrast, while a negative value decrease contrast. + +## Usage + +Create the filter: + +```go-html-template +{{ $filter := images.Sigmoid 0.6 -4 }} +``` + +{{% include "functions/images/_common/apply-image-filter.md" %}} + +## Example + +{{< img + src="images/examples/zion-national-park.jpg" + alt="Zion National Park" + filter="Sigmoid" + filterArgs="0.6,-4" + example=true +>}} diff --git a/content/en/functions/images/Text.md b/content/en/functions/images/Text.md new file mode 100644 index 00000000000..0c1e74bce5e --- /dev/null +++ b/content/en/functions/images/Text.md @@ -0,0 +1,95 @@ +--- +title: images.Text +description: Returns an image filter that adds text to an image. +categories: [] +keywords: [] +action: + aliases: [] + related: + - functions/images/Filter + - methods/resource/Filter + returnType: images.filter + signatures: ['images.Text TEXT [OPTIONS]'] +toc: true +--- + +## Options + +Although none of the options are required, at a minimum you will want to set the `size` to be some reasonable percentage of the image height. + +color +: (`string`) The font color, either a 3-digit or 6-digit hexadecimal color code. Default is `#ffffff` (white). + +font +: (`resource.Resource`) The font can be a [global resource], a [page resource], or a [remote resource]. Default is the "Go Regular" TrueType font. + +linespacing +: (`int`) The number of pixels between each line. For a line height of 1.4, set the `linespacing` to 0.4 multiplied by the `size`. Default is `2`. + +size +: (`int`) The font size in pixels. Default is `20`. + +x +: (`int`) The horizontal offset, in pixels, relative to the left of the image. Default is `10`. + +y +: (`int`) The vertical offset, in pixels, relative to the top of the image. Default is `10`. + +[global resource]: /getting-started/glossary/#global-resource +[page resource]: /getting-started/glossary/#page-resource +[remote resource]: /getting-started/glossary/#remote-resource + +## Usage + +Capture the font as a resource: + +```go-html-template +{{ $font := "" }} +{{ $path := "https://github.com/google/fonts/raw/main/apache/roboto/static/Roboto-Regular.ttf" }} +{{ with resources.GetRemote $path }} + {{ with .Err }} + {{ errorf "%s" . }} + {{ else }} + {{ $font = . }} + {{ end }} +{{ else }} + {{ errorf "Unable to get resource %q" $path }} +{{ end }} +``` + +Create the options map: + +```go-html-template +{{ $opts := dict + "color" "#fbfaf5" + "font" $font + "linespacing" 8 + "size" 40 + "x" 25 + "y" 190 +}} +``` + +Set the text: + +```go-html-template +{{ $text := "Zion National Park" }} +``` + +Create the filter: + +```go-html-template +{{ $filter := images.Text $text $opts }} +``` + +{{% include "functions/images/_common/apply-image-filter.md" %}} + +## Example + +{{< img + src="images/examples/zion-national-park.jpg" + alt="Zion National Park" + filter="Text" + filterArgs="Zion National Park,25,190,40,1.2,#fbfaf5" + example=true +>}} diff --git a/content/en/functions/images/UnsharpMask.md b/content/en/functions/images/UnsharpMask.md new file mode 100644 index 00000000000..57a74a54a6a --- /dev/null +++ b/content/en/functions/images/UnsharpMask.md @@ -0,0 +1,40 @@ +--- +title: images.UnsharpMask +description: Returns an image filter that sharpens an image. +categories: [] +keywords: [] +action: + aliases: [] + related: + - functions/images/Filter + - methods/resource/Filter + returnType: images.filter + signatures: [images.UnsharpMask SIGMA AMOUNT THRESHOLD] +toc: true +--- + +The sigma parameter is used in a gaussian function and affects the radius of effect. Sigma must be positive. The sharpen radius is approximately 3 times the sigma value. + +The amount parameter controls how much darker and how much lighter the edge borders become. Typically between 0.5 and 1.5. + +The threshold parameter controls the minimum brightness change that will be sharpened. Typically between 0 and 0.05. + +## Usage + +Create the filter: + +```go-html-template +{{ $filter := images.UnsharpMask 10 0.4 0.03 }} +``` + +{{% include "functions/images/_common/apply-image-filter.md" %}} + +## Example + +{{< img + src="images/examples/zion-national-park.jpg" + alt="Zion National Park" + filter="UnsharpMask" + filterArgs="10,0.4,0.03" + example=true +>}} diff --git a/content/en/functions/images/_common/_index.md b/content/en/functions/images/_common/_index.md new file mode 100644 index 00000000000..47d5812fba5 --- /dev/null +++ b/content/en/functions/images/_common/_index.md @@ -0,0 +1,13 @@ +--- +cascade: + _build: + list: never + publishResources: false + render: never +--- + + diff --git a/content/en/functions/images/_common/apply-image-filter.md b/content/en/functions/images/_common/apply-image-filter.md new file mode 100644 index 00000000000..acd3a733dcb --- /dev/null +++ b/content/en/functions/images/_common/apply-image-filter.md @@ -0,0 +1,27 @@ +--- +# Do not remove front matter. +--- + +Apply the filter using the [`images.Filter`] function: + +[`images.Filter`]: /functions/images/filter + +```go-html-template +{{ with resources.Get "images/original.jpg" }} + {{ with . | images.Filter $filter }} + + {{ end }} +{{ end }} +``` + +You can also apply the filter using the [`Filter`] method on a `Resource` object: + +[`Filter`]: methods/resource/filter + +```go-html-template +{{ with resources.Get "images/original.jpg" }} + {{ with .Filter $filter }} + + {{ end }} +{{ end }} +``` diff --git a/content/en/functions/images/_index.md b/content/en/functions/images/_index.md new file mode 100644 index 00000000000..13542ea7306 --- /dev/null +++ b/content/en/functions/images/_index.md @@ -0,0 +1,12 @@ +--- +title: Image functions +linkTitle: images +description: Use these functions to create an image filter, apply an image filter to an image, and to retrieve image information. +categories: [] +keywords: [] +menu: + docs: + parent: functions +--- + +Use these functions to create an image filter, apply an image filter to an image, and to retrieve image information. diff --git a/content/en/functions/images/index.md b/content/en/functions/images/index.md deleted file mode 100644 index a1351f19046..00000000000 --- a/content/en/functions/images/index.md +++ /dev/null @@ -1,302 +0,0 @@ ---- -title: Image filters -description: The images namespace provides a list of filters and other image related functions. -categories: [functions] -keywords: [] -aliases: [/functions/imageconfig/] -menu: - docs: - parent: functions -keywords: [images] -toc: true ---- - -See [images.Filter](#filter) for how to apply these filters to an image. - -## Process - -{{< new-in "0.119.0" >}} - -{{< funcsig >}} -images.Process SRC SPEC -{{< /funcsig >}} - -A general purpose image processing function. - -This filter has all the same options as the [Process](/content-management/image-processing/#process) method, but using it as a filter may be more effective if you need to apply multiple filters to an image: - -```go-html-template -{{ $filters := slice - images.Grayscale - (images.GaussianBlur 8) - (images.Process "resize 200x jpg q30") -}} -{{ $img = $img | images.Filter $filters }} -``` - -## Overlay - -{{< funcsig >}} -images.Overlay SRC X Y -{{< /funcsig >}} - -Overlay creates a filter that overlays the source image at position x y, e.g: - - -```go-html-template -{{ $logoFilter := (images.Overlay $logo 50 50 ) }} -{{ $img := $img | images.Filter $logoFilter }} -``` - -A shorter version of the above, if you only need to apply the filter once: - -```go-html-template -{{ $img := $img.Filter (images.Overlay $logo 50 50 )}} -``` - -The above will overlay `$logo` in the upper left corner of `$img` (at position `x=50, y=50`). - -## Opacity - -{{< new-in "0.119.0" >}} - -{{< funcsig >}} -images.Opacity SRC OPACITY -{{< /funcsig >}} - -Opacity creates a filter that changes the opacity of an image. -The OPACITY parameter must be in range (0, 1). - -```go-html-template -{{ $img := $img.Filter (images.Opacity 0.5 )}} -``` - -This filter is most useful for target formats that support transparency, e.g. PNG. If the source image is e.g. JPG, the most effective way would be to combine it with the [`Process`] filter: - -```go-html-template -{{ $png := $jpg.Filter - (images.Opacity 0.5) - (images.Process "png") -}} -``` - -## Text - -Using the `Text` filter, you can add text to an image. - -{{< funcsig >}} -images.Text TEXT MAP) -{{< /funcsig >}} - -The following example will add the text `Hugo rocks!` to the image with the specified color, size and position. - -```go-html-template -{{ $img := resources.Get "/images/background.png" }} -{{ $img = $img.Filter (images.Text "Hugo rocks!" (dict - "color" "#ffffff" - "size" 60 - "linespacing" 2 - "x" 10 - "y" 20 -))}} -``` - -You can load a custom font if needed. Load the font as a Hugo `Resource` and set it as an option: - -```go-html-template -{{ $font := resources.GetRemote "https://github.com/google/fonts/raw/main/apache/roboto/static/Roboto-Black.ttf" }} -{{ $img := resources.Get "/images/background.png" }} -{{ $img = $img.Filter (images.Text "Hugo rocks!" (dict - "font" $font -))}} -``` - -## Padding - -{{< new-in "0.120.0" >}} - -Padding creates a filter that resizes the image canvas without resizing the image. The last argument is the canvas color, expressed as an RGB or RGBA [hexadecimal color]. The default value is `ffffffff` (opaque white). The preceding arguments are the padding values, in pixels, using the CSS [shorthand property] syntax. Negative padding values will crop the image. - -[hexadecimal color]: https://developer.mozilla.org/en-US/docs/Web/CSS/hex-color -[shorthand property]: https://developer.mozilla.org/en-US/docs/Web/CSS/Shorthand_properties#edges_of_a_box - -{{% funcsig %}} -images.Padding V1 [V2] [V3] [V4] [COLOR] -{{% /funcsig %}} - -This example resizes the image to 300px wide, converts it to the WebP format, adds 20px vertical padding and 50px horizontal padding, then sets the canvas color to dark green with 33% opacity. - -```go-html-template -{{ $img := resources.Get "images/a.jpg" }} -{{ $filters := slice - (images.Process "resize 300x webp") - (images.Padding 20 50 "#0705") -}} -{{ $img = $img.Filter $filters }} -``` - -To add a 2px gray border to an image: - -```go-html-template -{{ $img = $img.Filter (images.Padding 2 "#777") }} -``` - -## Brightness - -{{< funcsig >}} -images.Brightness PERCENTAGE -{{< /funcsig >}} - -Brightness creates a filter that changes the brightness of an image. -The percentage parameter must be in range (-100, 100). - -### ColorBalance - -{{< funcsig >}} -images.ColorBalance PERCENTAGERED PERCENTAGEGREEN PERCENTAGEBLUE -{{< /funcsig >}} - -ColorBalance creates a filter that changes the color balance of an image. -The percentage parameters for each color channel (red, green, blue) must be in range (-100, 500). - -## Colorize - -{{< funcsig >}} -images.Colorize HUE SATURATION PERCENTAGE -{{< /funcsig >}} - -Colorize creates a filter that produces a colorized version of an image. -The hue parameter is the angle on the color wheel, typically in range (0, 360). -The saturation parameter must be in range (0, 100). -The percentage parameter specifies the strength of the effect, it must be in range (0, 100). - -## Contrast - -{{< funcsig >}} -images.Contrast PERCENTAGE -{{< /funcsig >}} - -Contrast creates a filter that changes the contrast of an image. -The percentage parameter must be in range (-100, 100). - -## Gamma - -{{< funcsig >}} -images.Gamma GAMMA -{{< /funcsig >}} - -Gamma creates a filter that performs a gamma correction on an image. -The gamma parameter must be positive. Gamma = 1 gives the original image. -Gamma less than 1 darkens the image and gamma greater than 1 lightens it. - -## GaussianBlur - -{{< funcsig >}} -images.GaussianBlur SIGMA -{{< /funcsig >}} - -GaussianBlur creates a filter that applies a gaussian blur to an image. - -## Grayscale - -{{< funcsig >}} -images.Grayscale -{{< /funcsig >}} - -Grayscale creates a filter that produces a grayscale version of an image. - -## Hue - -{{< funcsig >}} -images.Hue SHIFT -{{< /funcsig >}} - -Hue creates a filter that rotates the hue of an image. -The hue angle shift is typically in range -180 to 180. - -## Invert - -{{< funcsig >}} -images.Invert -{{< /funcsig >}} - -Invert creates a filter that negates the colors of an image. - -## Pixelate - -{{< funcsig >}} -images.Pixelate SIZE -{{< /funcsig >}} - -Pixelate creates a filter that applies a pixelation effect to an image. - -## Saturation - -{{< funcsig >}} -images.Saturation PERCENTAGE -{{< /funcsig >}} - -Saturation creates a filter that changes the saturation of an image. - -## Sepia - -{{< funcsig >}} -images.Sepia PERCENTAGE -{{< /funcsig >}} - -Sepia creates a filter that produces a sepia-toned version of an image. - -## Sigmoid - -{{< funcsig >}} -images.Sigmoid MIDPOINT FACTOR -{{< /funcsig >}} - -Sigmoid creates a filter that changes the contrast of an image using a sigmoidal function and returns the adjusted image. -It's a non-linear contrast change useful for photo adjustments as it preserves highlight and shadow detail. - -## UnsharpMask - -{{< funcsig >}} -images.UnsharpMask SIGMA AMOUNT THRESHOLD -{{< /funcsig >}} - -UnsharpMask creates a filter that sharpens an image. -The sigma parameter is used in a gaussian function and affects the radius of effect. -Sigma must be positive. Sharpen radius roughly equals 3 * sigma. -The amount parameter controls how much darker and how much lighter the edge borders become. Typically between 0.5 and 1.5. -The threshold parameter controls the minimum brightness change that will be sharpened. Typically between 0 and 0.05. - -## Other Functions - -### Filter - -{{< funcsig >}} -IMAGE | images.Filter FILTERS... -{{< /funcsig >}} - -Can be used to apply a set of filters to an image: - -```go-html-template -{{ $img := $img | images.Filter (images.GaussianBlur 6) (images.Pixelate 8) }} -``` - -Also see the [Filter Method](/content-management/image-processing/#filter). - -### ImageConfig - -Parses the image and returns the height, width, and color model. - -The `imageConfig` function takes a single parameter, a file path (_string_) relative to the _project's root directory_, with or without a leading slash. - -{{< funcsig >}} -images.ImageConfig PATH -{{< /funcsig >}} - -```go-html-template -{{ with (imageConfig "favicon.ico") }} -favicon.ico: {{ .Width }} x {{ .Height }} -{{ end }} -``` - -[`Process`]: #process diff --git a/content/en/functions/inflect/Humanize.md b/content/en/functions/inflect/Humanize.md index 74d24f310e2..41d61a4e512 100644 --- a/content/en/functions/inflect/Humanize.md +++ b/content/en/functions/inflect/Humanize.md @@ -1,29 +1,26 @@ --- title: inflect.Humanize -linkTitle: humanize -description: Returns the humanized version of an argument with the first letter capitalized. -categories: [functions] +description: Returns the humanized version of the input with the first letter capitalized. +categories: [] keywords: [] -menu: - docs: - parent: functions -function: +action: aliases: [humanize] + related: + - functions/inflect/Pluralize + - functions/inflect/Singularize returnType: string signatures: [inflect.Humanize INPUT] -relatedFunctions: - - inflect.Humanize - - inflect.Pluralize - - inflect.Singularize aliases: [/functions/humanize] --- -If the input is either an int64 value or the string representation of an integer, humanize returns the number with the proper ordinal appended. +```go-html-template +{{ humanize "my-first-post" }} → My first post +{{ humanize "myCamelPost" }} → My camel post +``` +If the input is either an int64 value or the string representation of an integer, humanize returns the number with the proper ordinal appended. ```go-html-template -{{ humanize "my-first-post" }} → "My first post" -{{ humanize "myCamelPost" }} → "My camel post" -{{ humanize "52" }} → "52nd" -{{ humanize 103 }} → "103rd" +{{ humanize "52" }} → 52nd +{{ humanize 103 }} → 103rd ``` diff --git a/content/en/functions/inflect/Pluralize.md b/content/en/functions/inflect/Pluralize.md index 5bb444114a6..c25f8961701 100644 --- a/content/en/functions/inflect/Pluralize.md +++ b/content/en/functions/inflect/Pluralize.md @@ -1,23 +1,18 @@ --- title: inflect.Pluralize -linkTitle: pluralize -description: Pluralizes the given word according to a set of common English pluralization rules -categories: [functions] +description: Pluralizes the given word according to a set of common English pluralization rules. +categories: [] keywords: [] -menu: - docs: - parent: functions -function: +action: aliases: [pluralize] + related: + - functions/inflect/Humanize + - functions/inflect/Singularize returnType: string signatures: [inflect.Pluralize INPUT] -relatedFunctions: - - inflect.Humanize - - inflect.Pluralize - - inflect.Singularize aliases: [/functions/pluralize] --- ```go-html-template -{{ "cat" | pluralize }} → "cats" +{{ "cat" | pluralize }} → cats ``` diff --git a/content/en/functions/inflect/Singularize.md b/content/en/functions/inflect/Singularize.md index 5aba4e4ee68..29b54325779 100644 --- a/content/en/functions/inflect/Singularize.md +++ b/content/en/functions/inflect/Singularize.md @@ -1,25 +1,20 @@ --- title: inflect.Singularize -linkTitle: singularize -description: Converts a word according to a set of common English singularization rules. -categories: [functions] +description: Singularizes the given word according to a set of common English singularization rules. +categories: [] keywords: [] -menu: - docs: - parent: functions -function: +action: aliases: [singularize] + related: + - functions/inflect/Humanize + - functions/inflect/Pluralize returnType: string signatures: [inflect.Singularize INPUT] -relatedFunctions: - - inflect.Humanize - - inflect.Pluralize - - inflect.Singularize aliases: [/functions/singularize] --- ```go-html-template -{{ "cats" | singularize }} → "cat" +{{ "cats" | singularize }} → cat ``` See also the `.Data.Singular` [taxonomy variable](/variables/taxonomy/) for singularizing taxonomy names. diff --git a/content/en/functions/inflect/_index.md b/content/en/functions/inflect/_index.md new file mode 100644 index 00000000000..601b409e6a0 --- /dev/null +++ b/content/en/functions/inflect/_index.md @@ -0,0 +1,12 @@ +--- +title: Inflect functions +linkTitle: inflect +description: Template functions to inflect English nouns. +categories: [] +keywords: [] +menu: + docs: + parent: functions +--- + +These functions provide word inflection features such as singularization and pluralization of English nouns. diff --git a/content/en/functions/js/Build.md b/content/en/functions/js/Build.md new file mode 100644 index 00000000000..69f0b5fdaf8 --- /dev/null +++ b/content/en/functions/js/Build.md @@ -0,0 +1,182 @@ +--- +title: js.Build +description: Bundles, transpiles, tree shakes, and minifies JavaScript resources. +categories: [] +keywords: [] +action: + aliases: [] + related: + - functions/resources/Babel + - functions/resources/Fingerprint + - functions/resources/Minify + returnType: resource.Resource + signatures: ['js.Build [OPTIONS] RESOURCE'] +toc: true +--- + +The `js.Build` function uses the [evanw/esbuild] package to: + +- Bundle +- Transpile (TypeScript and JSX) +- Tree shake +- Minify +- Create source maps + +[evanw/esbuild]: https://github.com/evanw/esbuild + +```go-html-template +{{ with resources.Get "js/main.js" }} + {{ if eq hugo.Environment "development" }} + {{ with . | js.Build }} + + {{ end }} + {{ else }} + {{ $opts := dict "minify" true }} + {{ with . | js.Build $opts | fingerprint }} + + {{ end }} + {{ end }} +{{ end }} +``` + +## Options + +targetPath +: (`string`) If not set, the source path will be used as the base target path. +Note that the target path's extension may change if the target MIME type is different, e.g. when the source is TypeScript. + +params [map or slice] +: (`map` or `slice`) Params that can be imported as JSON in your JS files, e.g. + +```go-html-template +{{ $js := resources.Get "js/main.js" | js.Build (dict "params" (dict "api" "https://example.org/api")) }} +``` +And then in your JS file: + +```js +import * as params from '@params'; +``` + +Note that this is meant for small data sets, e.g. configuration settings. For larger data, please put/mount the files into `/assets` and import them directly. + +minify +: (`bool`)Let `js.Build` handle the minification. + +inject +: (`slice`) This option allows you to automatically replace a global variable with an import from another file. The path names must be relative to `assets`. See https://esbuild.github.io/api/#inject + +shims +: (`map`) This option allows swapping out a component with another. A common use case is to load dependencies like React from a CDN (with _shims_) when in production, but running with the full bundled `node_modules` dependency during development: + +```go-html-template +{{ $shims := dict "react" "js/shims/react.js" "react-dom" "js/shims/react-dom.js" }} +{{ $js = $js | js.Build dict "shims" $shims }} +``` + +The _shim_ files may look like these: + +```js +// js/shims/react.js +module.exports = window.React; +``` + +```js +// js/shims/react-dom.js +module.exports = window.ReactDOM; +``` + +With the above, these imports should work in both scenarios: + +```js +import * as React from 'react' +import * as ReactDOM from 'react-dom'; +``` + +target +: (`string`) The language target. One of: `es5`, `es2015`, `es2016`, `es2017`, `es2018`, `es2019`, `es2020` or `esnext`. Default is `esnext`. + +externals +: (`slice`) External dependencies. Use this to trim dependencies you know will never be executed. See https://esbuild.github.io/api/#external + +defines +: (`map`) Allow to define a set of string replacement to be performed when building. Should be a map where each key is to be replaced by its value. + +```go-html-template +{{ $defines := dict "process.env.NODE_ENV" `"development"` }} +``` + +format +: (`string`) The output format. One of: `iife`, `cjs`, `esm`. Default is `iife`, a self-executing function, suitable for inclusion as a ` +``` diff --git a/content/en/functions/js/_index.md b/content/en/functions/js/_index.md new file mode 100644 index 00000000000..3356e7c7be6 --- /dev/null +++ b/content/en/functions/js/_index.md @@ -0,0 +1,12 @@ +--- +title: JavaScript functions +linkTitle: js +description: Template functions to work with JavaScript and TypeScript files. +categories: [] +keywords: [] +menu: + docs: + parent: functions +--- + +Use these functions to work with JavaScript and TypeScript files. diff --git a/content/en/functions/lang/FormatAccounting.md b/content/en/functions/lang/FormatAccounting.md index 974dc4a1a78..70365c21626 100644 --- a/content/en/functions/lang/FormatAccounting.md +++ b/content/en/functions/lang/FormatAccounting.md @@ -1,27 +1,21 @@ --- title: lang.FormatAccounting -description: Returns a currency representation of a number for the given currency and precision for the current language in accounting notation. -categories: [functions] +description: Returns a currency representation of a number for the given currency and precision for the current language and region in accounting notation. +categories: [] keywords: [] -menu: - docs: - parent: functions -function: +action: aliases: [] + related: + - functions/lang/FormatCurrency + - functions/lang/FormatNumber + - functions/lang/FormatNumberCustom + - functions/lang/FormatPercent returnType: string signatures: [lang.FormatAccounting PRECISION CURRENCY NUMBER] -relatedFunctions: - - lang.FormatAccounting - - lang.FormatCurrency - - lang.FormatNumber - - lang.FormatNumberCustom - - lang.FormatPercent --- ```go-html-template {{ 512.5032 | lang.FormatAccounting 2 "NOK" }} → NOK512.50 ``` -{{% note %}} -{{% readfile file="/functions/_common/locales.md" %}} -{{% /note %}} +{{% include "functions/_common/locales.md" %}} diff --git a/content/en/functions/lang/FormatCurrency.md b/content/en/functions/lang/FormatCurrency.md index b29a807fee3..91148c5951d 100644 --- a/content/en/functions/lang/FormatCurrency.md +++ b/content/en/functions/lang/FormatCurrency.md @@ -1,27 +1,21 @@ --- title: lang.FormatCurrency -description: Returns a currency representation of a number for the given currency and precision for the current language. -categories: [functions] +description: Returns a currency representation of a number for the given currency and precision for the current language and region. +categories: [] keywords: [] -menu: - docs: - parent: functions -function: +action: aliases: [] + related: + - functions/lang/FormatAccounting + - functions/lang/FormatNumber + - functions/lang/FormatNumberCustom + - functions/lang/FormatPercent returnType: string signatures: [lang.FormatAccounting PRECISION CURRENCY NUMBER] -relatedFunctions: - - lang.FormatAccounting - - lang.FormatCurrency - - lang.FormatNumber - - lang.FormatNumberCustom - - lang.FormatPercent --- ```go-html-template {{ 512.5032 | lang.FormatCurrency 2 "USD" }} → $512.50 ``` -{{% note %}} -{{% readfile file="/functions/_common/locales.md" %}} -{{% /note %}} +{{% include "functions/_common/locales.md" %}} diff --git a/content/en/functions/lang/FormatNumber.md b/content/en/functions/lang/FormatNumber.md index dd878fdefeb..597df742a31 100644 --- a/content/en/functions/lang/FormatNumber.md +++ b/content/en/functions/lang/FormatNumber.md @@ -1,27 +1,21 @@ --- title: lang.FormatNumber -description: Returns a numeric representation of a number with the given precision for the current language. -categories: [functions] +description: Returns a numeric representation of a number with the given precision for the current language and region. +categories: [] keywords: [] -menu: - docs: - parent: functions -function: +action: aliases: [] + related: + - functions/lang/FormatAccounting + - functions/lang/FormatCurrency + - functions/lang/FormatNumberCustom + - functions/lang/FormatPercent returnType: string signatures: [lang.FormatNumber PRECISION NUMBER] -relatedFunctions: - - lang.FormatAccounting - - lang.FormatCurrency - - lang.FormatNumber - - lang.FormatNumberCustom - - lang.FormatPercent --- ```go-html-template {{ 512.5032 | lang.FormatNumber 2 }} → 512.50 ``` -{{% note %}} -{{% readfile file="/functions/_common/locales.md" %}} -{{% /note %}} +{{% include "functions/_common/locales.md" %}} diff --git a/content/en/functions/lang/FormatNumberCustom.md b/content/en/functions/lang/FormatNumberCustom.md index 97b02256706..0b72f49833c 100644 --- a/content/en/functions/lang/FormatNumberCustom.md +++ b/content/en/functions/lang/FormatNumberCustom.md @@ -1,31 +1,26 @@ --- title: lang.FormatNumberCustom description: Returns a numeric representation of a number with the given precision using negative, decimal, and grouping options. -categories: [functions] +categories: [] keywords: [] -menu: - docs: - parent: functions -function: +action: aliases: [] + related: + - functions/lang/FormatAccounting + - functions/lang/FormatCurrency + - functions/lang/FormatNumber + - functions/lang/FormatPercent returnType: string signatures: ['lang.FormatNumberCustom PRECISION NUMBER [OPTIONS...]'] -relatedFunctions: - - lang.FormatAccounting - - lang.FormatCurrency - - lang.FormatNumber - - lang.FormatNumberCustom - - lang.FormatPercent aliases: ['/functions/numfmt/'] --- This function formats a number with the given precision. The first options parameter is a space-delimited string of characters to represent negativity, the decimal point, and grouping. The default value is `- . ,`. The second options parameter defines an alternate delimiting character. -Note that numbers are rounded up at 5 or greater. So, with precision set to 0, 1.5 becomes 2, and 1.4 becomes 1. +Note that numbers are rounded up at 5 or greater. So, with precision set to 0, 1.5 becomes 2, and 1.4 becomes 1. For a simpler function that adapts to the current language, see [`lang.FormatNumber`]. - ```go-html-template {{ lang.FormatNumberCustom 2 12345.6789 }} → 12,345.68 {{ lang.FormatNumberCustom 2 12345.6789 "- , ." }} → 12.345,68 @@ -34,8 +29,6 @@ For a simpler function that adapts to the current language, see [`lang.FormatNum {{ lang.FormatNumberCustom 0 -12345.6789 "-|.| " "|" }} → -12 346 ``` -{{% note %}} -{{% readfile file="/functions/_common/locales.md" %}} -{{% /note %}} +{{% include "functions/_common/locales.md" %}} [`lang.FormatNumber`]: /functions/lang/formatnumber diff --git a/content/en/functions/lang/FormatPercent.md b/content/en/functions/lang/FormatPercent.md index dd204249005..529ada67bf7 100644 --- a/content/en/functions/lang/FormatPercent.md +++ b/content/en/functions/lang/FormatPercent.md @@ -1,27 +1,21 @@ --- title: lang.FormatPercent -description: Returns a percentage representation of a number with the given precision for the current language. -categories: [functions] +description: Returns a percentage representation of a number with the given precision for the current language and region. +categories: [] keywords: [] -menu: - docs: - parent: functions -function: +action: aliases: [] + related: + - functions/lang/FormatAccounting + - functions/lang/FormatCurrency + - functions/lang/FormatNumber + - functions/lang/FormatNumberCustom returnType: string signatures: [lang.FormatPercent PRECISION NUMBER] -relatedFunctions: - - lang.FormatAccounting - - lang.FormatCurrency - - lang.FormatNumber - - lang.FormatNumberCustom - - lang.FormatPercent --- ```go-html-template {{ 512.5032 | lang.FormatPercent 2 }} → 512.50% ``` -{{% note %}} -{{% readfile file="/functions/_common/locales.md" %}} -{{% /note %}} +{{% include "functions/_common/locales.md" %}} diff --git a/content/en/functions/lang/Merge.md b/content/en/functions/lang/Merge.md index b3d21cd7ab8..75f5cdf01ee 100644 --- a/content/en/functions/lang/Merge.md +++ b/content/en/functions/lang/Merge.md @@ -1,31 +1,27 @@ --- title: lang.Merge description: Merge missing translations from other languages. -categories: [functions] +categories: [] keywords: [] -menu: - docs: - parent: functions -function: +action: aliases: [] + related: [] returnType: any signatures: [lang.Merge FROM TO] -relatedFunctions: [] aliases: [/functions/lang.merge] --- As an example: -```bash +```sh {{ $pages := .Site.RegularPages | lang.Merge $frSite.RegularPages | lang.Merge $enSite.RegularPages }} ``` Will "fill in the gaps" in the current site with, from left to right, content from the French site, and lastly the English. - A more practical example is to fill in the missing translations from the other languages: -```bash +```sh {{ $pages := .Site.RegularPages }} {{ range .Site.Home.Translations }} {{ $pages = $pages | lang.Merge .Site.RegularPages }} diff --git a/content/en/functions/lang/Translate.md b/content/en/functions/lang/Translate.md index 718d8cfb2ca..3d091d67d4b 100644 --- a/content/en/functions/lang/Translate.md +++ b/content/en/functions/lang/Translate.md @@ -1,23 +1,19 @@ --- title: lang.Translate -linkTitle: i18n description: Translates a string using the translation tables in the i18n directory. -categories: [functions] +categories: [] keywords: [] -menu: - docs: - parent: functions -function: +action: aliases: [i18n,T] + related: [] returnType: string signatures: ['lang.Translate KEY [CONTEXT]'] -relatedFunctions: [] aliases: [/functions/i18n] --- Let's say your multilingual site supports two languages, English and Polish. Create a translation table for each language in the `i18n` directory. -``` +```text i18n/ ├── en.toml └── pl.toml @@ -34,12 +30,10 @@ The Unicode [CLDR Plural Rules chart] describes the pluralization categories for The English translation table: -{{< code-toggle file=i18n/en copy=false >}} -# simple translations +{{< code-toggle file=i18n/en >}} privacy = 'privacy' security = 'security' -# translations with pluralization [day] one = 'day' other = 'days' @@ -51,12 +45,10 @@ other = '{{ . }} days' The Polish translation table: -{{< code-toggle file=i18n/pl copy=false >}} -# simple translations +{{< code-toggle file=i18n/pl >}} privacy = 'prywatność' security = 'bezpieczeństwo' -# translations with pluralization [day] one = 'miesiąc' few = 'miesiące' @@ -108,17 +100,17 @@ When viewing the Polish language site: {{ T "day_with_count" 5 }} → 5 miesięcy ``` -In the pluralization examples above, we passed an integer in context (the second argument). You can also pass a map in context, creating a `count` key to control pluralization. +In the pluralization examples above, we passed an integer in context (the second argument). You can also pass a map in context, providing a `count` key to control pluralization. Translation table: -{{< code-toggle file=i18n/en copy=false >}} +{{< code-toggle file=i18n/en >}} [age] one = '{{ .name }} is {{ .count }} year old.' other = '{{ .name }} is {{ .count }} years old.' {{< /code-toggle >}} -Template: +Template code: ```go-html-template {{ T "age" (dict "name" "Will" "count" 1) }} → Will is 1 year old. diff --git a/content/en/functions/lang/_index.md b/content/en/functions/lang/_index.md new file mode 100644 index 00000000000..934d97bffd2 --- /dev/null +++ b/content/en/functions/lang/_index.md @@ -0,0 +1,12 @@ +--- +title: Lang functions +linkTitle: lang +description: Template functions to adapt your site to meet language and regional requirements. +categories: [] +keywords: [] +menu: + docs: + parent: functions +--- + +Use these functions to adapt your site to meet language and regional requirements. diff --git a/content/en/functions/math/Abs.md b/content/en/functions/math/Abs.md new file mode 100644 index 00000000000..6e907d56452 --- /dev/null +++ b/content/en/functions/math/Abs.md @@ -0,0 +1,15 @@ +--- +title: math.Abs +description: Returns the absolute value of the given number. +categories: [] +keywords: [] +action: + aliases: [] + related: [] + returnType: float64 + signatures: [math.Abs VALUE] +--- + +```go-html-template +{{ math.Abs -2.1 }} → 2.1 +``` diff --git a/content/en/functions/math/Add.md b/content/en/functions/math/Add.md new file mode 100644 index 00000000000..5e3bfb16223 --- /dev/null +++ b/content/en/functions/math/Add.md @@ -0,0 +1,22 @@ +--- +title: math.Add +description: Adds two or more numbers. +categories: [] +keywords: [] +action: + aliases: [add] + related: + - functions/math/Div + - functions/math/Mul + - functions/math/Product + - functions/math/Sub + - functions/math/Sum + returnType: any + signatures: [math.Add VALUE VALUE...] +--- + +If one of the numbers is a float, the result is a float. + +```go-html-template +{{ add 12 3 2 }} → 17 +``` diff --git a/content/en/functions/math/Ceil.md b/content/en/functions/math/Ceil.md new file mode 100644 index 00000000000..9f74991c32b --- /dev/null +++ b/content/en/functions/math/Ceil.md @@ -0,0 +1,17 @@ +--- +title: math.Ceil +description: Returns the least integer value greater than or equal to the given number. +categories: [] +keywords: [] +action: + aliases: [] + related: + - functions/math/Floor + - functions/math/Round + returnType: float64 + signatures: [math.Ceil VALUE] +--- + +```go-html-template +{{ math.Ceil 2.1 }} → 3 +``` diff --git a/content/en/functions/math/Counter.md b/content/en/functions/math/Counter.md new file mode 100644 index 00000000000..7f53bdd0c1e --- /dev/null +++ b/content/en/functions/math/Counter.md @@ -0,0 +1,35 @@ +--- +title: math.Counter +description: Increments and returns a global counter. +categories: [] +keywords: [] +action: + aliases: [] + related: [] + returnType: uint64 + signatures: [math.Counter] +--- + +The counter is global for both monolingual and multilingual sites, and its initial value for each build is 1. + +```go-html-template +{{ warnf "single.html called %d times" math.Counter }} +``` + +```sh +WARN single.html called 1 times +WARN single.html called 2 times +WARN single.html called 3 times +``` + +Use this function to: + +- Create unique warnings as shown above; the [`warnf`] function suppresses duplicate messages +- Create unique target paths for the `resources.FromString` function where the target path is also the cache key + +[`warnf`]: /functions/fmt/warnf +[`resources.FromString`]: /functions/resources/fromstring + +{{% note %}} +Due to concurrency, the value returned in a given template for a given page will vary from one build to the next. You cannot use this function to assign a static id to each page. +{{% /note %}} diff --git a/content/en/functions/math/Div.md b/content/en/functions/math/Div.md new file mode 100644 index 00000000000..5123791b2be --- /dev/null +++ b/content/en/functions/math/Div.md @@ -0,0 +1,22 @@ +--- +title: math.Div +description: Divides the first number by one or more numbers. +categories: [] +keywords: [] +action: + aliases: [div] + related: + - functions/math/Add + - functions/math/Mul + - functions/math/Product + - functions/math/Sub + - functions/math/Sum + returnType: any + signatures: [math.Div VALUE VALUE...] +--- + +If one of the numbers is a float, the result is a float. + +```go-html-template +{{ div 12 3 2 }} → 2 +``` diff --git a/content/en/functions/math/Floor.md b/content/en/functions/math/Floor.md new file mode 100644 index 00000000000..10ad758b6b9 --- /dev/null +++ b/content/en/functions/math/Floor.md @@ -0,0 +1,17 @@ +--- +title: math.Floor +description: Returns the greatest integer value less than or equal to the given number. +categories: [] +keywords: [] +action: + aliases: [] + related: + - functions/math/Ceil + - functions/math/Round + returnType: float64 + signatures: [math.Floor VALUE] +--- + +```go-html-template +{{ math.Floor 1.9 }} → 1 +``` diff --git a/content/en/functions/math/Log.md b/content/en/functions/math/Log.md new file mode 100644 index 00000000000..84edcb28851 --- /dev/null +++ b/content/en/functions/math/Log.md @@ -0,0 +1,15 @@ +--- +title: math.Log +description: Returns the natural logarithm of the given number. +categories: [] +keywords: [] +action: + aliases: [] + related: [] + returnType: float64 + signatures: [math.Log VALUE] +--- + +```go-html-template +{{ math.Log 42 }} → 3.737 +``` diff --git a/content/en/functions/math/Max.md b/content/en/functions/math/Max.md new file mode 100644 index 00000000000..9beff563006 --- /dev/null +++ b/content/en/functions/math/Max.md @@ -0,0 +1,16 @@ +--- +title: math.Max +description: Returns the greater of all numbers. Accepts scalars, slices, or both. +categories: [] +keywords: [] +action: + aliases: [] + related: + - functions/math/Min + returnType: float64 + signatures: [math.Max VALUE...] +--- + +```go-html-template +{{ math.Max 1 (slice 2 3) 4 }} → 4 +``` diff --git a/content/en/functions/math/Min.md b/content/en/functions/math/Min.md new file mode 100644 index 00000000000..79e464c74ab --- /dev/null +++ b/content/en/functions/math/Min.md @@ -0,0 +1,16 @@ +--- +title: math.Min +description: Returns the smaller of all numbers. Accepts scalars, slices, or both. +categories: [] +keywords: [] +action: + aliases: [] + related: + - functions/math/Max + returnType: float64 + signatures: [math.Min VALUE...] +--- + +```go-html-template +{{ math.Min 1 (slice 2 3) 4 }} → 1 +``` diff --git a/content/en/functions/math/Mod.md b/content/en/functions/math/Mod.md new file mode 100644 index 00000000000..d312730c542 --- /dev/null +++ b/content/en/functions/math/Mod.md @@ -0,0 +1,16 @@ +--- +title: math.Mod +description: Returns the modulus of two integers. +categories: [] +keywords: [] +action: + aliases: [mod] + related: + - functions/math/ModBool + returnType: int64 + signatures: [math.Mod VALUE1 VALUE2] +--- + +```go-html-template +{{ mod 15 3 }} → 0 +``` diff --git a/content/en/functions/math/ModBool.md b/content/en/functions/math/ModBool.md new file mode 100644 index 00000000000..d915ff6141b --- /dev/null +++ b/content/en/functions/math/ModBool.md @@ -0,0 +1,16 @@ +--- +title: math.ModBool +description: Reports whether the modulus of two integers equals 0. +categories: [] +keywords: [] +action: + aliases: [modBool] + related: + - functions/math/Mod + returnType: bool + signatures: [math.ModBool VALUE1 VALUE2] +--- + +```go-html-template +{{ modBool 15 3 }} → true +``` diff --git a/content/en/functions/math/Mul.md b/content/en/functions/math/Mul.md new file mode 100644 index 00000000000..1db8ad9eb38 --- /dev/null +++ b/content/en/functions/math/Mul.md @@ -0,0 +1,22 @@ +--- +title: math.Mul +description: Multiplies two or more numbers. +categories: [] +keywords: [] +action: + aliases: [mul] + related: + - functions/math/Add + - functions/math/Div + - functions/math/Product + - functions/math/Sub + - functions/math/Sum + returnType: any + signatures: [math.Mul VALUE VALUE...] +--- + +If one of the numbers is a float, the result is a float. + +```go-html-template +{{ mul 12 3 2 }} → 72 +``` diff --git a/content/en/functions/math/Pow.md b/content/en/functions/math/Pow.md new file mode 100644 index 00000000000..5a1482d73c7 --- /dev/null +++ b/content/en/functions/math/Pow.md @@ -0,0 +1,16 @@ +--- +title: math.Pow +description: Returns the first number raised to the power of the second number. +categories: [] +keywords: [] +action: + aliases: [pow] + related: + - functions/math/Sqrt + returnType: float64 + signatures: [math.Pow VALUE1 VALUE2] +--- + +```go-html-template +{{ math.Pow 2 3 }} → 8 +``` diff --git a/content/en/functions/math/Product.md b/content/en/functions/math/Product.md new file mode 100644 index 00000000000..343e9f73d84 --- /dev/null +++ b/content/en/functions/math/Product.md @@ -0,0 +1,20 @@ +--- +title: math.Product +description: Returns the product of all numbers. Accepts scalars, slices, or both. +categories: [] +keywords: [] +action: + aliases: [] + related: + - functions/math/Add + - functions/math/Div + - functions/math/Mul + - functions/math/Sub + - functions/math/Sum + returnType: float64 + signatures: [math.Product VALUE...] +--- + +```go-html-template +{{ math.Product 1 (slice 2 3) 4 }} → 24 +``` diff --git a/content/en/functions/math/Round.md b/content/en/functions/math/Round.md new file mode 100644 index 00000000000..e0678eb784c --- /dev/null +++ b/content/en/functions/math/Round.md @@ -0,0 +1,17 @@ +--- +title: math.Round +description: Returns the nearest integer, rounding half away from zero. +categories: [] +keywords: [] +action: + aliases: [] + related: + - functions/math/Ceil + - functions/math/Floor + returnType: float64 + signatures: [math.Round VALUE] +--- + +```go-html-template +{{ math.Round 1.5 }} → 2 +``` diff --git a/content/en/functions/math/Sqrt.md b/content/en/functions/math/Sqrt.md new file mode 100644 index 00000000000..436cb31c327 --- /dev/null +++ b/content/en/functions/math/Sqrt.md @@ -0,0 +1,16 @@ +--- +title: math.Sqrt +description: Returns the square root of the given number. +categories: [] +keywords: [] +action: + aliases: [] + related: + - functions/math/Pow + returnType: float64 + signatures: [math.Sqrt VALUE] +--- + +```go-html-template +{{ math.Sqrt 81 }} → 9 +``` diff --git a/content/en/functions/math/Sub.md b/content/en/functions/math/Sub.md new file mode 100644 index 00000000000..6c4ef10d580 --- /dev/null +++ b/content/en/functions/math/Sub.md @@ -0,0 +1,21 @@ +--- +title: math.Sub +description: Subtracts one or more numbers from the first number. +categories: [] +action: + aliases: [sub] + related: + - functions/math/Add + - functions/math/Div + - functions/math/Mul + - functions/math/Product + - functions/math/Sum + returnType: any + signatures: [math.Sub VALUE VALUE...] +--- + +If one of the numbers is a float, the result is a float. + +```go-html-template +{{ sub 12 3 2 }} → 7 +``` diff --git a/content/en/functions/math/Sum.md b/content/en/functions/math/Sum.md new file mode 100644 index 00000000000..ae1419fb356 --- /dev/null +++ b/content/en/functions/math/Sum.md @@ -0,0 +1,19 @@ +--- +title: math.Sum +description: Returns the sum of all numbers. Accepts scalars, slices, or both. +categories: [] +action: + aliases: [] + related: + - functions/math/Add + - functions/math/Div + - functions/math/Mul + - functions/math/Product + - functions/math/Sub + returnType: float64 + signatures: [math.Sum VALUE...] +--- + +```go-html-template +{{ math.Sum 1 (slice 2 3) 4 }} → 10 +``` diff --git a/content/en/functions/math/_index.md b/content/en/functions/math/_index.md new file mode 100644 index 00000000000..76713bc99a4 --- /dev/null +++ b/content/en/functions/math/_index.md @@ -0,0 +1,11 @@ +--- +title: Math functions +linkTitle: math +description: Template functions to perform mathematical operations. +categories: [] +menu: + docs: + parent: functions +--- + +Use these functions to perform mathematical operations. diff --git a/content/en/functions/math/index.md b/content/en/functions/math/index.md deleted file mode 100644 index fd4d10a3177..00000000000 --- a/content/en/functions/math/index.md +++ /dev/null @@ -1,39 +0,0 @@ ---- -title: math -description: Hugo provides mathematical operators in templates. -categories: [functions] -keywords: [] - -menu: - docs: - parent: functions -function: - aliases: [] - returnType: - signatures: [] -relatedFunctions: [] ---- - -| Function | Description | Example | -|-----------------|-----------------------------------------------------------------------------|---------------------------------------------------| -| `add` | Adds two or more numbers. | `{{ add 12 3 2 }}` → `17` | -| | *If one of the numbers is a float, the result is a float.* | `{{ add 1.1 2 }}` → `3.1` | -| `sub` | Subtracts one or more numbers from the first number. | `{{ sub 12 3 2 }}` → `7` | -| | *If one of the numbers is a float, the result is a float.* | `{{ sub 3 2.5 }}` → `0.5` | -| `mul` | Multiplies two or more numbers. | `{{ mul 12 3 2 }}` → `72` | -| | *If one of the numbers is a float, the result is a float.* | `{{ mul 2 3.1 }}` → `6.2` | -| `div` | Divides the first number by one or more numbers. | `{{ div 12 3 2 }}` → `2` | -| | *If one of the numbers is a float, the result is a float.* | `{{ div 6 4.0 }}` → `1.5` | -| `mod` | Modulus of two integers. | `{{ mod 15 3 }}` → `0` | -| `modBool` | Boolean of modulus of two integers. Evaluates to `true` if result equals 0. | `{{ modBool 15 3 }}` → `true` | -| `math.Abs` | Returns the absolute value of the given number. | `{{ math.Abs -2.1 }}` → `2.1` | -| `math.Ceil` | Returns the least integer value greater than or equal to the given number. | `{{ math.Ceil 2.1 }}` → `3` | -| `math.Floor` | Returns the greatest integer value less than or equal to the given number. | `{{ math.Floor 1.9 }}` → `1` | -| `math.Log` | Returns the natural logarithm of the given number. | `{{ math.Log 42 }}` → `3.737` | -| `math.Max` | Returns the greater of all numbers. Accepts scalars, slices, or both. | `{{ math.Max 1 (slice 2 3) 4 }}` → `4` | -| `math.Min` | Returns the smaller of all numbers. Accepts scalars, slices, or both. | `{{ math.Min 1 (slice 2 3) 4 }}` → `1` | -| `math.Product` | Returns the product of all numbers. Accepts scalars, slices, or both. | `{{ math.Product 1 (slice 2 3) 4 }}` → `24` | -| `math.Pow` | Returns the first number raised to the power of the second number. | `{{ math.Pow 2 3 }}` → `8` | -| `math.Round` | Returns the nearest integer, rounding half away from zero. | `{{ math.Round 1.5 }}` → `2` | -| `math.Sqrt` | Returns the square root of the given number. | `{{ math.Sqrt 81 }}` → `9` | -| `math.Sum` | Returns the sum of all numbers. Accepts scalars, slices, or both. | `{{ math.Sum 1 (slice 2 3) 4 }}` → `10` | diff --git a/content/en/functions/os/FileExists.md b/content/en/functions/os/FileExists.md index 52cfe32c82f..b8104a066de 100644 --- a/content/en/functions/os/FileExists.md +++ b/content/en/functions/os/FileExists.md @@ -1,22 +1,17 @@ --- title: os.FileExists -linkTitle: fileExists description: Reports whether the file or directory exists. -categories: [functions] +categories: [] keywords: [] -menu: - docs: - parent: functions -function: +action: aliases: [fileExists] + related: + - functions/os/Getenv + - functions/os/ReadDir + - functions/os/ReadFile + - functions/os/Stat returnType: bool signatures: [os.FileExists PATH] -relatedFunctions: - - os.FileExists - - os.Getenv - - os.ReadDir - - os.ReadFile - - os.Stat aliases: [/functions/fileexists] --- @@ -36,11 +31,11 @@ content/ The function returns these values: ```go-html-template -{{ os.FileExists "content" }} → true -{{ os.FileExists "content/news" }} → true -{{ os.FileExists "content/news/article-1" }} → false -{{ os.FileExists "content/news/article-1.md" }} → true -{{ os.FileExists "news" }} → true -{{ os.FileExists "news/article-1" }} → false -{{ os.FileExists "news/article-1.md" }} → true +{{ fileExists "content" }} → true +{{ fileExists "content/news" }} → true +{{ fileExists "content/news/article-1" }} → false +{{ fileExists "content/news/article-1.md" }} → true +{{ fileExists "news" }} → true +{{ fileExists "news/article-1" }} → false +{{ fileExists "news/article-1.md" }} → true ``` diff --git a/content/en/functions/os/Getenv.md b/content/en/functions/os/Getenv.md index 16f73f5aad2..084d498ce80 100644 --- a/content/en/functions/os/Getenv.md +++ b/content/en/functions/os/Getenv.md @@ -1,35 +1,49 @@ --- title: os.Getenv -linkTitle: getenv description: Returns the value of an environment variable, or an empty string if the environment variable is not set. -categories: [functions] +categories: [] keywords: [] -menu: - docs: - parent: functions -function: +action: aliases: [getenv] + related: + - functions/os/FileExists + - functions/os/ReadDir + - functions/os/ReadFile + - functions/os/Stat returnType: string signatures: [os.Getenv VARIABLE] -relatedFunctions: - - os.FileExists - - os.Getenv - - os.ReadDir - - os.ReadFile - - os.Stat aliases: [/functions/getenv] +toc: true --- -Examples: +## Security + +By default, when using the `os.Getenv` function Hugo allows access to: + +- The `CI` environment variable +- Any environment variable beginning with `HUGO_` + +To access other environment variables, adjust your site configuration. For example, to allow access to the `HOME` and `USER` environment variables: + +{{< code-toggle file=hugo >}} +[security.funcs] +getenv = ['^HUGO_', '^CI$', '^USER$', '^HOME$'] +{{< /code-toggle >}} + +Read more about Hugo's [security policy]. + +[security policy]: /about/security-model/#security-policy + +## Examples ```go-html-template -{{ os.Getenv "HOME" }} → /home/victor -{{ os.Getenv "USER" }} → victor +{{ getenv "HOME" }} → /home/victor +{{ getenv "USER" }} → victor ``` You can pass values when building your site: -```bash +```sh MY_VAR1=foo MY_VAR2=bar hugo OR @@ -42,8 +56,6 @@ hugo And then retrieve the values within a template: ```go-html-template -{{ os.Getenv "MY_VAR1" }} → foo -{{ os.Getenv "MY_VAR2" }} → bar +{{ getenv "MY_VAR1" }} → foo +{{ getenv "MY_VAR2" }} → bar ``` - -With Hugo v0.91.0 and later, you must explicitly allow access to environment variables. For details, review [Hugo's Security Policy](/about/security-model/#security-policy). By default, environment variables beginning with `HUGO_` are allowed when using the `os.Getenv` function. diff --git a/content/en/functions/os/ReadDir.md b/content/en/functions/os/ReadDir.md index d0ed87bdfea..63af850b7e7 100644 --- a/content/en/functions/os/ReadDir.md +++ b/content/en/functions/os/ReadDir.md @@ -1,22 +1,17 @@ --- title: os.ReadDir -linkTitle: readDir description: Returns an array of FileInfo structures sorted by file name, one element for each directory entry. -categories: [functions] +categories: [] keywords: [] -menu: - docs: - parent: functions -function: +action: aliases: [readDir] - returnType: FileInfo + related: + - functions/os/FileExists + - functions/os/Getenv + - functions/os/ReadFile + - functions/os/Stat + returnType: os.FileInfo signatures: [os.ReadDir PATH] -relatedFunctions: - - os.FileExists - - os.Getenv - - os.ReadDir - - os.ReadFile - - os.Stat aliases: [/functions/readdir] --- @@ -36,7 +31,7 @@ content/ This template code: ```go-html-template -{{ range os.ReadDir "content" }} +{{ range readDir "content" }} {{ .Name }} → {{ .IsDir }} {{ end }} ``` diff --git a/content/en/functions/os/ReadFile.md b/content/en/functions/os/ReadFile.md index 30f2b305691..654e300ac88 100644 --- a/content/en/functions/os/ReadFile.md +++ b/content/en/functions/os/ReadFile.md @@ -1,22 +1,17 @@ --- title: os.ReadFile -linkTitle: readFile description: Returns the contents of a file. -categories: [functions] +categories: [] keywords: [] -menu: - docs: - parent: functions -function: +action: aliases: [readFile] + related: + - functions/os/FileExists + - functions/os/Getenv + - functions/os/ReadDir + - functions/os/Stat returnType: string signatures: [os.ReadFile PATH] -relatedFunctions: - - os.FileExists - - os.Getenv - - os.ReadDir - - os.ReadFile - - os.Stat aliases: [/functions/readfile] --- @@ -31,7 +26,7 @@ This is **bold** text. This template code: ```go-html-template -{{ os.ReadFile "README.md" }} +{{ readFile "README.md" }} ``` Produces: diff --git a/content/en/functions/os/Stat.md b/content/en/functions/os/Stat.md index dfef3c8151d..6b6f668de48 100644 --- a/content/en/functions/os/Stat.md +++ b/content/en/functions/os/Stat.md @@ -1,21 +1,17 @@ --- title: os.Stat description: Returns a FileInfo structure describing a file or directory. -categories: [functions] +categories: [] keywords: [] -menu: - docs: - parent: functions -function: +action: aliases: [] - returnType: FileInfo + related: + - functions/os/FileExists + - functions/os/Getenv + - functions/os/ReadDir + - functions/os/ReadFile + returnType: os.FileInfo signatures: [os.Stat PATH] -relatedFunctions: - - os.FileExists - - os.Getenv - - os.ReadDir - - os.ReadFile - - os.Stat aliases: [/functions/os.stat] --- diff --git a/content/en/functions/os/_index.md b/content/en/functions/os/_index.md new file mode 100644 index 00000000000..c080d00925a --- /dev/null +++ b/content/en/functions/os/_index.md @@ -0,0 +1,12 @@ +--- +title: OS functions +linkTitle: os +description: Template functions to interact with the operating system. +categories: [] +keywords: [] +menu: + docs: + parent: functions +--- + +Use these functions to interact with the operating system. diff --git a/content/en/functions/partials/Include.md b/content/en/functions/partials/Include.md index ea9dfb31aef..e9ed093d895 100644 --- a/content/en/functions/partials/Include.md +++ b/content/en/functions/partials/Include.md @@ -1,19 +1,16 @@ --- title: partials.Include -linkTitle: partial -description: Executes the named partial template. If the partial contains a return statement, returns that value, else returns the rendered output. -categories: [functions] +description: Executes the given partial template, optionally passing context. If the partial contains a return statement, returns that value, else returns the rendered output. +categories: [] keywords: [] -menu: - docs: - parent: functions -function: +action: aliases: [partial] + related: + - functions/go-template/template + - functions/partials/IncludeCached + - methods/page/Render returnType: any - signatures: ['partials.Include LAYOUT [CONTEXT]'] -relatedFunctions: - - partials.Include - - partials.IncludeCached + signatures: ['partials.Include NAME [CONTEXT]'] aliases: [/functions/partial] --- @@ -63,5 +60,4 @@ Then, within the partial template:

{{ .name }} is majoring in {{ .major }}. Their grade point average is {{ .gpa }}.

``` - [breadcrumb navigation]: /content-management/sections/#ancestors-and-descendants diff --git a/content/en/functions/partials/IncludeCached.md b/content/en/functions/partials/IncludeCached.md index ab9a77835cb..4067eb9051f 100644 --- a/content/en/functions/partials/IncludeCached.md +++ b/content/en/functions/partials/IncludeCached.md @@ -1,22 +1,19 @@ --- title: partials.IncludeCached -linkTitle: partialCached -description: Allows for caching of partials that do not need to be re-rendered on every invocation. -categories: [functions] +description: Executes the given template and caches the result, optionally passing context. If the partial contains a return statement, returns that value, else returns the rendered output. +categories: [] keywords: [] -menu: - docs: - parent: functions -function: +action: aliases: [partialCached] + related: + - functions/go-template/template + - functions/partials/Include + - methods/page/Render returnType: any signatures: ['partials.IncludeCached LAYOUT CONTEXT [VARIANT...]'] -relatedFunctions: - - partials.Include - - partials.IncludeCached signatures: - - partials.IncludeCached LAYOUT CONTEXT [VARIANT...] - - partialCached LAYOUT CONTEXT [VARIANT...] + - partials.IncludeCached NAME CONTEXT [VARIANT...] + - partialCached NAME CONTEXT [VARIANT...] aliases: [/functions/partialcached] --- diff --git a/content/en/functions/partials/_index.md b/content/en/functions/partials/_index.md new file mode 100644 index 00000000000..0a7d4d6d016 --- /dev/null +++ b/content/en/functions/partials/_index.md @@ -0,0 +1,12 @@ +--- +title: Partial functions +linkTitle: partials +description: Template functions to call partial templates. +categories: [] +keywords: [] +menu: + docs: + parent: functions +--- + +Use these functions to call partial templates. diff --git a/content/en/functions/path/Base.md b/content/en/functions/path/Base.md index c008735243c..2071f8bea71 100644 --- a/content/en/functions/path/Base.md +++ b/content/en/functions/path/Base.md @@ -1,35 +1,26 @@ --- title: path.Base -description: Base returns the last element of a path. -categories: [functions] +description: Replaces path separators with slashes (`/`) and returns the last element of the given path. +categories: [] keywords: [] -menu: - docs: - parent: functions -function: +action: aliases: [] + related: + - functions/path/BaseName + - functions/path/Clean + - functions/path/Dir + - functions/path/Ext + - functions/path/Join + - functions/path/Split returnType: string signatures: [path.Base PATH] -relatedFunctions: - - path.Base - - path.BaseName - - path.Clean - - path.Dir - - path.Ext - - path.Join - - path.Split aliases: [/functions/path.base] --- -`path.Base` returns the last element of `PATH`. - -If `PATH` is empty, `.` is returned. - -**Note:** On Windows, `PATH` is converted to slash (`/`) separators. - ```go-html-template -{{ path.Base "a/news.html" }} → "news.html" -{{ path.Base "news.html" }} → "news.html" -{{ path.Base "a/b/c" }} → "c" -{{ path.Base "/x/y/z/" }} → "z" +{{ path.Base "a/news.html" }} → news.html +{{ path.Base "news.html" }} → news.html +{{ path.Base "a/b/c" }} → c +{{ path.Base "/x/y/z/" }} → z +{{ path.Base "" }} → . ``` diff --git a/content/en/functions/path/BaseName.md b/content/en/functions/path/BaseName.md index a357ed4037d..477c8d31960 100644 --- a/content/en/functions/path/BaseName.md +++ b/content/en/functions/path/BaseName.md @@ -1,33 +1,26 @@ --- title: path.BaseName -description: BaseName returns the last element of a path, removing the extension if present. -categories: [functions] +description: Replaces path separators with slashes (`/`) and returns the last element of the given path, removing the extension if present. +categories: [] keywords: [] -menu: - docs: - parent: functions -function: +action: aliases: [] + related: + - functions/path/Base + - functions/path/Clean + - functions/path/Dir + - functions/path/Ext + - functions/path/Join + - functions/path/Split returnType: string signatures: [path.BaseName PATH] -relatedFunctions: - - path.Base - - path.BaseName - - path.Clean - - path.Dir - - path.Ext - - path.Join - - path.Split aliases: [/functions/path.basename] --- -If `PATH` is empty, `.` is returned. - -**Note:** On Windows, `PATH` is converted to slash (`/`) separators. - ```go-html-template -{{ path.BaseName "a/news.html" }} → "news" -{{ path.BaseName "news.html" }} → "news" -{{ path.BaseName "a/b/c" }} → "c" -{{ path.BaseName "/x/y/z/" }} → "z" +{{ path.BaseName "a/news.html" }} → news +{{ path.BaseName "news.html" }} → news +{{ path.BaseName "a/b/c" }} → c +{{ path.BaseName "/x/y/z/" }} → z +{{ path.BaseName "" }} → . ``` diff --git a/content/en/functions/path/Clean.md b/content/en/functions/path/Clean.md index 98160c56865..57a665c03a1 100644 --- a/content/en/functions/path/Clean.md +++ b/content/en/functions/path/Clean.md @@ -1,35 +1,33 @@ --- title: path.Clean -description: Replaces path separators with slashes (`/`) and removes extraneous separators. -categories: [functions] +description: Replaces path separators with slashes (`/`) and returns the shortest path name equivalent to the given path. +categories: [] keywords: [] -menu: - docs: - parent: functions -function: +action: aliases: [] + related: + - functions/path/Base + - functions/path/BaseName + - functions/path/Dir + - functions/path/Ext + - functions/path/Join + - functions/path/Split returnType: string signatures: [path.Clean PATH] -relatedFunctions: - - path.Base - - path.BaseName - - path.Clean - - path.Dir - - path.Ext - - path.Join - - path.Split aliases: [/functions/path.clean] --- -`path.Clean` replaces path separators with slashes (`/`) and removes extraneous separators, including trailing separators. +See Go's [`path.Clean`] documentation for details. -```go-html-template -{{ path.Clean "foo//bar" }} → "foo/bar" -{{ path.Clean "/foo/bar/" }} → "/foo/bar" -``` - -On a Windows system, if `.File.Path` is `foo\bar.md`, then: +[`path.Clean`]: https://pkg.go.dev/path#Clean ```go-html-template -{{ path.Clean .File.Path }} → "foo/bar.md" +{{ path.Clean "foo/bar" }} → foo/bar +{{ path.Clean "/foo/bar" }} → /foo/bar +{{ path.Clean "/foo/bar/" }} → /foo/bar +{{ path.Clean "/foo//bar/" }} → /foo/bar +{{ path.Clean "/foo/./bar/" }} → /foo/bar +{{ path.Clean "/foo/../bar/" }} → /bar +{{ path.Clean "/../foo/../bar/" }} → /bar +{{ path.Clean "" }} → . ``` diff --git a/content/en/functions/path/Dir.md b/content/en/functions/path/Dir.md index 0a292869673..6d5e5c97528 100644 --- a/content/en/functions/path/Dir.md +++ b/content/en/functions/path/Dir.md @@ -1,36 +1,27 @@ --- title: path.Dir -description: Dir returns all but the last element of a path. -categories: [functions] +description: Replaces path separators with slashes (/) and returns all but the last element of the given path. +categories: [] keywords: [] -menu: - docs: - parent: functions -function: +action: aliases: [] + related: + - functions/path/Base + - functions/path/BaseName + - functions/path/Clean + - functions/path/Ext + - functions/path/Join + - functions/path/Split returnType: string signatures: [path.Dir PATH] -relatedFunctions: - - path.Base - - path.BaseName - - path.Clean - - path.Dir - - path.Ext - - path.Join - - path.Split aliases: [/functions/path.dir] --- -`path.Dir` returns all but the last element of `PATH`, typically `PATH`'s directory. - -The returned path will never end in a slash. -If `PATH` is empty, `.` is returned. - -**Note:** On Windows, `PATH` is converted to slash (`/`) separators. - ```go-html-template -{{ path.Dir "a/news.html" }} → "a" -{{ path.Dir "news.html" }} → "." -{{ path.Dir "a/b/c" }} → "a/b" -{{ path.Dir "/x/y/z" }} → "/x/y" +{{ path.Dir "a/news.html" }} → a +{{ path.Dir "news.html" }} → . +{{ path.Dir "a/b/c" }} → a/b +{{ path.Dir "/a/b/c" }} → /a/b +{{ path.Dir "/a/b/c/" }} → /a/b/c +{{ path.Dir "" }} → . ``` diff --git a/content/en/functions/path/Ext.md b/content/en/functions/path/Ext.md index 6b568594841..f3e47aecd77 100644 --- a/content/en/functions/path/Ext.md +++ b/content/en/functions/path/Ext.md @@ -1,33 +1,24 @@ --- title: path.Ext -description: Ext returns the file name extension of a path. -categories: [functions] +description: Replaces path separators with slashes (`/`) and returns the file name extension of the given path. +categories: [] keywords: [] -menu: - docs: - parent: functions -function: +action: aliases: [] + related: + - functions/path/Base + - functions/path/BaseName + - functions/path/Clean + - functions/path/Dir + - functions/path/Join + - functions/path/Split returnType: string signatures: [path.Ext PATH] -relatedFunctions: - - path.Base - - path.BaseName - - path.Clean - - path.Dir - - path.Ext - - path.Join - - path.Split aliases: [/functions/path.ext] --- -`path.Ext` returns the file name extension `PATH`. - -The extension is the suffix beginning at the final dot in the final slash-separated element `PATH`; -it is empty if there is no dot. - -**Note:** On Windows, `PATH` is converted to slash (`/`) separators. +The extension is the suffix beginning at the final dot in the final slash-separated element of path; it is empty if there is no dot. ```go-html-template -{{ path.Ext "a/b/c/news.html" }} → ".html" +{{ path.Ext "a/b/c/news.html" }} → .html ``` diff --git a/content/en/functions/path/Join.md b/content/en/functions/path/Join.md index 4f5c51c0f39..dc701b7314e 100644 --- a/content/en/functions/path/Join.md +++ b/content/en/functions/path/Join.md @@ -1,34 +1,36 @@ --- title: path.Join -description: Join path elements into a single path. -categories: [functions] +description: Replaces path separators with slashes (`/`), joins the given path elements into a single path, and returns the shortest path name equivalent to the result. +categories: [] keywords: [] -menu: - docs: - parent: functions -function: +action: aliases: [] + related: + - functions/path/Base + - functions/path/BaseName + - functions/path/Clean + - functions/path/Dir + - functions/path/Ext + - functions/path/Split + - functions/urls/JoinPath returnType: string signatures: [path.Join ELEMENT...] -relatedFunctions: - - path.Base - - path.BaseName - - path.Clean - - path.Dir - - path.Ext - - path.Join - - path.Split - - urls.JoinPath aliases: [/functions/path.join] --- -`path.Join` joins path elements into a single path, adding a separating slash if necessary. -All empty strings are ignored. +See Go's [`path.Join`] and [`path.Clean`] documentation for details. + +[`path.Clean`]: https://pkg.go.dev/path#Clean +[`path.Join`]: https://pkg.go.dev/path#Join -**Note:** All path elements on Windows are converted to slash ('/') separators. ```go-html-template -{{ path.Join "partial" "news.html" }} → "partial/news.html" -{{ path.Join "partial/" "news.html" }} → "partial/news.html" -{{ path.Join "foo/baz" "bar" }} → "foo/baz/bar" +{{ path.Join "partial" "news.html" }} → partial/news.html +{{ path.Join "partial/" "news.html" }} → partial/news.html +{{ path.Join "foo/bar" "baz" }} → foo/bar/baz +{{ path.Join "foo" "bar" "baz" }} → foo/bar/baz +{{ path.Join "foo" "" "baz" }} → foo/baz +{{ path.Join "foo" "." "baz" }} → foo/baz +{{ path.Join "foo" ".." "baz" }} → baz +{{ path.Join "/.." "foo" ".." "baz" }} → baz ``` diff --git a/content/en/functions/path/Split.md b/content/en/functions/path/Split.md index bde412743e4..329d73954a2 100644 --- a/content/en/functions/path/Split.md +++ b/content/en/functions/path/Split.md @@ -1,43 +1,34 @@ --- title: path.Split -description: Split path immediately following the final slash. -categories: [functions] +description: Replaces path separators with slashes (`/`) and splits the resulting path immediately following the final slash, separating it into a directory and file name component. +categories: [] keywords: [] -menu: - docs: - parent: functions -function: +action: aliases: [] - returnType: DirFile + related: + - functions/path/Base + - functions/path/BaseName + - functions/path/Clean + - functions/path/Dir + - functions/path/Ext + - functions/path/Join + returnType: paths.DirFile signatures: [path.Split PATH] -relatedFunctions: - - path.Base - - path.BaseName - - path.Clean - - path.Dir - - path.Ext - - path.Join - - path.Split aliases: [/functions/path.split] --- -`path.Split` splits `PATH` immediately following the final slash, separating it into a directory and a base component. - -The returned values have the property that `PATH` = `DIR`+`BASE`. -If there is no slash in `PATH`, it returns an empty directory and the base is set to `PATH`. - -**Note:** On Windows, `PATH` is converted to slash (`/`) separators. +If there is no slash in the given path, `path.Split` returns an empty directory, and file set to path. The returned values have the property that path = dir+file. ```go-html-template {{ $dirFile := path.Split "a/news.html" }} -{{ $dirFile.Dir }} → "a/" -{{ $dirFile.File }} → "news.html" +{{ $dirFile.Dir }} → a/ +{{ $dirFile.File }} → news.html {{ $dirFile := path.Split "news.html" }} -{{ $dirFile.Dir }} → "" -{{ $dirFile.File }} → "news.html" +{{ $dirFile.Dir }} → "" (empty string) +{{ $dirFile.File }} → news.html {{ $dirFile := path.Split "a/b/c" }} -{{ $dirFile.Dir }} → "a/b/" -{{ $dirFile.File }} → "c" +{{ $dirFile.Dir }} → a/b/ +{{ $dirFile.File }} → c ``` diff --git a/content/en/functions/path/_index.md b/content/en/functions/path/_index.md new file mode 100644 index 00000000000..2d7ce8e90ec --- /dev/null +++ b/content/en/functions/path/_index.md @@ -0,0 +1,12 @@ +--- +title: Path functions +linkTitle: path +description: Template functions to work with file paths. +categories: [] +keywords: [] +menu: + docs: + parent: functions +--- + +Use these functions to work with file paths. diff --git a/content/en/functions/reflect/IsMap.md b/content/en/functions/reflect/IsMap.md index 7d89bb38aac..c2c441b5c01 100644 --- a/content/en/functions/reflect/IsMap.md +++ b/content/en/functions/reflect/IsMap.md @@ -1,19 +1,14 @@ --- title: reflect.IsMap description: Reports whether the value is a map. -categories: [functions] +categories: [] keywords: [] -menu: - docs: - parent: functions -function: +action: aliases: [] + related: + - functions/reflect/IsSlice returnType: bool signatures: [reflect.IsMap INPUT] -namespace: reflect -relatedFunctions: - - reflect.IsMap - - reflect.IsSlice aliases: [/functions/reflect.ismap] --- diff --git a/content/en/functions/reflect/IsSlice.md b/content/en/functions/reflect/IsSlice.md index 09bab7127e3..13ab6e519ea 100644 --- a/content/en/functions/reflect/IsSlice.md +++ b/content/en/functions/reflect/IsSlice.md @@ -1,19 +1,14 @@ --- title: reflect.IsSlice description: Reports whether the value is a slice. -categories: [functions] +categories: [] keywords: [] -menu: - docs: - parent: functions -function: +action: aliases: [] + related: + - functions/reflect/IsMap returnType: bool signatures: [reflect.IsSlice INPUT] -namespace: reflect -relatedFunctions: - - reflect.IsMap - - reflect.IsSlice aliases: [/functions/reflect.isslice] --- diff --git a/content/en/functions/reflect/_index.md b/content/en/functions/reflect/_index.md new file mode 100644 index 00000000000..711908ee2b9 --- /dev/null +++ b/content/en/functions/reflect/_index.md @@ -0,0 +1,12 @@ +--- +title: Reflect functions +linkTitle: reflect +description: Template functions to determine a value's data type. +categories: [] +keywords: [] +menu: + docs: + parent: functions +--- + +Use these functions to determine a value's data type. diff --git a/content/en/functions/resources/Babel.md b/content/en/functions/resources/Babel.md new file mode 100644 index 00000000000..4a6c64a75a6 --- /dev/null +++ b/content/en/functions/resources/Babel.md @@ -0,0 +1,88 @@ +--- +title: resources.Babel +description: Compiles the given JavaScript resource with Babel. +categories: [] +keywords: [] +action: + aliases: [babel] + related: + - functions/js/Build + - functions/resources/Fingerprint + - functions/resources/Minify + returnType: resource.Resource + signatures: ['resources.Babel [OPTIONS] RESOURCE'] +toc: true +--- + +```go-html-template +{{ with resources.Get "js/main.js" }} + {{ if eq hugo.Environment "development" }} + {{ with . | babel }} + + {{ end }} + {{ else }} + {{ $opts := dict "minified" true }} + {{ with . | babel $opts | fingerprint }} + + {{ end }} + {{ end }} +{{ end }} +``` + +## Setup + +Step 1 +: Install [Node.js](https://nodejs.org/en/download) + +Step 2 +: Install the required Node.js packages in the root of your project. + +```sh +npm install --save-dev @babel/core @babel/cli +``` + +Step 3 +: Add the babel executable to Hugo's `security.exec.allow` list in your site configuration: + +{{< code-toggle file=hugo >}} +[security.exec] + allow = ['^(dart-)?sass(-embedded)?$', '^go$', '^npx$', '^postcss$', '^babel$'] +{{< /code-toggle >}} + +## Configuration + +We add the main project's `node_modules` to `NODE_PATH` when running Babel and similar tools. There are some known [issues](https://github.com/babel/babel/issues/5618) with Babel in this area, so if you have a `babel.config.js` living in a Hugo Module (and not in the project itself), we recommend using `require` to load the presets/plugins, e.g.: + +```js +module.exports = { + presets: [ + [ + require("@babel/preset-env"), + { + useBuiltIns: "entry", + corejs: 3, + }, + ], + ], +}; +``` + +## Options + +config +: (`string`) Path to the Babel configuration file. Hugo will, by default, look for a `babel.config.js` in your project. More information on these configuration files can be found here: [babel configuration](https://babeljs.io/docs/en/configuration). + +minified +: (`bool`) Save as many bytes as possible when printing + +noComments +: (`bool`) Write comments to generated output (true by default) + +compact +: (`bool`) Do not include superfluous whitespace characters and line terminators. Defaults to `auto` if not set. + +verbose +: (`bool`) Log everything + +sourceMap +: (`string`) Output `inline` or `external` sourcemap from the babel compile. External sourcemaps will be written to the target with the output file name + ".map". Input sourcemaps can be read from js.Build and node modules and combined into the output sourcemaps. diff --git a/content/en/functions/resources/ByType.md b/content/en/functions/resources/ByType.md new file mode 100644 index 00000000000..a5df3befb83 --- /dev/null +++ b/content/en/functions/resources/ByType.md @@ -0,0 +1,34 @@ +--- +title: resources.ByType +description: Returns a collection of global resources of the given media type, or nil if none found. +categories: [] +keywords: [] +action: + aliases: [] + related: + - functions/resources/Get + - functions/resources/GetMatch + - functions/resources/GetRemote + - functions/resources/Match + - methods/page/Resources + returnType: resource.Resources + signatures: [resources.ByType MEDIATYPE] +--- + +The [media type] is typically one of `image`, `text`, `audio`, `video`, or `application`. + +```go-html-template +{{ range resources.ByType "image" }} + +{{ end }} +``` + +{{% note %}} +This function operates on global resources. A global resource is a file within the assets directory, or within any directory mounted to the assets directory. + +For page resources, use the [`Resources.ByType`] method on the Page object. + +[`Resources.ByType`]: /methods/page/resources +{{% /note %}} + +[media type]: https://en.wikipedia.org/wiki/Media_type diff --git a/content/en/functions/resources/Concat.md b/content/en/functions/resources/Concat.md new file mode 100644 index 00000000000..3bdf975bf9a --- /dev/null +++ b/content/en/functions/resources/Concat.md @@ -0,0 +1,21 @@ +--- +title: resources.Concat +description: Concatenates a slice of resources. +categories: [] +keywords: [] +action: + aliases: [] + related: [] + returnType: resource.Resource + signatures: ['resources.Concat TARGETPATH [RESOURCE...]'] +--- + +```go-html-template +{{ $plugins := resources.Get "js/plugins.js" }} +{{ $global := resources.Get "js/global.js" }} +{{ $js := slice $plugins $global | resources.Concat "js/bundle.js" }} +``` + +Asset files of the same [media type] can be bundled into one resource using the `resources.Concat` function which takes two arguments, the target path for the created resource bundle and a slice of resource objects to be concatenated. + +[media type]: https://en.wikipedia.org/wiki/Media_type diff --git a/content/en/functions/resources/Copy.md b/content/en/functions/resources/Copy.md new file mode 100644 index 00000000000..9df88b0a0b6 --- /dev/null +++ b/content/en/functions/resources/Copy.md @@ -0,0 +1,30 @@ +--- +title: resources.Copy +description: Copies the given resource to the target path. +categories: [] +action: + aliases: [] + related: [] + returnType: resource.Resource + signatures: [resources.Copy TARGETPATH RESOURCE] +--- + +```go-html-template +{{ with resources.Get "images/a.jpg" }} + {{ with resources.Copy "img/new-image-name.jpg" . }} + + {{ end }} +{{ end }} +``` + +The relative URL of the new published resource will be: + +```text +/img/new-image-name.jpg +``` + +The target path must be different than the source path, as shown in the example above. + +{{% note %}} +Use the `resources.Copy` function with global, page, and remote resources. +{{% /note %}} diff --git a/content/en/functions/resources/ExecuteAsTemplate.md b/content/en/functions/resources/ExecuteAsTemplate.md new file mode 100644 index 00000000000..815cbe49017 --- /dev/null +++ b/content/en/functions/resources/ExecuteAsTemplate.md @@ -0,0 +1,56 @@ +--- +title: resources.ExecuteAsTemplate +description: Creates a resource from a Go template, parsed and executed with the given context. +categories: [] +keywords: [] +action: + aliases: [] + related: + - functions/resources/FromString + returnType: resource.Resource + signatures: [resources.ExecuteAsTemplate TARGETPATH CONTEXT RESOURCE] +--- + +Hugo publishes the resource to the target path when you call its`.Publish`, `.Permalink`, or `.RelPermalink` method. The resource is cached, using the target path as the cache key. + +Let's say you have a CSS file that you wish to populate with values from your site configuration: + +{{< code lang=go-html-template file=assets/css/template.css >}} +body { + background-color: {{ site.Params.style.bg_color }}; + color: {{ site.Params.style.text_color }}; +} +{{< /code >}} + +And your site configuration contains: + +{{< code-toggle file=hugo >}} +[params.style] +bg_color = '#fefefe' +text_color = '#222' +{{< /code-toggle >}} + +Place this in your baseof.html template: + +```go-html-template +{{ with resources.Get "css/template.css" }} + {{ with resources.ExecuteAsTemplate "css/main.css" $ . }} + + {{ end }} +{{ end }} +``` + +The example above: + +1. Captures the template as a resource +2. Executes the resource as a template, passing the current page in context +3. Publishes the resource to css/main.css + +The result is: + +{{< code file="public/css/main.css" >}} +body { + background-color: #fefefe; + color: #222; +} +{{< /code >}} diff --git a/content/en/functions/resources/Fingerprint.md b/content/en/functions/resources/Fingerprint.md new file mode 100644 index 00000000000..685214f96ce --- /dev/null +++ b/content/en/functions/resources/Fingerprint.md @@ -0,0 +1,42 @@ +--- +title: resources.Fingerprint +description: Cryptographically hashes the content of the given resource. +categories: [] +keywords: [] +action: + aliases: [fingerprint] + related: + - functions/js/Build + - functions/resources/Babel + - functions/resources/Minify + - functions/resources/PostCSS + - functions/resources/PostProcess + - functions/resources/ToCSS + returnType: resource.Resource + signatures: ['resources.Fingerprint [ALGORITHM] RESOURCE'] +--- + +```go-html-template +{{ with resources.Get "js/main.js" }} + {{ with . | fingerprint "sha256" }} + + {{ end }} +{{ end }} +``` + +Hugo renders this to something like: + +```html + +``` + +Although most commonly used with CSS and JavaScript resources, you can use the `resources.Fingerprint` function with any resource type. + +The hash algorithm may be one of `md5`, `sha256` (default), `sha384`, or `sha512`. + +After cryptographically hashing the resource content: + +1. The values returned by the `.Permalink` and `.RelPermalink` methods include the hash sum +2. The resource's `.Data.Integrity` method returns a [Subresource Integrity] (SRI) value consisting of the name of the hash algorithm, one hyphen, and the base64-encoded hash sum + +[Subresource Integrity]: https://developer.mozilla.org/en-US/docs/Web/Security/Subresource_Integrity diff --git a/content/en/functions/resources/FromString.md b/content/en/functions/resources/FromString.md new file mode 100644 index 00000000000..6c22b53104c --- /dev/null +++ b/content/en/functions/resources/FromString.md @@ -0,0 +1,71 @@ +--- +title: resources.FromString +description: Creates a resource from a string. +categories: [] +keywords: [] +action: + aliases: [] + related: + - functions/resources/ExecuteAsTemplate + returnType: resource.Resource + signatures: [resources.FromString TARGETPATH STRING] +--- + +Hugo publishes the resource to the target path when you call its`.Publish`, `.Permalink`, or `.RelPermalink` method. The resource is cached, using the target path as the cache key. + +Let's say you need to publish a file named "site.json" in the root of your public directory, containing the build date, the Hugo version used to build the site, and the date that the content was last modified. For example: + +```json +{ + "build_date": "2023-10-03T10:50:40-07:00", + "hugo_version": "0.120.0", + "last_modified": "2023-10-02T15:21:27-07:00" +} +``` + +Place this in your baseof.html template: + +```go-html-template +{{ if .IsHome }} + {{ $rfc3339 := "2006-01-02T15:04:05Z07:00" }} + {{ $m := dict + "hugo_version" hugo.Version + "build_date" (now.Format $rfc3339) + "last_modified" (site.LastChange.Format $rfc3339) + }} + {{ $json := jsonify $m }} + {{ $r := resources.FromString "site.json" $json }} + {{ $r.Publish }} +{{ end }} +``` + +The example above: + +1. Creates a map with the relevant key/value pairs using the [`dict`] function +2. Encodes the map as a JSON string using the [`jsonify`] function +3. Creates a resource from the JSON string using the `resources.FromString` function +4. Publishes the file to the root of the public directory using the resource's `.Publish` method + +Combine `resources.FromString` with [`resources.ExecuteAsTemplate`] if your string contains template actions. Rewriting the example above: + +```go-html-template +{{ if .IsHome }} + {{ $string := ` + {{ $rfc3339 := "2006-01-02T15:04:05Z07:00" }} + {{ $m := dict + "hugo_version" hugo.Version + "build_date" (now.Format $rfc3339) + "last_modified" (site.LastChange.Format $rfc3339) + }} + {{ $json := jsonify $m }} + ` + }} + {{ $r := resources.FromString "" $string }} + {{ $r = $r | resources.ExecuteAsTemplate "site.json" . }} + {{ $r.Publish }} +{{ end }} +``` + +[`dict`]: /functions/collections/dictionary +[`jsonify`]: /functions/encoding/jsonify +[`resources.ExecuteAsTemplate`]: /functions/resources/executeastemplate diff --git a/content/en/functions/resources/Get.md b/content/en/functions/resources/Get.md new file mode 100644 index 00000000000..a8b75d52b4d --- /dev/null +++ b/content/en/functions/resources/Get.md @@ -0,0 +1,30 @@ +--- +title: resources.Get +description: Returns a global resource from the given path, or nil if none found. +categories: [] +keywords: [] +action: + aliases: [] + related: + - functions/resources/ByType + - functions/resources/GetMatch + - functions/resources/GetRemote + - functions/resources/Match + - methods/page/Resources + returnType: resource.Resource + signatures: [resources.Get PATH] +--- + +```go-html-template +{{ with resources.Get "images/a.jpg" }} + +{{ end }} +``` + +{{% note %}} +This function operates on global resources. A global resource is a file within the assets directory, or within any directory mounted to the assets directory. + +For page resources, use the [`Resources.Get`] method on the Page object. + +[`Resources.Get`]: /methods/page/resources +{{% /note %}} diff --git a/content/en/functions/resources/GetMatch.md b/content/en/functions/resources/GetMatch.md new file mode 100644 index 00000000000..fde26c09db2 --- /dev/null +++ b/content/en/functions/resources/GetMatch.md @@ -0,0 +1,36 @@ +--- +title: resources.GetMatch +description: Returns the first global resource from paths matching the given glob pattern, or nil if none found. +categories: [] +keywords: [] +action: + aliases: [] + related: + - functions/resources/ByType + - functions/resources/Get + - functions/resources/GetRemote + - functions/resources/Match + - methods/page/Resources + returnType: resource.Resource + signatures: [resources.GetMatch PATTERN] +--- + +```go-html-template +{{ with resources.GetMatch "images/*.jpg" }} + +{{ end }} +``` + +{{% note %}} +This function operates on global resources. A global resource is a file within the assets directory, or within any directory mounted to the assets directory. + +For page resources, use the [`Resources.GetMatch`] method on the Page object. + +[`Resources.GetMatch`]: /methods/page/resources +{{% /note %}} + +Hugo determines a match using a case-insensitive [glob pattern]. + +{{% include "functions/_common/glob-patterns.md" %}} + +[glob pattern]: https://github.com/gobwas/glob#example diff --git a/content/en/functions/resources/GetRemote.md b/content/en/functions/resources/GetRemote.md new file mode 100644 index 00000000000..fdd639195e5 --- /dev/null +++ b/content/en/functions/resources/GetRemote.md @@ -0,0 +1,176 @@ +--- +title: resources.GetRemote +description: Returns a remote resource from the given URL, or nil if none found. +categories: [] +keywords: [] +action: + aliases: [] + related: + - functions/data/GetCSV + - functions/data/GetJSON + - functions/resources/ByType + - functions/resources/Get + - functions/resources/GetMatch + - functions/resources/Match + - methods/page/Resources + returnType: resource.Resource + signatures: ['resources.GetRemote URL [OPTIONS]'] +toc: true +--- + +```go-html-template +{{ $url := "https://example.org/images/a.jpg" }} +{{ with resources.GetRemote $url }} + {{ with .Err }} + {{ errorf "%s" . }} + {{ else }} + + {{ end }} +{{ else }} + {{ errorf "Unable to get remote resource %q" $url }} +{{ end }} +``` + +## Options + +The `resources.GetRemote` function takes an optional map of options. + +```go-html-template +{{ $url := "https://example.org/api" }} +{{ $opts := dict + "headers" (dict "Authorization" "Bearer abcd") +}} +{{ $resource := resources.GetRemote $url $opts }} +``` + +If you need multiple values for the same header key, use a slice: + +```go-html-template +{{ $url := "https://example.org/api" }} +{{ $opts := dict + "headers" (dict "X-List" (slice "a" "b" "c")) +}} +{{ $resource := resources.GetRemote $url $opts }} +``` + +You can also change the request method and set the request body: + +```go-html-template +{{ $url := "https://example.org/api" }} +{{ $opts := dict + "method" "post" + "body" `{"complete": true}` + "headers" (dict "Content-Type" "application/json") +}} +{{ $resource := resources.GetRemote $url $opts }} +``` + +## Remote data + +When retrieving remote data, use the [`transform.Unmarshal`] function to unmarshal the response to a map. + +[`transform.Unmarshal`]: /functions/transform/unmarshal + +```go-html-template +{{ $data := "" }} +{{ $url := "https://example.org/books.json" }} +{{ with resources.GetRemote $url }} + {{ with .Err }} + {{ errorf "%s" . }} + {{ else }} + {{ $data = . | transform.Unmarshal }} + {{ end }} +{{ else }} + {{ errorf "Unable to get remote resource %q" $url }} +{{ end }} +``` + +## Error handling + +The [`Err`] method on a resource returned by the `resources.GetRemote` function returns an error message if the HTTP request fails, else nil. If you do not handle the error yourself, Hugo will fail the build. + +[`Err`]: /methods/resource/err + +```go-html-template +{{ $url := "https://broken-example.org/images/a.jpg" }} +{{ with resources.GetRemote $url }} + {{ with .Err }} + {{ errorf "%s" . }} + {{ else }} + + {{ end }} +{{ else }} + {{ errorf "Unable to get remote resource %q" $url }} +{{ end }} +``` + +To log an error as a warning instead of an error: + +```go-html-template +{{ $url := "https://broken-example.org/images/a.jpg" }} +{{ with resources.GetRemote $url }} + {{ with .Err }} + {{ warnf "%s" . }} + {{ else }} + + {{ end }} +{{ else }} + {{ errorf "Unable to get remote resource %q" $url }} +{{ end }} +``` + +## HTTP response + +The [`Data`] method on a resource returned by the `resources.GetRemote` function returns information from the HTTP response. + +[`Data`]: /methods/resource/data + +```go-html-template +{{ $url := "https://example.org/images/a.jpg" }} +{{ with resources.GetRemote $url }} + {{ with .Err }} + {{ errorf "%s" . }} + {{ else }} + {{ with .Data }} + {{ .ContentLength }} → 42764 + {{ .ContentType }} → image/jpeg + {{ .Status }} → 200 OK + {{ .StatusCode }} → 200 + {{ .TransferEncoding }} → [] + {{ end }} + {{ end }} +{{ else }} + {{ errorf "Unable to get remote resource %q" $url }} +{{ end }} +``` + +ContentLength +: (`int`) The content length in bytes. + +ContentType +: (`string`) The content type. + +Status +: (`string`) The HTTP status text. + +StatusCode +: (`int`) The HTTP status code. + +TransferEncoding +: (`string`) The transfer encoding. + +## Caching + +Resources returned from `resources.GetRemote` are cached to disk. See [configure file caches] for details. + +By default, Hugo derives the cache key from the arguments passed to the function, the URL and the options map, if any. + +Override the cache key by setting a `key` in the options map. Use this approach to have more control over how often Hugo fetches a remote resource. + +```go-html-template +{{ $url := "https://example.org/images/a.jpg" }} +{{ $cacheKey := print $url (now.Format "2006-01-02") }} +{{ $resource := resource.GetRemote $url (dict "key" $cacheKey) }} +``` + +[configure file caches]: /getting-started/configuration/#configure-file-caches diff --git a/content/en/functions/resources/Match.md b/content/en/functions/resources/Match.md new file mode 100644 index 00000000000..0044351f1f0 --- /dev/null +++ b/content/en/functions/resources/Match.md @@ -0,0 +1,36 @@ +--- +title: resources.Match +description: Returns a collection of global resources from paths matching the given glob pattern, or nil if none found. +categories: [] +keywords: [] +action: + aliases: [] + related: + - functions/resources/ByType + - functions/resources/Get + - functions/resources/GetMatch + - functions/resources/GetRemote + - methods/page/Resources + returnType: resource.Resources + signatures: [resources.Match PATTERN] +--- + +```go-html-template +{{ range resources.Match "images/*.jpg" }} + +{{ end }} +``` + +{{% note %}} +This function operates on global resources. A global resource is a file within the assets directory, or within any directory mounted to the assets directory. + +For page resources, use the [`Resources.Match`] method on the Page object. + +[`Resources.Match`]: /methods/page/resources +{{% /note %}} + +Hugo determines a match using a case-insensitive [glob pattern]. + +{{% include "functions/_common/glob-patterns.md" %}} + +[glob pattern]: https://github.com/gobwas/glob#example diff --git a/content/en/functions/resources/Minify.md b/content/en/functions/resources/Minify.md new file mode 100644 index 00000000000..44f5f990b61 --- /dev/null +++ b/content/en/functions/resources/Minify.md @@ -0,0 +1,23 @@ +--- +title: resources.Minify +description: Minifies the given resource. +categories: [] +keywords: [] +action: + aliases: [minify] + related: + - functions/js/Build + - functions/resources/Babel + - functions/resources/Fingerprint + - functions/resources/PostCSS + - functions/resources/ToCSS + returnType: resource.Resource + signatures: [resources.Minify RESOURCE] +--- + +```go-html-template +{{ $css := resources.Get "css/main.css" }} +{{ $style := $css | minify }} +``` + +Any CSS, JS, JSON, HTML, SVG or XML resource can be minified using resources.Minify which takes for argument the resource object. diff --git a/content/en/functions/resources/PostCSS.md b/content/en/functions/resources/PostCSS.md new file mode 100644 index 00000000000..a9f9ed3c8d3 --- /dev/null +++ b/content/en/functions/resources/PostCSS.md @@ -0,0 +1,129 @@ +--- +title: resources.PostCSS +description: Processes the given resource with PostCSS using any PostCSS plugin. +categories: [] +keywords: [] +action: + aliases: [postCSS] + related: + - functions/resources/Fingerprint + - functions/resources/Minify + - functions/resources/PostProcess + - functions/resources/ToCSS + returnType: resource.Resource + signatures: ['resources.PostCSS [OPTIONS] RESOURCE'] +toc: true +--- + +```go-html-template +{{ with resources.Get "css/main.css" | postCSS }} + +{{ end }} +``` + +## Setup + +Follow the steps below to transform CSS using any of the available [PostCSS plugins]. + +Step 1 +: Install [Node.js]. + +Step 2 +: Install the required Node.js packages in the root of your project. For example, to add vendor prefixes to your CSS rules: + +```sh +npm i -D postcss postcss-cli autoprefixer +``` + +Step 3 +: Create a PostCSS configuration file in the root of your project. You must name this file `postcss.config.js` or another [supported file name]. For example: + +```js +module.exports = { + plugins: [ + require('autoprefixer') + ] +}; +``` + +{{% note %}} +{{% include "functions/resources/_common/postcss-windows-warning.md" %}} +{{% /note %}} + +Step 4 +: Place your CSS file within the `assets/css` directory. + +Step 5 +: Process the resource with PostCSS: + +```go-html-template +{{ with resources.Get "css/main.css" | postCSS }} + +{{ end }} +``` + +## Options + +The `resources.PostCSS` method takes an optional map of options. + +config +: (`string`) The directory that contains the PostCSS configuration file. Default is the root of the project directory. + +noMap +: (`bool`) Default is `false`. If `true`, disables inline sourcemaps. + +inlineImports +: (`bool`) Default is `false`. Enable inlining of @import statements. It does so recursively, but will only import a file once. URL imports (e.g. `@import url('https://fonts.googleapis.com/css?family=Open+Sans&display=swap');`) and imports with media queries will be ignored. Note that this import routine does not care about the CSS spec, so you can have @import anywhere in the file. Hugo will look for imports relative to the module mount and will respect theme overrides. + +skipInlineImportsNotFound +: (`bool`) Default is `false`. Before Hugo 0.99.0 when `inlineImports` was enabled and we failed to resolve an import, we logged it as a warning. We now fail the build. If you have regular CSS imports in your CSS that you want to preserve, you can either use imports with URL or media queries (Hugo does not try to resolve those) or set `skipInlineImportsNotFound` to true. + +```go-html-template +{{ $opts := dict "config" "config-directory" "noMap" true }} +{{ with resources.Get "css/main.css" | postCSS $opts }} + +{{ end }} +``` + +## No configuration file + +To avoid using a PostCSS configuration file, you can specify a minimal configuration using the options map. + +use +: (`string`) A space-delimited list of PostCSS plugins to use. + +parser +: (`string`) A custom PostCSS parser. + +stringifier +: (`string`) A custom PostCSS stringifier. + +syntax +: (`string`) Custom postcss syntax. + +```go-html-template +{{ $opts := dict "use" "autoprefixer postcss-color-alpha" }} +{{ with resources.Get "css/main.css" | postCSS $opts }} + +{{ end }} +``` + +## Check environment + +The current Hugo environment name (set by `--environment` or in configuration or OS environment) is available in the Node context, which allows constructs like this: + +```js +const autoprefixer = require('autoprefixer'); +const purgecss = require('@fullhuman/postcss-purgecss'); +module.exports = { + plugins: [ + autoprefixer, + process.env.HUGO_ENVIRONMENT !== 'development' ? purgecss : null + ] +} +``` + +[node.js]: https://nodejs.org/en/download +[postcss plugins]: https://www.postcss.parts/ +[supported file name]: https://github.com/postcss/postcss-load-config#usage +[transpile to CSS]: /functions/resources/tocss.md diff --git a/content/en/functions/resources/PostProcess.md b/content/en/functions/resources/PostProcess.md new file mode 100644 index 00000000000..6f730a7db9e --- /dev/null +++ b/content/en/functions/resources/PostProcess.md @@ -0,0 +1,160 @@ +--- +title: resources.PostProcess +description: Processes the given resource after the build. +categories: [] +keywords: [] +action: + aliases: [] + related: + - functions/resources/Fingerprint + - functions/resources/Minify + - functions/resources/PostCSS + - functions/resources/ToCSS + returnType: postpub.PostPublishedResource + signatures: [resources.PostProcess RESOURCE] +toc: true +--- + +```go-html-template +{{ with resources.Get "css/main.css" }} + {{ if eq hugo.Environment "development" }} + + {{ else }} + {{ with . | postCSS | minify | fingerprint | resources.PostProcess }} + + {{ end }} + {{ end }} +{{ end }} +``` + +Marking a resource with `resources.PostProcess` postpones transformations until the build has finished. + +Call `resources.PostProcess` when one or more of the steps in the transformation chain depends on the result of the build. + +A prime use case for this is purging unused CSS rules using the [PurgeCSS] plugin for the PostCSS Node.js package. + +## CSS Purging + +{{% note %}} +There are several ways to set up CSS purging with PostCSS in Hugo. If you have a simple project, you should consider going the simpler route and drop the use of `resources.PostProcess` and just extract keywords from the templates. See the [Tailwind documentation](https://tailwindcss.com/docs/controlling-file-size/#app) for examples. +{{% /note %}} + +Step 1 +: Install [Node.js]. + +Step 2 +: Install the required Node.js packages in the root of your project: + +```sh +npm i -D postcss postcss-cli autoprefixer @fullhuman/postcss-purgecss +``` + +Step 3 +: Create a PostCSS configuration file in the root of your project. You must name this file `postcss.config.js` or another [supported file name]. For example: + +```js +const autoprefixer = require('autoprefixer'); +const purgecss = require('@fullhuman/postcss-purgecss')({ + content: ['./hugo_stats.json'], + defaultExtractor: content => { + const els = JSON.parse(content).htmlElements; + return [ + ...(els.tags || []), + ...(els.classes || []), + ...(els.ids || []), + ]; + }, + // https://purgecss.com/safelisting.html + safelist: [] +}); + +module.exports = { + plugins: [ + autoprefixer, + process.env.HUGO_ENVIRONMENT !== 'development' ? purgecss : null + ] +}; +``` + +{{% note %}} +{{% include "functions/resources/_common/postcss-windows-warning.md" %}} +{{% /note %}} + +Step 4 +: Enable creation of the `hugo_stats.json` file when building the site. If you are only using this for the production build, consider placing it below [config/production]. + +{{< code-toggle file=hugo >}} +[build.buildStats] +enable = true +{{< /code-toggle >}} + +See the [configure build] documentation for details and options. + +Step 5 +: Place your CSS file within the `assets/css` directory. + +Step 6 +: If the current environment is not `development`, process the resource with PostCSS: + +```go-html-template +{{ with resources.Get "css/main.css" }} + {{ if eq hugo.Environment "development" }} + + {{ else }} + {{ with . | postCSS | minify | fingerprint | resources.PostProcess }} + + {{ end }} + {{ end }} +{{ end }} +``` + +## Environment variables + +Hugo passes these environment variables to PostCSS, which allows you to do something like: + +```js +process.env.HUGO_ENVIRONMENT === 'production' ? [autoprefixer] : [] +``` + +PWD +: The absolute path to the project working directory. + +HUGO_ENVIRONMENT +: The current Hugo environment, set with the `--environment` command line flag. +Default is `production` for `hugo` and `development` for `hugo server`. + +HUGO_PUBLISHDIR +: The absolute path to the publish directory (the `public` directory). Note that the value will always point to a directory on disk even when running `hugo server` in memory mode. If you write to this folder from PostCSS when running the server, you could run the server with one of these flags: + +```sh +hugo server --renderToDisk +hugo server --renderStaticToDisk +``` + +Also, Hugo will add environment variables for all files mounted below `assets/_jsconfig`. A default mount will be set up with files in the project root matching this regexp: `(babel|postcss|tailwind)\.config\.js`. + +These will get environment variables named on the form `HUGO_FILE_:filename:` where `:filename:` is all upper case with periods replaced with underscore. This allows you to do something like: + +```js +let tailwindConfig = process.env.HUGO_FILE_TAILWIND_CONFIG_JS || './tailwind.config.js'; +``` + +## Limitations + +Do not use `resources.PostProcess` when running Hugo's built-in development server. The examples above specifically prevent this by verifying that the current environment is not "development". + +The `resources.PostProcess` function only works within templates that produce HTML files. + +You cannot manipulate the values returned from the resource’s methods. For example, the `strings.ToUpper` function in this example will not work as expected: + +```go-html-template +{{ $css := resources.Get "css/main.css" }} +{{ $css = $css | resources.PostCSS | minify | fingerprint | resources.PostProcess }} +{{ $css.RelPermalink | strings.ToUpper }} +``` + +[node.js]: https://nodejs.org/en/download +[supported file name]: https://github.com/postcss/postcss-load-config#usage +[config/production]: /getting-started/configuration/#configuration-directory +[configure build]: /getting-started/configuration/#configure-build +[purgecss]: https://github.com/FullHuman/purgecss#readme diff --git a/content/en/functions/resources/ToCSS.md b/content/en/functions/resources/ToCSS.md new file mode 100644 index 00000000000..970ee72317d --- /dev/null +++ b/content/en/functions/resources/ToCSS.md @@ -0,0 +1,224 @@ +--- +title: resources.ToCSS +description: Transpiles Sass to CSS. +categories: [] +keywords: [] +action: + aliases: [toCSS] + related: + - functions/resources/Fingerprint + - functions/resources/Minify + - functions/resources/PostCSS + - functions/resources/PostProcess + returnType: resource.Resource + signatures: ['resources.ToCSS [OPTIONS] RESOURCE'] +toc: true +--- + +```go-html-template +{{ with resources.Get "sass/main.scss" }} + {{ $opts := dict "transpiler" "libsass" "targetPath" "css/style.css" }} + {{ with . | toCSS $opts }} + {{ if eq hugo.Environment "development" }} + + {{ else }} + {{ with . | minify | fingerprint }} + + {{ end }} + {{ end }} + {{ end }} +{{ end }} +``` + +Transpile Sass to CSS using the LibSass transpiler included in Hugo's extended edition, or [install Dart Sass](#dart-sass) to use the latest features of the Sass language. + +Sass has two forms of syntax: [SCSS] and [indented]. Hugo supports both. + +[scss]: https://sass-lang.com/documentation/syntax#scss +[indented]: https://sass-lang.com/documentation/syntax#the-indented-syntax + +## Options + +transpiler +: (`string`) The transpiler to use, either `libsass` (default) or `dartsass`. Hugo's extended edition includes the LibSass transpiler. To use the Dart Sass transpiler, see the [installation instructions](#dart-sass) below. + +targetPath +: (`string`) If not set, the transformed resource's target path will be the original path of the asset file with its extension replaced by `.css`. + +vars +: (`map`) A map of key/value pairs that will be available in the `hugo:vars` namespace. Useful for [initializing Sass variables from Hugo templates](https://discourse.gohugo.io/t/42053/). + +```scss +// LibSass +@import "hugo:vars"; + +// Dart Sass +@use "hugo:vars" as v; +``` + +outputStyle +: (`string`) Output styles available to LibSass include `nested` (default), `expanded`, `compact`, and `compressed`. Output styles available to Dart Sass include `expanded` (default) and `compressed`. + +precision +: (`int`) Precision of floating point math. Not applicable to Dart Sass. + +enableSourceMap +: (`bool`) If `true`, generates a source map. + +sourceMapIncludeSources +: (`bool`) If `true`, embeds sources in the generated source map. Not applicable to LibSass. + +includePaths +: (`slice`) A slice of paths, relative to the project root, that the transpiler will use when resolving `@use` and `@import` statements. + +```go-html-template +{{ $opts := dict + "transpiler" "dartsass" + "targetPath" "css/style.css" + "vars" site.Params.styles + "enableSourceMap" (not hugo.IsProduction) + "includePaths" (slice "node_modules/bootstrap/scss") +}} +{{ with resources.Get "sass/main.scss" | toCSS $opts | minify | fingerprint }} + +{{ end }} +``` + +## Dart Sass + +The extended version of Hugo includes [LibSass] to transpile Sass to CSS. In 2020, the Sass team deprecated LibSass in favor of [Dart Sass]. + +Use the latest features of the Sass language by installing Dart Sass in your development and production environments. + +### Installation overview + +Dart Sass is compatible with Hugo v0.114.0 and later. + +If you have been using Embedded Dart Sass[^1] with Hugo v0.113.0 and earlier, uninstall Embedded Dart Sass, then install Dart Sass. If you have installed both, Hugo will use Dart Sass. + +If you install Hugo as a [Snap package] there is no need to install Dart Sass. The Hugo Snap package includes Dart Sass. + +[^1]: In 2023, the Sass team deprecated Embedded Dart Sass in favor of Dart Sass. + +### Installing in a development environment + +When you install Dart Sass somewhere in your PATH, Hugo will find it. + +OS|Package manager|Site|Installation +:--|:--|:--|:-- +Linux|Homebrew|[brew.sh]|`brew install sass/sass/sass` +Linux|Snap|[snapcraft.io]|`sudo snap install dart-sass` +macOS|Homebrew|[brew.sh]|`brew install sass/sass/sass` +Windows|Chocolatey|[chocolatey.org]|`choco install sass` +Windows|Scoop|[scoop.sh]|`scoop install sass` + +You may also install [prebuilt binaries] for Linux, macOS, and Windows. + +Run `hugo env` to list the active transpilers. + +### Installing in a production environment + +For [CI/CD] deployments (e.g., GitHub Pages, GitLab Pages, Netlify, etc.) you must edit the workflow to install Dart Sass before Hugo builds the site[^2]. Some providers allow you to use one of the package managers above, or you can download and extract one of the prebuilt binaries. + +[^2]: You do not have to do this if (a) you have not modified the assets cache location, and (b) you have not set `useResourceCacheWhen` to `never` in your [site configuration], and (c) you add and commit your resources directory to your repository. + +#### GitHub Pages + +To install Dart Sass for your builds on GitHub Pages, add this step to the GitHub Pages workflow file: + +```yaml +- name: Install Dart Sass + run: sudo snap install dart-sass +``` + +If you are using GitHub Pages for the first time with your repository, GitHub provides a [starter workflow] for Hugo that includes Dart Sass. This is the simplest way to get started. + +#### GitLab Pages + +To install Dart Sass for your builds on GitLab Pages, the `.gitlab-ci.yml` file should look something like this: + +```yaml +variables: + HUGO_VERSION: 0.115.1 + DART_SASS_VERSION: 1.63.6 + GIT_DEPTH: 0 + GIT_STRATEGY: clone + GIT_SUBMODULE_STRATEGY: recursive + TZ: America/Los_Angeles +image: + name: golang:1.20-buster +pages: + script: + # Install Dart Sass + - curl -LJO https://github.com/sass/dart-sass/releases/download/${DART_SASS_VERSION}/dart-sass-${DART_SASS_VERSION}-linux-x64.tar.gz + - tar -xf dart-sass-${DART_SASS_VERSION}-linux-x64.tar.gz + - cp -r dart-sass/* /usr/local/bin + - rm -rf dart-sass* + # Install Hugo + - curl -LJO https://github.com/gohugoio/hugo/releases/download/v${HUGO_VERSION}/hugo_extended_${HUGO_VERSION}_linux-amd64.deb + - apt install -y ./hugo_extended_${HUGO_VERSION}_linux-amd64.deb + - rm hugo_extended_${HUGO_VERSION}_linux-amd64.deb + # Build + - hugo --gc --minify + artifacts: + paths: + - public + rules: + - if: $CI_COMMIT_BRANCH == $CI_DEFAULT_BRANCH +``` + +#### Netlify + +To install Dart Sass for your builds on Netlify, the `netlify.toml` file should look something like this: + +```toml +[build.environment] +HUGO_VERSION = "0.115.1" +DART_SASS_VERSION = "1.63.6" +TZ = "America/Los_Angeles" + +[build] +publish = "public" +command = """\ + curl -LJO https://github.com/sass/dart-sass/releases/download/${DART_SASS_VERSION}/dart-sass-${DART_SASS_VERSION}-linux-x64.tar.gz && \ + tar -xf dart-sass-${DART_SASS_VERSION}-linux-x64.tar.gz && \ + rm dart-sass-${DART_SASS_VERSION}-linux-x64.tar.gz && \ + export PATH=/opt/build/repo/dart-sass:$PATH && \ + hugo --gc --minify \ + """ +``` + +### Example + +To transpile with Dart Sass, set `transpiler` to `dartsass` in the options map passed to `resources.ToCSS`. For example: + +```go-html-template +{{ with resources.Get "sass/main.scss" }} + {{ $opts := dict "transpiler" "dartsass" "targetPath" "css/style.css" }} + {{ with . | toCSS $opts }} + {{ if eq hugo.Environment "development" }} + + {{ else }} + {{ with . | minify | fingerprint }} + + {{ end }} + {{ end }} + {{ end }} +{{ end }} +``` + +### Miscellaneous + +If you build Hugo from source and run `mage test -v`, the test will fail if you install Dart Sass as a Snap package. This is due to the Snap package's strict confinement model. + +[brew.sh]: https://brew.sh/ +[chocolatey.org]: https://community.chocolatey.org/packages/sass +[ci/cd]: https://en.wikipedia.org/wiki/CI/CD +[dart sass]: https://sass-lang.com/dart-sass +[libsass]: https://sass-lang.com/libsass +[prebuilt binaries]: https://github.com/sass/dart-sass/releases/latest +[scoop.sh]: https://scoop.sh/#/apps?q=sass +[site configuration]: /getting-started/configuration/#configure-build +[snap package]: /installation/linux/#snap +[snapcraft.io]: https://snapcraft.io/dart-sass +[starter workflow]: https://github.com/actions/starter-workflows/blob/main/pages/hugo.yml diff --git a/content/en/functions/resources/_common/_index.md b/content/en/functions/resources/_common/_index.md new file mode 100644 index 00000000000..b57b688b310 --- /dev/null +++ b/content/en/functions/resources/_common/_index.md @@ -0,0 +1,12 @@ +--- +_build: + list: never + publishResources: false + render: never +--- + + diff --git a/content/en/functions/resources/_common/postcss-windows-warning.md b/content/en/functions/resources/_common/postcss-windows-warning.md new file mode 100644 index 00000000000..1b72e74db32 --- /dev/null +++ b/content/en/functions/resources/_common/postcss-windows-warning.md @@ -0,0 +1,8 @@ +--- +# Do not remove front matter. +--- + +If you are a Windows user, and the path to your project contains a space, you must place the PostCSS configuration within the package.json file. See [this example] and issue [#7333]. + +[this example]: https://github.com/postcss/postcss-load-config#packagejson +[#7333]: https://github.com/gohugoio/hugo/issues/7333 diff --git a/content/en/functions/resources/_index.md b/content/en/functions/resources/_index.md new file mode 100644 index 00000000000..364b9448d4f --- /dev/null +++ b/content/en/functions/resources/_index.md @@ -0,0 +1,12 @@ +--- +title: Resource functions +linkTitle: resources +description: Template functions to work with resources. +categories: [] +keywords: [] +menu: + docs: + parent: functions +--- + +Use these functions to work with resources. diff --git a/content/en/functions/safe/CSS.md b/content/en/functions/safe/CSS.md index d5dcdfb660e..dc0539bde3e 100644 --- a/content/en/functions/safe/CSS.md +++ b/content/en/functions/safe/CSS.md @@ -1,23 +1,18 @@ --- title: safe.CSS -linkTitle: safeCSS -description: Declares the provided string as a known "safe" CSS string. -categories: [functions] +description: Declares the provided string as safe CSS string. +categories: [] keywords: [] -menu: - docs: - parent: functions -function: +action: aliases: [safeCSS] + related: + - functions/safe/HTML + - functions/safe/HTMLAttr + - functions/safe/JS + - functions/safe/JSStr + - functions/safe/URL returnType: template.CSS signatures: [safe.CSS INPUT] -relatedFunctions: - - safe.CSS - - safe.HTML - - safe.HTMLAttr - - safe.JS - - safe.JSStr - - safe.URL aliases: [/functions/safecss] --- diff --git a/content/en/functions/safe/HTML.md b/content/en/functions/safe/HTML.md index ea3afe8f3ee..3cfb6ebdc47 100644 --- a/content/en/functions/safe/HTML.md +++ b/content/en/functions/safe/HTML.md @@ -1,23 +1,18 @@ --- title: safe.HTML -linkTitle: safeHTML -description: Declares a provided string as a "safe" HTML document to avoid escaping by Go templates. -categories: [functions] +description: Declares a provided string as a safeHTML string. +categories: [] keywords: [] -menu: - docs: - parent: functions -function: +action: aliases: [safeHTML] + related: + - functions/safe/CSS + - functions/safe/HTMLAttr + - functions/safe/JS + - functions/safe/JSStr + - functions/safe/URL returnType: template.HTML signatures: [safe.HTML INPUT] -relatedFunctions: - - safe.CSS - - safe.HTML - - safe.HTMLAttr - - safe.JS - - safe.JSStr - - safe.URL aliases: [/functions/safehtml] --- @@ -25,7 +20,7 @@ It should not be used for HTML from a third-party, or HTML with unclosed tags or Given a site-wide [`hugo.toml`][config] with the following `copyright` value: -{{< code-toggle file="hugo" >}} +{{< code-toggle file=hugo >}} copyright = "© 2015 Jane Doe. Some rights reserved." {{< /code-toggle >}} diff --git a/content/en/functions/safe/HTMLAttr.md b/content/en/functions/safe/HTMLAttr.md index 7d1b06c472f..1ac8b294ef1 100644 --- a/content/en/functions/safe/HTMLAttr.md +++ b/content/en/functions/safe/HTMLAttr.md @@ -1,29 +1,24 @@ --- title: safe.HTMLAttr -linkTitle: safeHTMLAttr description: Declares the provided string as a safe HTML attribute. -categories: [functions] +categories: [] keywords: [] -menu: - docs: - parent: functions -function: +action: aliases: [safeHTMLAttr] + related: + - functions/safe/CSS + - functions/safe/HTML + - functions/safe/JS + - functions/safe/JSStr + - functions/safe/URL returnType: template.HTMLAttr signatures: [safe.HTMLAttr INPUT] -relatedFunctions: - - safe.CSS - - safe.HTML - - safe.HTMLAttr - - safe.JS - - safe.JSStr - - safe.URL aliases: [/functions/safehtmlattr] --- Given a site configuration that contains this menu entry: -{{< code-toggle file="hugo" >}} +{{< code-toggle file=hugo >}} [[menu.main]] name = "IRC" url = "irc://irc.freenode.net/#golang" @@ -35,7 +30,7 @@ Attempting to use the `url` value directly in an attribute: {{ range site.Menus.main }} {{ .Name }} {{ end }} -``` +``` Will produce: diff --git a/content/en/functions/safe/JS.md b/content/en/functions/safe/JS.md index e679b5f852f..a33e19f0fb0 100644 --- a/content/en/functions/safe/JS.md +++ b/content/en/functions/safe/JS.md @@ -1,23 +1,18 @@ --- title: safe.JS -linkTitle: safeJS -description: Declares the provided string as a known safe JavaScript string. -categories: [functions] +description: Declares the provided string as a safe JavaScript string. +categories: [] keywords: [] -menu: - docs: - parent: functions -function: +action: aliases: [safeJS] + related: + - functions/safe/CSS + - functions/safe/HTML + - functions/safe/HTMLAttr + - functions/safe/JSStr + - functions/safe/URL returnType: template.JS signatures: [safe.JS INPUT] -relatedFunctions: - - safe.CSS - - safe.HTML - - safe.HTMLAttr - - safe.JS - - safe.JSStr - - safe.URL aliases: [/functions/safejs] --- diff --git a/content/en/functions/safe/JSStr.md b/content/en/functions/safe/JSStr.md index 790de3a7388..762d2a0df1c 100644 --- a/content/en/functions/safe/JSStr.md +++ b/content/en/functions/safe/JSStr.md @@ -1,23 +1,18 @@ --- title: safe.JSStr -linkTitle: safeJSStr description: Declares the provided string as a known safe JavaScript string. -categories: [functions] +categories: [] keywords: [] -menu: - docs: - parent: functions -function: +action: aliases: [safeJSStr] + related: + - functions/safe/CSS + - functions/safe/HTML + - functions/safe/HTMLAttr + - functions/safe/JS + - functions/safe/URL returnType: template.JSStr signatures: [safe.JSStr INPUT] -relatedFunctions: - - safe.CSS - - safe.HTML - - safe.HTMLAttr - - safe.JS - - safe.JSStr - - safe.URL aliases: [/functions/safejsstr] --- @@ -34,7 +29,6 @@ Without declaring a variable to be a safe JavaScript string: Rendered: - ```html + + {{ end }} + ... + +{{< /code >}} diff --git a/content/en/methods/page/HeadingsFiltered.md b/content/en/methods/page/HeadingsFiltered.md new file mode 100644 index 00000000000..a39c48da1ff --- /dev/null +++ b/content/en/methods/page/HeadingsFiltered.md @@ -0,0 +1,18 @@ +--- +title: HeadingsFiltered +description: Returns a slice of headings for each page related to the given page. +categories: [] +keywords: [] +action: + related: + - methods/pages/Related + - methods/page/Fragments + returnType: tableofcontents.Headings + signatures: [PAGE.HeadingsFiltered] +--- + +Use in conjunction with the [`Related`] method on a [`Pages`] object. See [details]. + +[`Pages`]: /methods/pages/ +[`Related`]: /methods/pages/related +[details]: /content-management/related/#index-content-headings-in-related-content diff --git a/content/en/methods/page/InSection.md b/content/en/methods/page/InSection.md new file mode 100644 index 00000000000..b98fbc808c5 --- /dev/null +++ b/content/en/methods/page/InSection.md @@ -0,0 +1,102 @@ +--- +title: InSection +description: Reports whether the given page is in the given section. +categories: [] +keywords: [] +action: + related: + - methods/page/Ancestors + - methods/page/CurrentSection + - methods/page/FirstSection + - methods/page/IsAncestor + - methods/page/IsDescendant + - methods/page/Parent + - methods/page/Sections + returnType: bool + signatures: [PAGE.InSection SECTION] +toc: true +--- + +The `InSection` method on a page object reports whether the given page is in the given section. Note that the method returns `true` when comparing a page to a sibling. + +{{% include "methods/page/_common/definition-of-section.md" %}} + +With this content structure: + +```text +content/ +├── auctions/ +│ ├── 2023-11/ +│ │ ├── _index.md +│ │ ├── auction-1.md +│ │ └── auction-2.md +│ ├── 2023-12/ +│ │ ├── _index.md +│ │ ├── auction-3.md +│ │ └── auction-4.md +│ ├── _index.md +│ ├── bidding.md +│ └── payment.md +└── _index.md +``` + +When rendering the "auction-1" page: + +```go-html-template +{{ with .Site.GetPage "/" }} + {{ $.InSection . }} → false +{{ end }} + +{{ with .Site.GetPage "/auctions" }} + {{ $.InSection . }} → false +{{ end }} + +{{ with .Site.GetPage "/auctions/2023-11" }} + {{ $.InSection . }} → true +{{ end }} + +{{ with .Site.GetPage "/auctions/2023-11/auction-2" }} + {{ $.InSection . }} → true +{{ end }} +``` + +In the examples above we are coding defensively using the [`with`] statement, returning nothing if the page does not exist. By adding an [`else`] clause we can do some error reporting: + +```go-html-template +{{ $path := "/auctions/2023-11" }} +{{ with .Site.GetPage $path }} + {{ $.InSection . }} → true +{{ else }} + {{ errorf "Unable to find the section with path %s" $path }} +{{ end }} + ``` + +## Understanding context + +Inside of the `with` block, the [context] (the dot) is the section `Page` object, not the `Page` object passed into the template. If we were to use this syntax: + +```go-html-template +{{ with .Site.GetPage "/auctions" }} + {{ .InSection . }} → true +{{ end }} +``` + +The result would be wrong when rendering the "auction-1" page because we are comparing the section page to itself. + +{{% note %}} +Use the `$` to get the context passed into the template. +{{% /note %}} + +```go-html-template +{{ with .Site.GetPage "/auctions" }} + {{ $.InSection . }} → true +{{ end }} +``` + +{{% note %}} +Gaining a thorough understanding of context is critical for anyone writing template code. +{{% /note %}} + +[context]: /getting-started/glossary/#context +[`with`]: /functions/go-template/with +[`else`]: /functions/go-template/else diff --git a/content/en/methods/page/IsAncestor.md b/content/en/methods/page/IsAncestor.md new file mode 100644 index 00000000000..ca23c0868cc --- /dev/null +++ b/content/en/methods/page/IsAncestor.md @@ -0,0 +1,100 @@ +--- +title: IsAncestor +description: Reports whether PAGE1 in an ancestor of PAGE2. +categories: [] +keywords: [] +action: + related: + - methods/page/Ancestors + - methods/page/CurrentSection + - methods/page/FirstSection + - methods/page/InSection + - methods/page/IsDescendant + - methods/page/Parent + - methods/page/Sections + returnType: bool + signatures: [PAGE1.IsAncestor PAGE2] +toc: true +--- + +{{% include "methods/page/_common/definition-of-section.md" %}} + +With this content structure: + +```text +content/ +├── auctions/ +│ ├── 2023-11/ +│ │ ├── _index.md +│ │ ├── auction-1.md +│ │ └── auction-2.md +│ ├── 2023-12/ +│ │ ├── _index.md +│ │ ├── auction-3.md +│ │ └── auction-4.md +│ ├── _index.md +│ ├── bidding.md +│ └── payment.md +└── _index.md +``` + +When rendering the "auctions" page: + +```go-html-template +{{ with .Site.GetPage "/" }} + {{ $.IsAncestor . }} → false +{{ end }} + +{{ with .Site.GetPage "/auctions" }} + {{ $.IsAncestor . }} → false +{{ end }} + +{{ with .Site.GetPage "/auctions/2023-11" }} + {{ $.IsAncestor . }} → true +{{ end }} + +{{ with .Site.GetPage "/auctions/2023-11/auction-2" }} + {{ $.IsAncestor . }} → true +{{ end }} +``` + +In the examples above we are coding defensively using the [`with`] statement, returning nothing if the page does not exist. By adding an [`else`] clause we can do some error reporting: + +```go-html-template +{{ $path := "/auctions/2023-11" }} +{{ with .Site.GetPage $path }} + {{ $.IsAncestor . }} → true +{{ else }} + {{ errorf "Unable to find the section with path %s" $path }} +{{ end }} + ``` + +## Understanding context + +Inside of the `with` block, the [context] (the dot) is the section `Page` object, not the `Page` object passed into the template. If we were to use this syntax: + +```go-html-template +{{ with .Site.GetPage "/auctions" }} + {{ .IsAncestor . }} → true +{{ end }} +``` + +The result would be wrong when rendering the "auction-1" page because we are comparing the section page to itself. + +{{% note %}} +Use the `$` to get the context passed into the template. +{{% /note %}} + +```go-html-template +{{ with .Site.GetPage "/auctions" }} + {{ $.IsAncestor . }} → true +{{ end }} +``` + +{{% note %}} +Gaining a thorough understanding of context is critical for anyone writing template code. +{{% /note %}} + +[context]: /getting-started/glossary/#context +[`with`]: /functions/go-template/with +[`else`]: /functions/go-template/else diff --git a/content/en/methods/page/IsDescendant.md b/content/en/methods/page/IsDescendant.md new file mode 100644 index 00000000000..f1042564e40 --- /dev/null +++ b/content/en/methods/page/IsDescendant.md @@ -0,0 +1,99 @@ +--- +title: IsDescendant +description: Reports whether PAGE1 in a descendant of PAGE2. +categories: [] +keywords: [] +action: + related: + - methods/page/Ancestors + - methods/page/CurrentSection + - methods/page/FirstSection + - methods/page/InSection + - methods/page/IsAncestor + - methods/page/Parent + - methods/page/Sections + returnType: bool + signatures: [PAGE1.IsDescendant PAGE2] +--- + +{{% include "methods/page/_common/definition-of-section.md" %}} + +With this content structure: + +```text +content/ +├── auctions/ +│ ├── 2023-11/ +│ │ ├── _index.md +│ │ ├── auction-1.md +│ │ └── auction-2.md +│ ├── 2023-12/ +│ │ ├── _index.md +│ │ ├── auction-3.md +│ │ └── auction-4.md +│ ├── _index.md +│ ├── bidding.md +│ └── payment.md +└── _index.md +``` + +When rendering the "auctions" page: + +```go-html-template +{{ with .Site.GetPage "/" }} + {{ $.IsDescendant . }} → true +{{ end }} + +{{ with .Site.GetPage "/auctions" }} + {{ $.IsDescendant . }} → false +{{ end }} + +{{ with .Site.GetPage "/auctions/2023-11" }} + {{ $.IsDescendant . }} → false +{{ end }} + +{{ with .Site.GetPage "/auctions/2023-11/auction-2" }} + {{ $.IsDescendant . }} → false +{{ end }} +``` + +In the examples above we are coding defensively using the [`with`] statement, returning nothing if the page does not exist. By adding an [`else`] clause we can do some error reporting: + +```go-html-template +{{ $path := "/auctions/2023-11" }} +{{ with .Site.GetPage $path }} + {{ $.IsDescendant . }} → true +{{ else }} + {{ errorf "Unable to find the section with path %s" $path }} +{{ end }} + ``` + +## Understanding context + +Inside of the `with` block, the [context] (the dot) is the section `Page` object, not the `Page` object passed into the template. If we were to use this syntax: + +```go-html-template +{{ with .Site.GetPage "/auctions" }} + {{ .IsDescendant . }} → true +{{ end }} +``` + +The result would be wrong when rendering the "auction-1" page because we are comparing the section page to itself. + +{{% note %}} +Use the `$` to get the context passed into the template. +{{% /note %}} + +```go-html-template +{{ with .Site.GetPage "/auctions" }} + {{ $.IsDescendant . }} → true +{{ end }} +``` + +{{% note %}} +Gaining a thorough understanding of context is critical for anyone writing template code. +{{% /note %}} + +[context]: /getting-started/glossary/#context +[`with`]: /functions/go-template/with +[`else`]: /functions/go-template/else diff --git a/content/en/methods/page/IsHome.md b/content/en/methods/page/IsHome.md new file mode 100644 index 00000000000..b688f88c0e6 --- /dev/null +++ b/content/en/methods/page/IsHome.md @@ -0,0 +1,31 @@ +--- +title: IsHome +description: Reports whether the given page is the home page. +categories: [] +keywords: [] +action: + related: + - methods/page/IsNode + - methods/page/IsPage + - methods/page/IsSection + returnType: bool + signatures: [PAGE.IsHome] +--- + +The `IsHome` method on a `Page` object returns `true` if the [page kind] is `home`. + +```text +content/ +├── books/ +│ ├── book-1/ +│ │ └── index.md <-- kind = page +│ ├── book-2.md <-- kind = page +│ └── _index.md <-- kind = section +└── _index.md <-- kind = home +``` + +```go-html-template +{{ .IsHome }} +``` + +[page kind]: /getting-started/glossary/#page-kind diff --git a/content/en/functions/IsMenuCurrent.md b/content/en/methods/page/IsMenuCurrent.md similarity index 81% rename from content/en/functions/IsMenuCurrent.md rename to content/en/methods/page/IsMenuCurrent.md index c9980b3e8a7..61283fd8ba1 100644 --- a/content/en/functions/IsMenuCurrent.md +++ b/content/en/methods/page/IsMenuCurrent.md @@ -1,18 +1,14 @@ --- -title: .IsMenuCurrent +title: IsMenuCurrent description: Reports whether the given page object matches the page object associated with the given menu entry in the given menu. -categories: [functions] +categories: [] keywords: [] -menu: - docs: - parent: functions -function: - aliases: [] +action: + related: + - methods/page/HasMenuCurrent returnType: bool signatures: [PAGE.IsMenuCurrent MENU MENUENTRY] -relatedFunctions: - - .HasMenuCurrent - - .IsMenuCurrent +aliases: [/functions/ismenucurrent] --- ```go-html-template diff --git a/content/en/methods/page/IsNode.md b/content/en/methods/page/IsNode.md new file mode 100644 index 00000000000..dfdf435c50d --- /dev/null +++ b/content/en/methods/page/IsNode.md @@ -0,0 +1,36 @@ +--- +title: IsNode +description: Reports whether the given page is a node. +categories: [] +keywords: [] +action: + related: + - methods/page/IsHome + - methods/page/IsPage + - methods/page/IsSection + returnType: bool + signatures: [PAGE.IsNode] +--- + +The `IsNode` method on a `Page` object returns `true` if the [page kind] is `home`, `section`, `taxonomy`, or `term`. + +It returns `false` is the page kind is `page`. + +```text +content/ +├── books/ +│ ├── book-1/ +│ │ └── index.md <-- kind = page, node = false +│ ├── book-2.md <-- kind = page, node = false +│ └── _index.md <-- kind = section, node = true +├── tags/ +│ ├── fiction/ +│ │ └── _index.md <-- kind = term, node = true +│ └── _index.md <-- kind = taxonomy, node = true +└── _index.md <-- kind = home, node = true +``` + +```go-html-template +{{ .IsNode }} +``` +[page kind]: /getting-started/glossary/#page-kind diff --git a/content/en/methods/page/IsPage.md b/content/en/methods/page/IsPage.md new file mode 100644 index 00000000000..672ee61f47e --- /dev/null +++ b/content/en/methods/page/IsPage.md @@ -0,0 +1,31 @@ +--- +title: IsPage +description: Reports whether the given page is a regular page. +categories: [] +keywords: [] +action: + related: + - methods/page/IsHome + - methods/page/IsNode + - methods/page/IsSection + returnType: bool + signatures: [PAGE.IsPage] +--- + +The `IsPage` method on a `Page` object returns `true` if the [page kind] is `page`. + +```text +content/ +├── books/ +│ ├── book-1/ +│ │ └── index.md <-- kind = page +│ ├── book-2.md <-- kind = page +│ └── _index.md <-- kind = section +└── _index.md <-- kind = home +``` + +```go-html-template +{{ .IsPage }} +``` + +[page kind]: /getting-started/glossary/#page-kind diff --git a/content/en/methods/page/IsSection.md b/content/en/methods/page/IsSection.md new file mode 100644 index 00000000000..b02e58a4521 --- /dev/null +++ b/content/en/methods/page/IsSection.md @@ -0,0 +1,31 @@ +--- +title: IsSection +description: Reports whether the given page is a section page. +categories: [] +keywords: [] +action: + related: + - methods/page/IsHome + - methods/page/IsNode + - methods/page/IsPage + returnType: bool + signatures: [PAGE.IsSection] +--- + +The `IsSection` method on a `Page` object returns `true` if the [page kind] is `section`. + +```text +content/ +├── books/ +│ ├── book-1/ +│ │ └── index.md <-- kind = page +│ ├── book-2.md <-- kind = page +│ └── _index.md <-- kind = section +└── _index.md <-- kind = home +``` + +```go-html-template +{{ .IsSection }} +``` + +[page kind]: /getting-started/glossary/#page-kind diff --git a/content/en/methods/page/IsTranslated.md b/content/en/methods/page/IsTranslated.md new file mode 100644 index 00000000000..6a8f3f69e03 --- /dev/null +++ b/content/en/methods/page/IsTranslated.md @@ -0,0 +1,59 @@ +--- +title: IsTranslated +description: Reports whether the given page has one or more translations. +categories: [] +keywords: [] +action: + related: + - methods/page/Translations + - methods/page/AllTranslations + - methods/page/TranslationKey + returnType: bool + signatures: [PAGE.IsTranslated] +--- + +With this site configuration: + +{{< code-toggle file=hugo >}} +defaultContentLanguage = 'en' + +[languages.en] +contentDir = 'content/en' +languageCode = 'en-US' +languageName = 'English' +weight = 1 + +[languages.de] +contentDir = 'content/de' +languageCode = 'de-DE' +languageName = 'Deutsch' +weight = 2 +{{< /code-toggle >}} + +And this content: + +```text +content/ +├── de/ +│ ├── books/ +│ │ └── book-1.md +│ └── _index.md +├── en/ +│ ├── books/ +│ │ ├── book-1.md +│ │ └── book-2.md +│ └── _index.md +└── _index.md +``` + +When rendering content/en/books/book-1.md: + +```go-html-template +{{ .IsTranslated }} → true +``` + +When rendering content/en/books/book-2.md: + +```go-html-template +{{ .IsTranslated }} → false +``` diff --git a/content/en/methods/page/Keywords.md b/content/en/methods/page/Keywords.md new file mode 100644 index 00000000000..5ad37ce51e6 --- /dev/null +++ b/content/en/methods/page/Keywords.md @@ -0,0 +1,46 @@ +--- +title: Keywords +description: Returns a slice of keywords as defined in front matter. +categories: [] +keywords: [] +action: + related: [] + returnType: '[]string' + signatures: [PAGE.Keywords] +--- + +By default, Hugo evaluates the keywords when creating collections of [related content]. + +[related content]: /content-management/related + +{{< code-toggle file=content/recipes/sushi.md fm=true >}} +title = 'How to make spicy tuna hand rolls' +keywords = ['tuna','sriracha','nori','rice'] +{{< /code-toggle >}} + +To list the keywords within a template: + +```go-html-template +{{ range .Keywords }} + {{ . }} +{{ end }} +``` + +Or use the [delimit] function: + +```go-html-template +{{ delimit .Keywords ", " ", and " }} → tuna, sriracha, nori, and rice +``` + +[delimit]: /functions/collections/delimit + +Keywords are also a useful [taxonomy]: + +{{< code-toggle file=hugo >}} +[taxonomies] +tag = 'tags' +keyword = 'keywords' +category = 'categories' +{{< /code-toggle >}} + +[taxonomy]: /content-management/taxonomies diff --git a/content/en/methods/page/Kind.md b/content/en/methods/page/Kind.md new file mode 100644 index 00000000000..d901e9a7d39 --- /dev/null +++ b/content/en/methods/page/Kind.md @@ -0,0 +1,35 @@ +--- +title: Kind +description: Returns the kind of the given page. +categories: [] +keywords: [] +action: + related: + - methods/page/Type + returnType: string + signatures: [PAGE.Kind] +--- + +The [page kind] is one of `home`, `page`, `section`, `taxonomy`, or `term`. + +```text +content/ +├── books/ +│ ├── book-1/ +│ │ └── index.md <-- kind = page +│ ├── book-2.md <-- kind = page +│ └── _index.md <-- kind = section +├── tags/ +│ ├── fiction/ +│ │ └── _index.md <-- kind = term +│ └── _index.md <-- kind = taxonomy +└── _index.md <-- kind = home +``` + +To get the value within a template: + +```go-html-template +{{ .Kind }} +``` + +[page kind]: /getting-started/glossary/#page-kind diff --git a/content/en/methods/page/Language.md b/content/en/methods/page/Language.md new file mode 100644 index 00000000000..4e65107da0c --- /dev/null +++ b/content/en/methods/page/Language.md @@ -0,0 +1,65 @@ +--- +title: Language +description: Returns the language object for the given page. +categories: [] +keywords: [] +action: + related: + - methods/site/Language + returnType: langs.Language + signatures: [PAGE.Language] +--- + +The `Language` method on a `Page` object returns the language object for the given page. The language object points to the language definition in the site configuration. + +You can also use the `Language` method on a `Site` object. See [details]. + +## Methods + +The examples below assume the following in your site configuration: + +{{< code-toggle file=hugo >}} +[languages.de] +languageCode = 'de-DE' +languageDirection = 'ltr' +languageName = 'Deutsch' +weight = 2 +{{< /code-toggle >}} + +Lang +: (`string`) The language tag as defined by [RFC 5646]. + +```go-html-template +{{ .Language.Lang }} → de +``` + +LanguageCode +: (`string`) The language code from the site configuration. + +```go-html-template +{{ .Language.LanguageCode }} → de-DE +``` + +LanguageDirection +: (`string`) The language direction from the site configuration, either `ltr` or `rtl`. + +```go-html-template +{{ .Language.LanguageDirection }} → ltr +``` + +LanguageName +: (`string`) The language name from the site configuration. + +```go-html-template +{{ .Language.LanguageName }} → Deutsch +``` + +Weight +: (`int`) The language weight from the site configuration which determines its order in the slice of languages returned by the `Languages` method on a `Site` object. + +```go-html-template +{{ .Language.Weight }} → 2 +``` + +[details]: /methods/site/language +[RFC 5646]: https://datatracker.ietf.org/doc/html/rfc5646 diff --git a/content/en/methods/page/Lastmod.md b/content/en/methods/page/Lastmod.md new file mode 100644 index 00000000000..c1692233d4b --- /dev/null +++ b/content/en/methods/page/Lastmod.md @@ -0,0 +1,40 @@ +--- +title: Lastmod +description: Returns the last modification date of the given page. +categories: [] +keywords: [] +action: + related: + - methods/page/Date + - methods/page/ExpiryDate + - methods/page/PublishDate + - methods/page/GitInfo + returnType: time.Time + signatures: [PAGE.Lastmod] +--- + +Set the last modification date in front matter: + +{{< code-toggle file=content/news/article-1.md fm=true >}} +title = 'Article 1' +lastmod = 2023-10-19T00:40:04-07:00 +{{< /code-toggle >}} + +The last modification date is a [time.Time] value. Format and localize the value with the [`time.Format`] function, or use it with any of the [time methods]. + +```go-html-template +{{ .Lastmod | time.Format ":date_medium" }} → Oct 19, 2023 +``` + +In the example above we explicitly set the last modification date in front matter. With Hugo's default configuration, the `Lastmod` method returns the front matter value. This behavior is configurable, allowing you to: + +- Set the last modification date to the Author Date of the last Git commit for that file. See [`GitInfo`] for details. +- Set fallback values if the last modification date is not defined in front matter. + +Learn more about [date configuration]. + +[`gitinfo`]: /methods/page/gitinfo +[`time.format`]: /functions/time/format +[date configuration]: /getting-started/configuration/#configure-dates +[time methods]: /methods/time +[time.time]: https://pkg.go.dev/time#time diff --git a/content/en/methods/page/Layout.md b/content/en/methods/page/Layout.md new file mode 100644 index 00000000000..3d0cdc43769 --- /dev/null +++ b/content/en/methods/page/Layout.md @@ -0,0 +1,40 @@ +--- +title: Layout +description: Returns the layout for the given page as defined in front matter. +categories: [] +keywords: [] +action: + related: + - methods/page/Type + returnType: string + signatures: [PAGE.Layout] +--- + +Specify the `layout` field in front matter to target a particular template. See [details]. + +[details]: /templates/lookup-order/#target-a-template + +{{< code-toggle file=content/contact.md >}} +title = 'Contact' +layout = 'contact' +{{< /code-toggle >}} + +Hugo will render the page using contact.html. + +```text +layouts/ +└── _default/ + ├── baseof.html + ├── contact.html + ├── home.html + ├── list.html + └── single.html +``` + +Although rarely used within a template, you can access the value with: + +```go-html-template +{{ .Layout }} +``` + +The `Layout` method returns an empty string if the `layout` field in front matter is not defined. diff --git a/content/en/methods/page/LinkTitle.md b/content/en/methods/page/LinkTitle.md new file mode 100644 index 00000000000..141dec8b768 --- /dev/null +++ b/content/en/methods/page/LinkTitle.md @@ -0,0 +1,30 @@ +--- +title: LinkTitle +description: Returns the link title of the given page. +categories: [] +keywords: [] +action: + related: + - methods/page/Title + returnType: string + signatures: [PAGE.LinkTitle] +--- + +The `.LinkTitle` method returns the `linkTitle` field as defined in front matter, falling back to the value returned by the [`.Title`] method. + +[`.Title`]: /methods/page/title + +{{< code-toggle file=content/articles/healthy-desserts.md fm=true >}} +title = 'Seventeen delightful recipes for healthy desserts' +linkTitle = 'Dessert recipes' +{{< /code-toggle >}} + +```go-html-template +{{ .LinkTitle }} → Dessert recipes +``` + +As demonstrated above, defining a link title in front matter is advantageous when the page title is long. Use it when generating anchor elements in your templates: + +```go-html-template +{{ .LinkTitle }} +``` diff --git a/content/en/methods/page/Next.md b/content/en/methods/page/Next.md new file mode 100644 index 00000000000..3c9e3495acc --- /dev/null +++ b/content/en/methods/page/Next.md @@ -0,0 +1,53 @@ +--- +title: Next +description: Returns the next page in a global page collection, relative to the given page. +categories: [] +keywords: [] +action: + related: + - methods/page/Prev + - methods/page/NextInSection + - methods/page/PrevInSection + - methods/pages/Next + - methods/pages/Prev + returnType: hugolib.pageState + signatures: [PAGE.Next] +toc: true +--- + +The behavior of the `Prev` and `Next` methods on a `Page` object is probably the reverse of what you expect. + +With this content structure: + +```text +content/ +├── pages/ +│ ├── _index.md +│ ├── page-1.md <-- front matter: weight = 10 +│ ├── page-2.md <-- front matter: weight = 20 +│ └── page-3.md <-- front matter: weight = 30 +└── _index.md +``` + +When you visit page-2: + +- The `Prev` method points to page-3 +- The `Next` method points to page-1 + +{{% note %}} +Use the opposite label in your navigation links as shown in the example below. +{{% /note %}} + +```go-html-template +{{ with .Next }} + Prev +{{ end }} + +{{ with .Prev }} + Next +{{ end }} +``` + +## Compare to Pages methods + +{{% include "methods/_common/next-prev-on-page-vs-next-prev-on-pages.md" %}} diff --git a/content/en/methods/page/NextInSection.md b/content/en/methods/page/NextInSection.md new file mode 100644 index 00000000000..10185cf146f --- /dev/null +++ b/content/en/methods/page/NextInSection.md @@ -0,0 +1,71 @@ +--- +title: NextInSection +description: Returns the next page within a section, relative to the given page. +categories: [] +keywords: [] +action: + related: + - methods/page/PrevInSection + - methods/page/Next + - methods/page/Prev + - methods/pages/Next + - methods/pages/Prev + returnType: hugolib.pageState + signatures: [PAGE.NextInSection] +--- + +The behavior of the `PrevInSection` and `NextInSection` methods on a `Page` object is probably the reverse of what you expect. + +With this content structure: + +```text +content/ +├── books/ +│ ├── _index.md +│ ├── book-1.md +│ ├── book-2.md +│ └── book-3.md +├── films/ +│ ├── _index.md +│ ├── film-1.md +│ ├── film-2.md +│ └── film-3.md +└── _index.md +``` + +When you visit book-2: + +- The `PrevInSection` method points to book-3 +- The `NextInSection` method points to book-1 + +{{% note %}} +Use the opposite label in your navigation links as shown in the example below. +{{% /note %}} + +```go-html-template +{{ with .NextInSection }} + Previous in section +{{ end }} + +{{ with .PrevInSection }} + Next in section +{{ end }} +``` + +{{% note %}} +The navigation sort order may be different than the page collection sort order. +{{% /note %}} + +With the `PrevInSection` and `NextInSection` methods, the navigation sort order is fixed, using Hugo’s default sort order. In order of precedence: + +1. Page [weight] +2. Page [date] +3. Page [linkTitle], falling back to page [title] +4. Page file path if the page is backed by a file + +For example, with a page collection sorted by title, the navigation sort order will use Hugo’s default sort order. This is probably not what you want or expect. For this reason, the Next and Prev methods on a Pages object are a generally a better choice. + +[date]: /methods/page/date +[weight]: /methods/page/weight +[linkTitle]: /methods/page/linktitle +[title]: /methods/page/title diff --git a/content/en/methods/page/OutputFormats.md b/content/en/methods/page/OutputFormats.md new file mode 100644 index 00000000000..03343cf8c07 --- /dev/null +++ b/content/en/methods/page/OutputFormats.md @@ -0,0 +1,40 @@ +--- +title: OutputFormats +description: Returns a slice of OutputFormat objects, each representing one of the output formats enabled for the given page. +categories: [] +keywords: [] +action: + related: + - methods/page/AlternativeOutputFormats + returnType: '[]OutputFormat' + signatures: [PAGE.OutputFormats] +toc: true +--- + +{{% include "methods/page/_common/output-format-definition.md" %}} + +The `OutputFormats` method on a `Page` object returns a slice of `OutputFormat` objects, each representing one of the output formats enabled for the given page. See [details](/templates/output-formats/). + +## Methods + +{{% include "methods/page/_common/output-format-methods.md" %}} + +## Example + +To link to the RSS feed for the current page: + +```go-html-template +{{ with .OutputFormats.Get "rss" -}} + RSS Feed +{{ end }} +``` + +On the site's home page, Hugo renders this to: + +```html +RSS Feed +``` + +Please see the [link to output formats] section to understand the importance of the construct above. + +[link to output formats]: /templates/output-formats/#link-to-output-formats diff --git a/content/en/methods/page/Page.md b/content/en/methods/page/Page.md new file mode 100644 index 00000000000..1c73c60fc39 --- /dev/null +++ b/content/en/methods/page/Page.md @@ -0,0 +1,40 @@ +--- +title: Page +description: Returns the Page object of the given page. +categories: [] +keywords: [] +action: + related: [] + returnType: hugolib.pageState + signatures: [PAGE.Page] +--- + +This is a convenience method, useful within partial templates that are called from both [shortcodes] and page templates. + +{{< code file="layouts/shortcodes/foo.html" lang=go-html-template >}} +{{ partial "my-partial.html" . }} +{{< /code >}} + +When the shortcode calls the partial, it passes the current [context] (the dot). The context includes identifiers such as `Page`, `Params`, `Inner`, and `Name`. + +{{< code file="layouts/_default/single.html" lang=go-html-template >}} +{{ partial "my-partial.html" . }} +{{< /code >}} + +When the page template calls the partial, it also passes the current context (the dot). But in this case, the dot _is_ the `Page` object. + +{{< code file="layouts/partials/my-partial.html" lang=go-html-template >}} +The page title is: {{ .Page.Title }} +{{< /code >}} + +To handle both scenarios, the partial template must be able to access the `Page` object with `Page.Page`. + +{{% note %}} +And yes, that means you can do `.Page.Page.Page.Page.Title` too. + +But don't. +{{% /note %}} + + +[context]: getting-started/glossary/#context +[shortcodes]: /getting-started/glossary/#shortcode diff --git a/content/en/methods/page/Pages.md b/content/en/methods/page/Pages.md new file mode 100644 index 00000000000..9bd8fc0a7ab --- /dev/null +++ b/content/en/methods/page/Pages.md @@ -0,0 +1,90 @@ +--- +title: Pages +description: Returns a collection of regular pages within the current section, and section pages of immediate descendant sections. +categories: [] +keywords: [] +action: + related: + - methods/page/RegularPages + - methods/page/RegularPagesRecursive + returnType: page.Pages + signatures: [PAGE.Pages] +--- + +The `Pages` method on a `Page` object is available to these [page kinds]: `home`, `section`, `taxonomy`, and `term`. The templates for these page kinds receive a page [collection] in [context]. + +Range through the page collection in your template: + +```go-html-template +{{ range .Pages.ByTitle }} +

{{ .LinkTitle }}

+{{ end }} +``` + +Consider this content structure: + +```text +content/ +├── lessons/ +│ ├── lesson-1/ +│ │ ├── _index.md +│ │ ├── part-1.md +│ │ └── part-2.md +│ ├── lesson-2/ +│ │ ├── resources/ +│ │ │ ├── task-list.md +│ │ │ └── worksheet.md +│ │ ├── _index.md +│ │ ├── part-1.md +│ │ └── part-2.md +│ ├── _index.md +│ ├── grading-policy.md +│ └── lesson-plan.md +├── _index.md +├── contact.md +└── legal.md +``` + +When rendering the home page, the `Pages` method returns: + + contact.md + legal.md + lessons/_index.md + +When rendering the lessons page, the `Pages` method returns: + + lessons/grading-policy.md + lessons/lesson-plan.md + lessons/lesson-1/_index.md + lessons/lesson-2/_index.md + +When rendering lesson-1, the `Pages` method returns: + + lessons/lesson-1/part-1.md + lessons/lesson-1/part-2.md + +When rendering lesson-2, the `Pages` method returns: + + lessons/lesson-2/part-1.md + lessons/lesson-2/part-2.md + lessons/lesson-2/resources/task-list.md + lessons/lesson-2/resources/worksheet.md + +In the last example, the collection includes pages in the resources subdirectory. That directory is not a [section]---it does not contain an _index.md file. Its contents are part of the lesson-2 section. + +{{% note %}} +When used with a `Site` object, the `Pages` method recursively returns all pages within the site. See [details]. + +[details]: /methods/site/pages +{{% /note %}} + +```go-html-template +{{ range .Site.Pages.ByTitle }} +

{{ .LinkTitle }}

+{{ end }} +``` + +[collection]: /getting-started/glossary/#collection +[context]: /getting-started/glossary/#context +[page kinds]: /getting-started/glossary/#page-kind +[section]: /getting-started/glossary/#section diff --git a/content/en/methods/page/Paginate.md b/content/en/methods/page/Paginate.md new file mode 100644 index 00000000000..878cae0a53c --- /dev/null +++ b/content/en/methods/page/Paginate.md @@ -0,0 +1,50 @@ +--- +title: Paginate +description: Paginates a collection of pages. +categories: [] +keywords: [] +action: + related: + - methods/page/Paginator + returnType: page.Pager + signatures: ['PAGE.Paginate COLLECTION [N]'] +--- + +[Pagination] is the process of splitting a list page into two or more pagers, where each pager contains a subset of the page collection and navigation links to other pagers. + +By default, the number of elements on each pager is determined by the value of the `paginate` setting in your site configuration. The default value is `10`. Override the value in your site configuration by providing a second argument, an integer, when calling the `Paginate` method. + +{{% note %}} +There is also a `Paginator` method on `Page` objects, but it can neither filter nor sort the page collection. + +The `Paginate` method is more flexible. +{{% /note %}} + +You can invoke pagination on the home page template, [`section`] templates, [`taxonomy`] templates, and [`term`] templates. + +{{< code file=layouts/_default/list.html >}} +{{ $pages := where .Site.RegularPages "Section" "articles" }} +{{ $pages = $pages.ByTitle }} +{{ range (.Paginate $pages 7).Pages }} +

{{ .LinkTitle }}

+{{ end }} +{{ template "_internal/pagination.html" . }} +{{< /code >}} + +In the example above, we: + +1. Build a page collection +2. Sort the collection by title +3. Paginate the collection, with 7 elements per pager +4. Range over the paginated page collection, rendering a link to each page +5. Call the internal "pagination" template to create the navigation links between pagers. + +{{% note %}} +Please note that the results of pagination are cached. Once you have invoked either the `Paginator` or `Paginate` method, the paginated collection is immutable. Additional invocations of these methods will have no effect. +{{% /note %}} + +[context]: /getting-started/glossary/#context +[pagination]: /templates/pagination/ +[`section`]: /getting-started/glossary/#section +[`taxonomy`]: /getting-started/glossary/#taxonomy +[`term`]: /getting-started/glossary/#term diff --git a/content/en/methods/page/Paginator.md b/content/en/methods/page/Paginator.md new file mode 100644 index 00000000000..b1540286ad5 --- /dev/null +++ b/content/en/methods/page/Paginator.md @@ -0,0 +1,42 @@ +--- +title: Paginator +description: Paginates the collection of regular pages received in context. +categories: [] +keywords: [] +action: + related: + - methods/page/Paginate + returnType: page.Pager + signatures: [PAGE.Paginator] +--- + +[Pagination] is the process of splitting a list page into two or more pagers, where each pager contains a subset of the page collection and navigation links to other pagers. The number of elements on each pager is determined by the value of the `paginate` setting in your site configuration. The default value is `10`. + +You can invoke pagination on the home page template, [`section`] templates, [`taxonomy`] templates, and [`term`] templates. Each of these receive a collection of regular pages in [context]. When you invoke the `Paginator` method, it paginates the page collection received in context. + +{{< code file=layouts/_default/list.html >}} +{{ range .Paginator.Pages }} +

{{ .LinkTitle }}

+{{ end }} +{{ template "_internal/pagination.html" . }} +{{< /code >}} + +In the example above, the internal "pagination" template creates the navigation links between pagers. + +{{% note %}} +Although simple to invoke, with the `Paginator` method you can neither filter nor sort the page collection. It acts upon the page collection received in context. + +The [`Paginate`] method is more flexible, and strongly recommended. + +[`paginate`]: /methods/page/paginate +{{% /note %}} + +{{% note %}} +Please note that the results of pagination are cached. Once you have invoked either the `Paginator` or `Paginate` method, the paginated collection is immutable. Additional invocations of these methods will have no effect. +{{% /note %}} + +[context]: /getting-started/glossary/#context +[pagination]: /templates/pagination/ +[`section`]: /getting-started/glossary/#section +[`taxonomy`]: /getting-started/glossary/#taxonomy +[`term`]: /getting-started/glossary/#term diff --git a/content/en/methods/page/Param.md b/content/en/methods/page/Param.md new file mode 100644 index 00000000000..6264e3064dc --- /dev/null +++ b/content/en/methods/page/Param.md @@ -0,0 +1,47 @@ +--- +title: Param +description: Returns a page parameter with the given key, falling back to a site parameter if present. +categories: [] +keywords: [] +action: + related: [] + returnType: any + signatures: [PAGE.Param KEY] +aliases: [/functions/param] +--- + +The `Param` method on a `Page` object looks for the given `KEY` in page parameters, and returns the corresponding value. If it cannot find the `KEY` in page parameters, it looks for the `KEY` in site parameters. If it cannot find the `KEY` in either location, the `.Param` method returns `nil`. + +Site and theme developers commonly set parameters at the site level, allowing content authors to override those parameters at the page level. + +For example, to show a table of contents on every page, but allow authors to hide the table of contents as needed: + +Configuration: + +{{< code-toggle file=hugo >}} +[params] +display_toc = true +{{< /code-toggle >}} + +Content: + +{{< code-toggle file="content/example.md" fm=true >}} +title = 'Example' +date = 2023-01-01 +draft = false +display_toc = false +{{< /code-toggle >}} + +Template: + +```go-html-template +{{ if .Param "display_toc" }} + {{ .TableOfContents }} +{{ end }} +``` + +The `Param` method returns the value associated with the given `KEY`, regardless of whether the value is truthy or falsy. If you need to ignore falsy values, use this construct instead: + +```go-html-template +{{ or .Params.foo site.Params.foo }} +``` diff --git a/content/en/methods/page/Params.md b/content/en/methods/page/Params.md new file mode 100644 index 00000000000..efe8287e03d --- /dev/null +++ b/content/en/methods/page/Params.md @@ -0,0 +1,43 @@ +--- +title: Params +description: Returns a map of custom parameters as defined in the front matter of the given page. +categories: [] +keywords: [] +action: + related: + - functions/collections/IndexFunction + - methods/site/Params + - methods/page/Param + returnType: maps.Params + signatures: [PAGE.Params] +--- + +With this front matter: + +{{< code-toggle file="content/news/annual-conference.md" >}} +title = 'Annual conference' +date = 2023-10-17T15:11:37-07:00 +display_related = true +event-date = '2023' +[params.author] + email = 'jsmith@example.org' + name = 'John Smith' +{{< /code-toggle >}} + +The `title` and `date` fields are standard parameters---the other fields are user-defined. + +Access the custom parameters by [chaining] the [identifiers]: + +```go-html-template +{{ .Params.display_related }} → true +{{ .Params.author.name }} → John Smith +``` + +In the template example above, each of the keys is a valid identifier. For example, none of the keys contains a hyphen. To access a key that is not a valid identifier, use the [`index`] function: + +```go-html-template +{{ index .Params "event-date" }} → 2023 +``` +[`index`]: /functions/collections/indexfunction +[chaining]: /getting-started/glossary/#chain +[identifiers]: /getting-started/glossary/#identifier diff --git a/content/en/methods/page/Parent.md b/content/en/methods/page/Parent.md new file mode 100644 index 00000000000..dbd2cb9b3f5 --- /dev/null +++ b/content/en/methods/page/Parent.md @@ -0,0 +1,60 @@ +--- +title: Parent +description: Returns the Page object of the parent section of the given page. +categories: [] +keywords: [] +action: + related: + - methods/page/Ancestors + - methods/page/CurrentSection + - methods/page/FirstSection + - methods/page/InSection + - methods/page/IsAncestor + - methods/page/IsDescendant + - methods/page/Sections + returnType: hugolib.pageState + signatures: [PAGE.Parent] +--- + +{{% include "methods/page/_common/definition-of-section.md" %}} + +{{% note %}} +The parent section of a regular page is the [current section]. + +[current section]: /methods/page/currentsection +{{% /note %}} + +Consider this content structure: + +```text +content/ +├── auctions/ +│ ├── 2023-11/ +│ │ ├── _index.md <-- parent: auctions +│ │ ├── auction-1.md +│ │ └── auction-2.md <-- parent: 2023-11 +│ ├── 2023-12/ +│ │ ├── _index.md +│ │ ├── auction-3.md +│ │ └── auction-4.md +│ ├── _index.md <-- parent: home +│ ├── bidding.md +│ └── payment.md <-- parent: auctions +├── books/ +│ ├── _index.md <-- parent: home +│ ├── book-1.md +│ └── book-2.md <-- parent: books +├── films/ +│ ├── _index.md <-- parent: home +│ ├── film-1.md +│ └── film-2.md <-- parent: films +└── _index.md <-- parent: nil +``` + +In the example above, note the parent section of the home page is nil. Code defensively by verifying existence of the parent section before calling methods on its `Page` object. To create a link to the parent section page of the current page: + +```go-html-template +{{ with .Parent }} + {{ .LinkTitle }} +{{ end }} +``` diff --git a/content/en/methods/page/Permalink.md b/content/en/methods/page/Permalink.md new file mode 100644 index 00000000000..be6df5aadfd --- /dev/null +++ b/content/en/methods/page/Permalink.md @@ -0,0 +1,25 @@ +--- +title: Permalink +description: Returns the permalink of the given page. +categories: [] +keywords: [] +action: + related: + - methods/page/RelPermalink + returnType: string + signatures: [PAGE.Permalink] +--- + +Site configuration: + +{{< code-toggle file=hugo >}} +title = 'Documentation' +baseURL = 'https://example.org/docs/' +{{< /code-toggle >}} + +Template: + +```go-html-template +{{ $page := .Site.GetPage "/about" }} +{{ $page.RelPermalink }} → https://example.org/docs/about/ +``` diff --git a/content/en/methods/page/Plain.md b/content/en/methods/page/Plain.md new file mode 100644 index 00000000000..6fdf60b62cf --- /dev/null +++ b/content/en/methods/page/Plain.md @@ -0,0 +1,28 @@ +--- +title: Plain +description: Returns the rendered content of the given page, removing all HTML tags. +categories: [] +keywords: [] +action: + related: + - methods/page/Content + - methods/page/RawContent + - methods/page/PlainWords + - methods/page/RenderShortcodes + returnType: string + signatures: [PAGE.Plain] +--- + +The `Plain` method on a `Page` object renders markdown and [shortcodes] to HTML, then strips the HTML [tags]. It does not strip HTML [entities]. The plain content does not include front matter. + +To prevent Go's [html/template] package from escaping HTML entities, pass the result through the [`htmlUnescape`] function. + +```go-html-template +{{ .Plain | htmlUnescape }} +``` + +[shortcodes]: /getting-started/glossary/#shortcode +[html/template]: https://pkg.go.dev/html/template +[entities]: https://developer.mozilla.org/en-US/docs/Glossary/Entity +[tags]: https://developer.mozilla.org/en-US/docs/Glossary/Tag +[`htmlUnescape`]: /functions/ diff --git a/content/en/methods/page/PlainWords.md b/content/en/methods/page/PlainWords.md new file mode 100644 index 00000000000..f0a6a9c4226 --- /dev/null +++ b/content/en/methods/page/PlainWords.md @@ -0,0 +1,34 @@ +--- +title: PlainWords +description: Calls the Plain method, splits the result into a slice of words, and returns the slice. +categories: [] +keywords: [] +action: + related: + - methods/page/Content + - methods/page/RawContent + - methods/page/Plain + returnType: '[]string' + signatures: [PAGE.PlainWords] +--- + +The `PlainWords` method on a `Page` object calls the [`Plain`] method, then uses Go's [`strings.Fields`] function to split the result into words. + +{{% note %}} +_Fields splits the string s around each instance of one or more consecutive white space characters, as defined by unicode.IsSpace, returning a slice of substrings of s or an empty slice if s contains only white space._ +{{% /note %}} + +As a result, elements within the slice may contain leading or trailing punctuation. + +```go-html-template +{{ .PlainWords }} +``` + +To determine the approximate number of unique words on a page: + +```go-html-template +{{ .PlainWords | uniq }} → 42 +``` + +[`Plain`]: /methods/page/plain +[`strings.Fields`]: https://pkg.go.dev/strings#Fields diff --git a/content/en/methods/page/Prev.md b/content/en/methods/page/Prev.md new file mode 100644 index 00000000000..cde83b0f275 --- /dev/null +++ b/content/en/methods/page/Prev.md @@ -0,0 +1,53 @@ +--- +title: Prev +description: Returns the previous page in a global page collection, relative to the given page. +categories: [] +keywords: [] +action: + related: + - methods/page/Next + - methods/page/PrevInSection + - methods/page/NextInSection + - methods/pages/Prev + - methods/pages/Next + returnType: hugolib.pageState + signatures: [PAGE.Prev] +toc: true +--- + +The behavior of the `Prev` and `Next` methods on a `Page` object is probably the reverse of what you expect. + +With this content structure: + +```text +content/ +├── pages/ +│ ├── _index.md +│ ├── page-1.md <-- front matter: weight = 10 +│ ├── page-2.md <-- front matter: weight = 20 +│ └── page-3.md <-- front matter: weight = 30 +└── _index.md +``` + +When you visit page-2: + +- The `Prev` method points to page-3 +- The `Next` method points to page-1 + +{{% note %}} +Use the opposite label in your navigation links as shown in the example below. +{{% /note %}} + +```go-html-template +{{ with .Next }} + Prev +{{ end }} + +{{ with .Prev }} + Next +{{ end }} +``` + +## Compare to Pages methods + +{{% include "methods/_common/next-prev-on-page-vs-next-prev-on-pages.md" %}} diff --git a/content/en/methods/page/PrevInSection.md b/content/en/methods/page/PrevInSection.md new file mode 100644 index 00000000000..b3206e25e3f --- /dev/null +++ b/content/en/methods/page/PrevInSection.md @@ -0,0 +1,72 @@ +--- +title: PrevInSection +description: Returns the previous page within a section, relative to the given page. +categories: [] +keywords: [] +action: + related: + - methods/page/NextInSection + - methods/page/Next + - methods/pages/Next + - methods/page/Prev + - methods/pages/Prev + returnType: hugolib.pageState + signatures: [PAGE.PrevInSection] +--- + + +The behavior of the `PrevInSection` and `NextInSection` methods on a `Page` object is probably the reverse of what you expect. + +With this content structure: + +```text +content/ +├── books/ +│ ├── _index.md +│ ├── book-1.md +│ ├── book-2.md +│ └── book-3.md +├── films/ +│ ├── _index.md +│ ├── film-1.md +│ ├── film-2.md +│ └── film-3.md +└── _index.md +``` + +When you visit book-2: + +- The `PrevInSection` method points to book-3 +- The `NextInSection` method points to book-1 + +{{% note %}} +Use the opposite label in your navigation links as shown in the example below. +{{% /note %}} + +```go-html-template +{{ with .NextInSection }} + Previous in section +{{ end }} + +{{ with .PrevInSection }} + Next in section +{{ end }} +``` + +{{% note %}} +The navigation sort order may be different than the page collection sort order. +{{% /note %}} + +With the `PrevInSection` and `NextInSection` methods, the navigation sort order is fixed, using Hugo’s default sort order. In order of precedence: + +1. Page [weight] +2. Page [date] +3. Page [linkTitle], falling back to page [title] +4. Page file path if the page is backed by a file + +For example, with a page collection sorted by title, the navigation sort order will use Hugo’s default sort order. This is probably not what you want or expect. For this reason, the Next and Prev methods on a Pages object are a generally a better choice. + +[date]: /methods/page/date +[weight]: /methods/page/weight +[linkTitle]: /methods/page/linktitle +[title]: /methods/page/title diff --git a/content/en/methods/page/PublishDate.md b/content/en/methods/page/PublishDate.md new file mode 100644 index 00000000000..b1c0717a9b8 --- /dev/null +++ b/content/en/methods/page/PublishDate.md @@ -0,0 +1,35 @@ +--- +title: PublishDate +description: Returns the publish date of the given page. +categories: [] +keywords: [] +action: + related: + - methods/page/Date + - methods/page/ExpiryDate + - methods/page/LastMod + returnType: time.Time + signatures: [PAGE.PublishDate] +--- + +By default, Hugo excludes pages with future publish dates when building your site. To include future pages, use the `--buildFuture` command line flag. + +Set the publish date in front matter: + +{{< code-toggle file=content/news/article-1.md fm=true >}} +title = 'Article 1' +publishDate = 2023-10-19T00:40:04-07:00 +{{< /code-toggle >}} + +The publish date is a [time.Time] value. Format and localize the value with the [`time.Format`] function, or use it with any of the [time methods]. + +```go-html-template +{{ .PublishDate | time.Format ":date_medium" }} → Oct 19, 2023 +``` + +In the example above we explicitly set the publish date in front matter. With Hugo's default configuration, the `PublishDate` method returns the front matter value. This behavior is configurable, allowing you to set fallback values if the publish date is not defined in front matter. See [details]. + +[`time.Format`]: /functions/time/format +[details]: /getting-started/configuration/#configure-dates +[time methods]: /methods/time +[time.Time]: https://pkg.go.dev/time#Time diff --git a/content/en/methods/page/RawContent.md b/content/en/methods/page/RawContent.md new file mode 100644 index 00000000000..9fea16db606 --- /dev/null +++ b/content/en/methods/page/RawContent.md @@ -0,0 +1,31 @@ +--- +title: RawContent +description: Returns the raw content of the given page. +categories: [] +keywords: [] +action: + related: + - methods/page/Content + - methods/page/Plain + - methods/page/PlainWords + - methods/page/RenderShortcodes + returnType: string + signatures: [PAGE.RawContent] +--- + +The `RawContent` method on a `Page` object returns the raw content. The raw content does not include front matter. + +```go-html-template +{{ .RawContent }} +``` + +This is useful when rendering a page in a plain text [content format]. + +{{% note %}} +[Shortcodes] within the content are not rendered. To get the raw content with shortcodes rendered, use the [`RenderShortcodes`] method on a `Page` object. + +[shortcodes]: /getting-started/glossary/#shortcode +[`RenderShortcodes`]: /methods/page/rendershortcodes +{{% /note %}} + +[content format]: /templates/output-formats diff --git a/content/en/methods/page/ReadingTime.md b/content/en/methods/page/ReadingTime.md new file mode 100644 index 00000000000..531824b9b58 --- /dev/null +++ b/content/en/methods/page/ReadingTime.md @@ -0,0 +1,49 @@ +--- +title: ReadingTime +description: Returns the estimated reading time, in minutes, for the given page. +categories: [] +keywords: [] +action: + related: + - methods/page/WordCount + - methods/page/FuzzyWordCount + returnType: int + signatures: [PAGE.ReadingTime] +--- + +The estimated reading time is calculated by dividing the number of words in the content by the reading speed. + +By default, Hugo assumes a reading speed of 212 words per minute. For CJK languages, it assumes 500 words per minute. + +```go-html-template +{{ printf "Estimated reading time: %d minutes" .ReadingTime }} +``` + +Reading speed varies by language. Create language-specific estimated reading times on your multilingual site using site parameters. + +{{< code-toggle file=hugo >}} +[languages] + [languages.de] + contentDir = 'content/de' + languageCode = 'de-DE' + languageName = 'Deutsch' + weight = 2 + [languages.de.params] + reading_speed = 179 + [languages.en] + contentDir = 'content/en' + languageCode = 'en-US' + languageName = 'English' + weight = 1 + [languages.en.params] + reading_speed = 228 +{{< /code-toggle >}} + +Then in your template: + +```go-html-template +{{ $readingTime := div (float .WordCount) .Site.Params.reading_speed }} +{{ $readingTime = math.Ceil $readingTime }} +``` + +We cast the `.WordCount` to a float to obtain a float when we divide by the reading speed. Then round up to the nearest integer. diff --git a/content/en/methods/page/Ref.md b/content/en/methods/page/Ref.md new file mode 100644 index 00000000000..2f52a74c11a --- /dev/null +++ b/content/en/methods/page/Ref.md @@ -0,0 +1,44 @@ +--- +title: Ref +description: Returns the absolute URL of the given page with the given path, language, and output format. +categories: [] +keywords: [] +action: + related: + - methods/page/RelRef + - functions/urls/RelRef + - functions/urls/Ref + returnType: string + signatures: [PAGE.Ref OPTIONS] +--- + +The map of option contains: + +path +: (`string`) The path to the page, relative to the content directory. Required. + +lang +: (`string`) The language (site) to search for the page. Default is the current language. Optional. + +outputFormat +: (`string`) The output format to search for the page. Default is the current output format. Optional. + +The examples below show the rendered output when visiting a page on the English language version of the site: + +```go-html-template +{{ $opts := dict "path" "/books/book-1" }} +{{ .Ref $opts }} → http://localhost:1314/en/books/book-1/ + +{{ $opts := dict "path" "/books/book-1" "lang" "de" }} +{{ .Ref $opts }} → http://localhost:1314/de/books/book-1/ + +{{ $opts := dict "path" "/books/book-1" "lang" "de" "outputFormat" "json" }} +{{ .Ref $opts }} → http://localhost:1314/de/books/book-1/index.json +``` + +By default, Hugo will throw an error and fail the build if it cannot resolve the path. You can change this to a warning in your site configuration, and specify a URL to return when the path cannot be resolved. + +{{< code-toggle file=hugo >}} +refLinksErrorLevel = 'warning' +refLinksNotFoundURL = '/some/other/url' +{{< /code-toggle >}} diff --git a/content/en/methods/page/RegularPages.md b/content/en/methods/page/RegularPages.md new file mode 100644 index 00000000000..da433ec8023 --- /dev/null +++ b/content/en/methods/page/RegularPages.md @@ -0,0 +1,87 @@ +--- +title: RegularPages +description: Returns a collection of regular pages within the current section. +categories: [] +keywords: [] +action: + related: + - methods/page/Pages + - methods/page/RegularPagesRecursive + returnType: page.Pages + signatures: [PAGE.RegularPages] +--- + +The `RegularPages` method on a `Page` object is available to these [page kinds]: `home`, `section`, `taxonomy`, and `term`. The templates for these page kinds receive a page [collection] in [context]. + +Range through the page collection in your template: + +```go-html-template +{{ range .RegularPages.ByTitle }} +

{{ .LinkTitle }}

+{{ end }} +``` + +Consider this content structure: + +```text +content/ +├── lessons/ +│ ├── lesson-1/ +│ │ ├── _index.md +│ │ ├── part-1.md +│ │ └── part-2.md +│ ├── lesson-2/ +│ │ ├── resources/ +│ │ │ ├── task-list.md +│ │ │ └── worksheet.md +│ │ ├── _index.md +│ │ ├── part-1.md +│ │ └── part-2.md +│ ├── _index.md +│ ├── grading-policy.md +│ └── lesson-plan.md +├── _index.md +├── contact.md +└── legal.md +``` + +When rendering the home page, the `RegularPages` method returns: + + contact.md + legal.md + +When rendering the lessons page, the `RegularPages` method returns: + + lessons/grading-policy.md + lessons/lesson-plan.md + +When rendering lesson-1, the `RegularPages` method returns: + + lessons/lesson-1/part-1.md + lessons/lesson-1/part-2.md + +When rendering lesson-2, the `RegularPages` method returns: + + lessons/lesson-2/part-1.md + lessons/lesson-2/part-2.md + lessons/lesson-2/resources/task-list.md + lessons/lesson-2/resources/worksheet.md + +In the last example, the collection includes pages in the resources subdirectory. That directory is not a [section]---it does not contain an _index.md file. Its contents are part of the lesson-2 section. + +{{% note %}} +When used with the `Site` object, the `RegularPages` method recursively returns all regular pages within the site. See [details]. + +[details]: /methods/site/regularpages +{{% /note %}} + +```go-html-template +{{ range .Site.RegularPages.ByTitle }} +

{{ .LinkTitle }}

+{{ end }} +``` + +[collection]: /getting-started/glossary/#collection +[context]: /getting-started/glossary/#context +[page kinds]: /getting-started/glossary/#page-kind +[section]: /getting-started/glossary/#section diff --git a/content/en/methods/page/RegularPagesRecursive.md b/content/en/methods/page/RegularPagesRecursive.md new file mode 100644 index 00000000000..5a314cef3e6 --- /dev/null +++ b/content/en/methods/page/RegularPagesRecursive.md @@ -0,0 +1,90 @@ +--- +title: RegularPagesRecursive +description: Returns a collection of regular pages within the current section, and regular pages within all descendant sections. +categories: [] +keywords: [] +action: + related: + - methods/page/Pages + - methods/page/RegularPages + returnType: page.Pages + signatures: [PAGE.RegularPagesRecursive] +--- + +The `RegularPagesRecursive` method on a `Page` object is available to these [page kinds]: `home`, `section`, `taxonomy`, and `term`. The templates for these page kinds receive a page [collection] in [context]. + +Range through the page collection in your template: + +```go-html-template +{{ range .RegularPagesRecursive.ByTitle }} +

{{ .LinkTitle }}

+{{ end }} +``` + +Consider this content structure: + +```text +content/ +├── lessons/ +│ ├── lesson-1/ +│ │ ├── _index.md +│ │ ├── part-1.md +│ │ └── part-2.md +│ ├── lesson-2/ +│ │ ├── resources/ +│ │ │ ├── task-list.md +│ │ │ └── worksheet.md +│ │ ├── _index.md +│ │ ├── part-1.md +│ │ └── part-2.md +│ ├── _index.md +│ ├── grading-policy.md +│ └── lesson-plan.md +├── _index.md +├── contact.md +└── legal.md +``` + +When rendering the home page, the `RegularPagesRecursive` method returns: + + contact.md + lessons/grading-policy.md + legal.md + lessons/lesson-plan.md + lessons/lesson-2/part-1.md + lessons/lesson-1/part-1.md + lessons/lesson-2/part-2.md + lessons/lesson-1/part-2.md + lessons/lesson-2/resources/task-list.md + lessons/lesson-2/resources/worksheet.md + +When rendering the lessons page, the `RegularPagesRecursive` method returns: + + lessons/grading-policy.md + lessons/lesson-plan.md + lessons/lesson-2/part-1.md + lessons/lesson-1/part-1.md + lessons/lesson-2/part-2.md + lessons/lesson-1/part-2.md + lessons/lesson-2/resources/task-list.md + lessons/lesson-2/resources/worksheet.md + +When rendering lesson-1, the `RegularPagesRecursive` method returns: + + lessons/lesson-1/part-1.md + lessons/lesson-1/part-2.md + +When rendering lesson-2, the `RegularPagesRecursive` method returns: + + lessons/lesson-2/part-1.md + lessons/lesson-2/part-2.md + lessons/lesson-2/resources/task-list.md + lessons/lesson-2/resources/worksheet.md + +{{% note %}} +The `RegularPagesRecursive` method in not available on a `Site` object. +{{% /note %}} + +[collection]: /getting-started/glossary/#collection +[context]: /getting-started/glossary/#context +[page kinds]: /getting-started/glossary/#page-kind diff --git a/content/en/methods/page/RelPermalink.md b/content/en/methods/page/RelPermalink.md new file mode 100644 index 00000000000..8a5972ccc22 --- /dev/null +++ b/content/en/methods/page/RelPermalink.md @@ -0,0 +1,25 @@ +--- +title: RelPermalink +description: Returns the relative permalink of the given page. +categories: [] +keywords: [] +action: + related: + - methods/page/Permalink + returnType: string + signatures: [PAGE.RelPermalink] +--- + +Site configuration: + +{{< code-toggle file=hugo >}} +title = 'Documentation' +baseURL = 'https://example.org/docs/' +{{< /code-toggle >}} + +Template: + +```go-html-template +{{ $page := .Site.GetPage "/about" }} +{{ $page.Permalink }} → /docs/about/ +``` diff --git a/content/en/methods/page/RelRef.md b/content/en/methods/page/RelRef.md new file mode 100644 index 00000000000..3d30bfbe988 --- /dev/null +++ b/content/en/methods/page/RelRef.md @@ -0,0 +1,44 @@ +--- +title: RelRef +description: Returns the relative URL of the given page with the given path, language, and output format. +categories: [] +keywords: [] +action: + related: + - methods/page/Ref + - functions/urls/Ref + - functions/urls/RelRef + returnType: string + signatures: [PAGE.RelRef OPTIONS] +--- + +The map of option contains: + +path +: (`string`) The path to the page, relative to the content directory. Required. + +lang +: (`string`) The language (site) to search for the page. Default is the current language. Optional. + +outputFormat +: (`string`) The output format to search for the page. Default is the current output format. Optional. + +The examples below show the rendered output when visiting a page on the English language version of the site: + +```go-html-template +{{ $opts := dict "path" "/books/book-1" }} +{{ .RelRef $opts }} → /en/books/book-1/ + +{{ $opts := dict "path" "/books/book-1" "lang" "de" }} +{{ .RelRef $opts }} → /de/books/book-1/ + +{{ $opts := dict "path" "/books/book-1" "lang" "de" "outputFormat" "json" }} +{{ .RelRef $opts }} → /de/books/book-1/index.json +``` + +By default, Hugo will throw an error and fail the build if it cannot resolve the path. You can change this to a warning in your site configuration, and specify a URL to return when the path cannot be resolved. + +{{< code-toggle file=hugo >}} +refLinksErrorLevel = 'warning' +refLinksNotFoundURL = '/some/other/url' +{{< /code-toggle >}} diff --git a/content/en/methods/page/Render.md b/content/en/methods/page/Render.md new file mode 100644 index 00000000000..bc3f58352c0 --- /dev/null +++ b/content/en/methods/page/Render.md @@ -0,0 +1,75 @@ +--- +title: Render +description: Renders the given template with the given page as context. +categories: [] +keywords: [] +action: + related: + - functions/partials/Include + - functions/partials/IncludeCached + returnType: template.HTML + signatures: [PAGE.Render NAME] +aliases: [/functions/render] +--- + +Typically used when ranging over a page collection, the `Render` method on a `Page` object renders the given template, passing the given page as context. + +```go-html-template +{{ range site.RegularPages }} +

{{ .LinkTitle }}

+ {{ .Render "summary" }} +{{ end }} +``` + +In the example above, note that the template ("summary") is identified by its file name without directory or extension. + +Although similar to the [`partial`] function, there are key differences. + +`Render` method|`partial` function| +:--|:-- +The `Page` object is automatically passed to the given template. You cannot pass additional context.| You must specify the context, allowing you to pass a combination of objects, slices, maps, and scalars. +The path to the template is determined by the [content type].|You must specify the path to the template, relative to the layouts/partials directory. + +Consider this layout structure: + +```text +layouts/ +├── _default/ +│ ├── baseof.html +│ ├── home.html +│ ├── li.html <-- used for other content types +│ ├── list.html +│ ├── single.html +│ └── summary.html +└── books/ + ├── li.html <-- used when content type is "books" + └── summary.html +``` + +And this template: + +```go-html-template +
    + {{ range site.RegularPages.ByDate }} + {{ .Render "li" }} + {{ end }} +
+``` + +When rendering content of type "books" the `Render` method calls: + +```text +layouts/books/li.html +``` + +For all other content types the `Render` methods calls: + +```text +layouts/_default/li.html +``` + +See [content views] for more examples. + +[content views]: /templates/views +[`partial`]: /functions/partials/include +[content type]: /getting-started/glossary/#content-type diff --git a/content/en/methods/page/RenderShortcodes.md b/content/en/methods/page/RenderShortcodes.md new file mode 100644 index 00000000000..a62893f52ed --- /dev/null +++ b/content/en/methods/page/RenderShortcodes.md @@ -0,0 +1,76 @@ +--- +title: RenderShortcodes +description: Renders all shortcodes in the content of the given page, preserving the surrounding markup. +categories: [] +keywords: [] +action: + related: + - methods/page/RenderString + - methods/page/Content + - methods/page/RawContent + - methods/page/Plain + - methods/page/PlainWords + returnType: template.HTML + signatures: [PAGE.RenderShortcodes] +toc: true +--- + +Use this method in shortcode templates to compose a page from multiple content files, while preserving a global context for footnotes and the table of contents. + +For example: + +{{< code file="layouts/shortcodes/include.html" >}} +{{ $p := site.GetPage (.Get 0) }} +{{ $p.RenderShortcodes }} +{{< /code >}} + +Then in your markdown: + +{{< code file="content/about.md" lang=md >}} +{{%/* include "/snippets/services.md" */%}} +{{%/* include "/snippets/values.md" */%}} +{{%/* include "/snippets/leadership.md" */%}} +{{< /code >}} + +Each of the included markdown files can contain calls to other shortcodes. + +## Shortcode notation + +In the example above it's important to understand the difference between the two delimiters used when calling a shortcode: + +- `{{}}` tells Hugo that the rendered shortcode does not need further processing. For example, the shortcode content is HTML. +- `{{%/* myshortcode */%}}` tells Hugo that the rendered shortcode needs further processing. For example, the shortcode content is markdown. + +Use the latter for the "include" shortcode described above. + +## Further explanation + +To understand what is returned by the `RenderShortcodes` method, consider this content file + +{{< code file="content/about.md" lang=text >}} ++++ +title = 'About' +date = 2023-10-07T12:28:33-07:00 ++++ + +{{}} + +An *emphasized* word. +{{< /code >}} + +With this template code: + +```go-html-template +{{ $p := site.GetPage "/about" }} +{{ $p.RenderShortcodes }} +``` + +Hugo renders this:; + +```html +https://example.org/privacy/ + +An *emphasized* word. +``` + +Note that the shortcode within the content file was rendered, but the surrounding markdown was preserved. diff --git a/content/en/methods/page/RenderString.md b/content/en/methods/page/RenderString.md new file mode 100644 index 00000000000..5782cd2b1ac --- /dev/null +++ b/content/en/methods/page/RenderString.md @@ -0,0 +1,51 @@ +--- +title: RenderString +description: Renders markup to HTML. +categories: [] +keywords: [] +action: + related: + - methods/page/RenderShortcodes + - functions/transform/Markdownify + returnType: template.HTML + signatures: ['PAGE.RenderString [OPTIONS] MARKUP'] +aliases: [/functions/renderstring] +--- + +```go-html-template +{{ $s := "An *emphasized* word" }} +{{ $s | .RenderString }} → An emphasized word +``` + +This method takes an optional map of options: + +display +: (`string`) Specify either `inline` or `block`. If `inline`, removes surrounding `p` tags from short snippets. Default is `inline`. + +markup +: (`string`) Specify a [markup identifier] for the provided markup. Default is the `markup` front matter value, falling back to the value derived from the page's file extension. + +Render with the default markup renderer: + +```go-html-template +{{ $s := "An *emphasized* word" }} +{{ $s | .RenderString }} → An emphasized word + +{{ $opts := dict "display" "block" }} +{{ $s | .RenderString $opts }} →

An emphasized word

+``` + +Render with [Pandoc]: + +```go-html-template +{{ $s := "H~2~O" }} + +{{ $opts := dict "markup" "pandoc" }} +{{ $s | .RenderString $opts }} → H2O + +{{ $opts := dict "display" "block" "markup" "pandoc" }} +{{ .RenderString $opts $s }} →

H2O

+``` + +[markup identifier]: /content-management/formats/#list-of-content-formats +[pandoc]: https://www.pandoc.org/ diff --git a/content/en/methods/page/Resources.md b/content/en/methods/page/Resources.md new file mode 100644 index 00000000000..a9fa3dab2f5 --- /dev/null +++ b/content/en/methods/page/Resources.md @@ -0,0 +1,81 @@ +--- +title: Resources +description: Returns a collection of page resources. +categories: [] +keywords: [] +action: + related: + - functions/resources/ByType + - functions/resources/Get + - functions/resources/GetMatch + - functions/resources/GetRemote + - functions/resources/Match + returnType: resource.Resources + signatures: [PAGE.Resources] +toc: true +--- + +The `Resources` method on a `Page` object returns a collection of page resources. A page resource is a file within a [page bundle]. + +To work with global or remote resources, see the [`resources`] functions. + +## Methods + +ByType +: (`resource.Resources`) Returns a collection of page resources of the given [media type], or nil if none found. The media type is typically one of `image`, `text`, `audio`, `video`, or `application`. + +```go-html-template +{{ range .Resources.ByType "image" }} + +{{ end }} +``` + +When working with global resources instead of page resources, use the [`resources.ByType`] function. + +Get +: (`resource.Resource`) Returns a page resource from the given path, or nil if none found. + +```go-html-template +{{ with .Resources.Get "images/a.jpg" }} + +{{ end }} +``` + +When working with global resources instead of page resources, use the [`resources.Get`] function. + +GetMatch +: (`resource.Resource`) Returns the first page resource from paths matching the given [glob pattern], or nil if none found. + +```go-html-template +{{ with .Resources.GetMatch "images/*.jpg" }} + +{{ end }} +``` + +When working with global resources instead of page resources, use the [`resources.GetMatch`] function. + +Match +: (`resource.Resources`) Returns a collection of page resources from paths matching the given [glob pattern], or nil if none found. + +```go-html-template +{{ range .Resources.Match "images/*.jpg" }} + +{{ end }} +``` + +When working with global resources instead of page resources, use the [`resources.Match`] function. + +## Pattern matching + +With the `GetMatch` and `Match` methods, Hugo determines a match using a case-insensitive [glob pattern]. + +{{% include "functions/_common/glob-patterns.md" %}} + +[`resources.ByType`]: /functions/resources/ByType +[`resources.GetMatch`]: /functions/resources/ByType +[`resources.Get`]: /functions/resources/ByType +[`resources.Match`]: /functions/resources/ByType +[`resources`]: /functions/resources +[glob pattern]: https://github.com/gobwas/glob#example +[media type]: https://en.wikipedia.org/wiki/Media_type +[page bundle]: /getting-started/glossary/#page-bundle diff --git a/content/en/methods/page/Scratch.md b/content/en/methods/page/Scratch.md new file mode 100644 index 00000000000..f9ce7f7fb68 --- /dev/null +++ b/content/en/methods/page/Scratch.md @@ -0,0 +1,23 @@ +--- +title: Scratch +description: Creates a "scratch pad" on the given page to store and manipulate data. +categories: [] +keywords: [] +action: + related: + - methods/page/Store + - functions/collections/NewScratch + returnType: maps.Scratch + signatures: [PAGE.Scratch] +aliases: [/extras/scratch/,/doc/scratch/,/functions/scratch] +--- + +The `Scratch` method on a `Page` object creates a [scratch pad] to store and manipulate data. To create a scratch pad that is not reset on server rebuilds, use the [`Store`] method instead. + +To create a locally scoped scratch pad that is not attached to a `Page` object, use the [`newScratch`] function. + +[`Store`]: /methods/page/store +[`newScratch`]: functions/collections/newscratch +[scratch pad]: /getting-started/glossary/#scratch-pad + +{{% include "methods/page/_common/scratch-methods.md" %}} diff --git a/content/en/methods/page/Section.md b/content/en/methods/page/Section.md new file mode 100644 index 00000000000..30c8a983788 --- /dev/null +++ b/content/en/methods/page/Section.md @@ -0,0 +1,54 @@ +--- +title: Section +description: Returns the name of the top level section in which the given page resides. +categories: [] +keywords: [] +action: + related: + - methods/page/Type + returnType: string + signatures: [PAGE.Section] +--- + +With this content structure: + +```text +content/ +├── lessons/ +│ ├── math/ +│ │ ├── _index.md +│ │ ├── lesson-1.md +│ │ └── lesson-2.md +│ └── _index.md +└── _index.md +``` + +When rendering lesson-1.md: + +```go-html-template +{{ .Section }} → lessons +``` + +In the example above "lessons" is the top level section. + +The `Section` method is often used with the [`where`] function to build a page collection. + +```go-html-template +{{ range where .Site.RegularPages "Section" "lessons" }} +

{{ .LinkTitle }}

+{{ end }} +``` + +This is similar to using the [`Type`] method with the `where` function + +```go-html-template +{{ range where .Site.RegularPages "Type" "lessons" }} +

{{ .LinkTitle }}

+{{ end }} +``` + +However, if the `type` field in front matter has been defined on one or more pages, the page collection based on `Type` will be different than the page collection based on `Section`. + + +[`where`]: /functions/collections/where +[`Type`]: /methods/page/type diff --git a/content/en/methods/page/Sections.md b/content/en/methods/page/Sections.md new file mode 100644 index 00000000000..68795c50eaa --- /dev/null +++ b/content/en/methods/page/Sections.md @@ -0,0 +1,69 @@ +--- +title: Sections +description: Returns a collection of section pages, one for each immediate descendant section of the given page. +categories: [] +keywords: [] +action: + related: + - methods/page/Ancestors + - methods/page/CurrentSection + - methods/page/FirstSection + - methods/page/InSection + - methods/page/IsAncestor + - methods/page/IsDescendant + - methods/page/Parent + returnType: page.Pages + signatures: [PAGE.Sections] +--- + +{{% include "methods/page/_common/definition-of-section.md" %}} + +With this content structure: + +```text +content/ +├── auctions/ +│ ├── 2023-11/ <-- front matter: weight = 202311 +│ │ ├── _index.md +│ │ ├── auction-1.md +│ │ └── auction-2.md +│ ├── 2023-12/ +│ │ ├── _index.md <-- front matter: weight = 202312 +│ │ ├── auction-3.md +│ │ └── auction-4.md +│ ├── _index.md <-- front matter: weight = 30 +│ ├── bidding.md +│ └── payment.md +├── books/ +│ ├── _index.md <-- front matter: weight = 10 +│ ├── book-1.md +│ └── book-2.md +├── films/ +│ ├── _index.md <-- front matter: weight = 20 +│ ├── film-1.md +│ └── film-2.md +└── _index.md +``` + +And this template: + +```go-html-template +{{ range .Sections.ByWeight }} +

{{ .LinkTitle }}

+{{ end }} +``` + +On the home page, Hugo renders: + +```html +

Films

+

Books

+

Auctions

+``` + +On the auctions page, Hugo renders: + +```html +

Auctions in November 2023

+

Auctions in December 2023

+``` diff --git a/content/en/methods/page/Site.md b/content/en/methods/page/Site.md new file mode 100644 index 00000000000..34748facd52 --- /dev/null +++ b/content/en/methods/page/Site.md @@ -0,0 +1,19 @@ +--- +title: Site +description: Returns the Site object. +categories: [] +keywords: [] +action: + related: + - methods/page/Sites + returnType: page.siteWrapper + signatures: [PAGE.Site] +--- + +See [Site methods]. + +[Site methods]: /methods/site + +```go-html-template +{{ .Site.Title }} +``` diff --git a/content/en/methods/page/Sitemap.md b/content/en/methods/page/Sitemap.md new file mode 100644 index 00000000000..08ff3f5d067 --- /dev/null +++ b/content/en/methods/page/Sitemap.md @@ -0,0 +1,70 @@ +--- +title: Sitemap +description: Returns the sitemap settings for the given page as defined in front matter, falling back to the sitemap settings as defined in the site configuration. +categories: [] +keywords: [] +action: + related: [] + returnType: config.SitemapConfig + signatures: [PAGE.Sitemap] +toc: true +--- + +Access to the `Sitemap` method on a `Page` object is restricted to [sitemap templates]. + +## Methods + +ChangeFreq +: (`string`) How frequently a page is likely to change. Valid values are `always`, `hourly`, `daily`, `weekly`, `monthly`, `yearly`, and `never`. Default is "" (change frequency omitted from rendered sitemap). + +```go-html-template +{{ .Sitemap.ChangeFreq }} +``` + +Priority +: (`float`) The priority of a page relative to any other page on the site. Valid values range from 0.0 to 1.0. Default is -1 (priority omitted from rendered sitemap). + +```go-html-template +{{ .Sitemap.Priority }} +``` + +## Example + +With this site configuration: + +{{< code-toggle file=hugo >}} +[sitemap] +changeFreq = 'monthly' +{{< /code-toggle >}} + +And this content: + +{{< code-toggle file=content/news.md fm=true >}} +title = 'News' +[sitemap] +changeFreq = 'hourly' +{{< /code-toggle >}} + +And this simplistic sitemap template: + +{{< code file=layouts/_default/sitemap.xml >}} +{{ printf "" | safeHTML }} + + {{ range .Pages }} + + {{ .Permalink }} + {{ if not .Lastmod.IsZero }} + {{ .Lastmod.Format "2006-01-02T15:04:05-07:00" | safeHTML }} + {{ end }} + {{ with .Sitemap.ChangeFreq }} + {{ . }} + {{ end }} + + {{ end }} + +{{< /code >}} + +The change frequency will be `hourly` for the news page, and `monthly` for other pages. + +[sitemap templates]: /templates/sitemap-template/ diff --git a/content/en/methods/page/Sites.md b/content/en/methods/page/Sites.md new file mode 100644 index 00000000000..1fbdfcdcde8 --- /dev/null +++ b/content/en/methods/page/Sites.md @@ -0,0 +1,69 @@ +--- +title: Sites +description: Returns a collection of all Site objects, one for each language, ordered by language weight. +categories: [] +keywords: [] +action: + related: + - methods/page/Site + returnType: page.Sites + signatures: [PAGE.Sites] +--- + +This is a convenience method to access `.Site.Sites`. + +With this site configuration: + +{{< code-toggle file=hugo >}} +defaultContentLanguage = 'de' +defaultContentLanguageInSubdir = false + +[languages.de] +languageCode = 'de-DE' +languageDirection = 'ltr' +languageName = 'Deutsch' +title = 'Projekt Dokumentation' +weight = 1 + +[languages.en] +languageCode = 'en-US' +languageDirection = 'ltr' +languageName = 'English' +title = 'Project Documentation' +weight = 2 +{{< /code-toggle >}} + +This template: + +```go-html-template + +``` + +Produces a list of links to each home page: + +```html + +``` + +To render a link to home page of the primary (first) language: + +```go-html-template +{{ with .Sites.First }} + {{ .Title }} +{{ end }} +``` + +This is equivalent to: + +```go-html-template +{{ with index .Sites 0 }} + {{ .Title }} +{{ end }} +``` diff --git a/content/en/methods/page/Slug.md b/content/en/methods/page/Slug.md new file mode 100644 index 00000000000..9fdb09b5774 --- /dev/null +++ b/content/en/methods/page/Slug.md @@ -0,0 +1,25 @@ +--- +title: Slug +description: Returns the URL slug of the given page as defined in front matter. +categories: [] +keywords: [] +action: + related: [] + returnType: string + signatures: [PAGE.Slug] +--- + +{{< code-toggle file=content/recipes/spicy-tuna-hand-rolls.md fm=true >}} +title = 'How to make spicy tuna hand rolls' +slug = 'sushi' +{{< /code-toggle >}} + +This page will be served from: + + https://example.org/recipes/sushi + +To get the slug value within a template: + +```go-html-template +{{ .Slug }} → sushi +``` diff --git a/content/en/methods/page/Store.md b/content/en/methods/page/Store.md new file mode 100644 index 00000000000..04e1d525663 --- /dev/null +++ b/content/en/methods/page/Store.md @@ -0,0 +1,97 @@ +--- +title: Store +description: Creates a persistent "scratch pad" on the given page to store and manipulate data. +categories: [] +keywords: [] +action: + related: + - methods/page/scratch + - functions/collections/NewScratch + returnType: maps.Scratch + signatures: [PAGE.Store] +aliases: [/functions/store] +--- + +The `Store` method on a `Page` object creates a persistent [scratch pad] to store and manipulate data. In contrast with the [`Scratch`] method, the scratch pad created by the `Store` method is not reset on server rebuilds. + +To create a locally scoped scratch pad that is not attached to a `Page` object, use the [`newScratch`] function. + +[`Scratch`]: /methods/page/scratch +[`newScratch`]: functions/collections/newscratch +[scratch pad]: /getting-started/glossary/#scratch-pad + +## Methods + +Set +: Sets the value of a given key. + +```go-html-template +{{ .Store.Set "greeting" "Hello" }} +``` + +Get +: Gets the value of a given key. + +```go-html-template +{{ .Store.Set "greeting" "Hello" }} +{{ .Store.Get "greeting" }} → Hello +``` + +Add +: Adds a given value to existing value(s) of the given key. + +: For single values, `Add` accepts values that support Go's `+` operator. If the first `Add` for a key is an array or slice, the following adds will be appended to that list. + +```go-html-template +{{ .Store.Set "greeting" "Hello" }} +{{ .Store.Add "greeting" "Welcome" }} +{{ .Store.Get "greeting" }} → HelloWelcome +``` + +```go-html-template +{{ .Store.Set "total" 3 }} +{{ .Store.Add "total" 7 }} +{{ .Store.Get "total" }} → 10 +``` + +```go-html-template +{{ .Store.Set "greetings" (slice "Hello") }} +{{ .Store.Add "greetings" (slice "Welcome" "Cheers") }} +{{ .Store.Get "greetings" }} → [Hello Welcome Cheers] +``` + +SetInMap +: Takes a `key`, `mapKey` and `value` and adds a map of `mapKey` and `value` to the given `key`. + +```go-html-template +{{ .Store.SetInMap "greetings" "english" "Hello" }} +{{ .Store.SetInMap "greetings" "french" "Bonjour" }} +{{ .Store.Get "greetings" }} → map[english:Hello french:Bonjour] +``` + +DeleteInMap +: Takes a `key` and `mapKey` and removes the map of `mapKey` from the given `key`. + +```go-html-template +{{ .Store.SetInMap "greetings" "english" "Hello" }} +{{ .Store.SetInMap "greetings" "french" "Bonjour" }} +{{ .Store.DeleteInMap "greetings" "english" }} +{{ .Store.Get "greetings" }} → map[french:Bonjour] +``` + +GetSortedMapValues +: Returns an array of values from `key` sorted by `mapKey`. + +```go-html-template +{{ .Store.SetInMap "greetings" "english" "Hello" }} +{{ .Store.SetInMap "greetings" "french" "Bonjour" }} +{{ .Store.GetSortedMapValues "greetings" }} → [Hello Bonjour] +``` + +Delete +: Removes the given key. + +```go-html-template +{{ .Store.Set "greeting" "Hello" }} +{{ .Store.Delete "greeting" }} +``` diff --git a/content/en/methods/page/Summary.md b/content/en/methods/page/Summary.md new file mode 100644 index 00000000000..37ce865893b --- /dev/null +++ b/content/en/methods/page/Summary.md @@ -0,0 +1,29 @@ +--- +title: Summary +description: Returns the content summary of the given page. +categories: [] +keywords: [] +action: + related: + - methods/page/Truncated + - methods/page/Description + returnType: template.HTML + signatures: [PAGE.Summary] +--- + +There are three ways to define the [content summary]: + +1. Let Hugo create the summary based on the first 70 words. You can change the number of words by setting the `summaryLength` in your site configuration. +2. Manually split the content with a `<--more-->` tag in markdown. Everything before the tag is included in the summary. +3. Create a `summary` field in front matter. + +To list the pages in a section with a summary beneath each link: + +```go-html-template +{{ range .Pages }} +

{{ .LinkTitle }}

+ {{ .Summary }} +{{ end }} +``` + +[content summary]: /content-management/summaries diff --git a/content/en/methods/page/TableOfContents.md b/content/en/methods/page/TableOfContents.md new file mode 100644 index 00000000000..2ab182e8ca7 --- /dev/null +++ b/content/en/methods/page/TableOfContents.md @@ -0,0 +1,47 @@ +--- +title: TableOfContents +description: Returns a table of contents for the given page. +categories: [] +keywords: [] +action: + related: + - methods/page/Fragments + returnType: template.HTML + signatures: [PAGE.TableOfContents] +--- + +The `TableOfContents` method on a `Page` object returns an ordered or unordered list of the markdown [ATX] and [setext] headings within the page content. + +[atx]: https://spec.commonmark.org/0.30/#atx-headings +[setext]: https://spec.commonmark.org/0.30/#setext-headings + +This template code: + +```go-html-template +{{ .TableOfContents }} +``` + +Produces this HTML: + +```html + +``` + +By default, the `TableOfContents` method returns an unordered list of level 2 and level 3 headings. You can adjust this in your site configuration: + +{{< code-toggle file=hugo >}} +[markup.tableOfContents] +endLevel = 3 +ordered = false +startLevel = 2 +{{< /code-toggle >}} diff --git a/content/en/methods/page/Title.md b/content/en/methods/page/Title.md new file mode 100644 index 00000000000..fc0475d592b --- /dev/null +++ b/content/en/methods/page/Title.md @@ -0,0 +1,23 @@ +--- +title: Title +description: Returns the title of the given page. +categories: [] +keywords: [] +action: + related: + - methods/page/LinkTitle + returnType: string + signatures: [PAGE.Title] +--- + +For pages backed by a file, the `.Title` method returns the `title` field in front matter. For section pages that are automatically generated, the `.Title` method returns the section name. + +{{< code-toggle file=content/about.md fm=true >}} +title = 'About us' +{{< /code-toggle >}} + +```go-html-template +{{ .Title }} → About us +``` + +With automatic section pages, the title is capitalized by default, using the `titleCaseStyle` defined in your site configuration. To disable this behavior, set `pluralizeListTitles` to `false` in your site configuration. diff --git a/content/en/methods/page/TranslationKey.md b/content/en/methods/page/TranslationKey.md new file mode 100644 index 00000000000..f022612ae12 --- /dev/null +++ b/content/en/methods/page/TranslationKey.md @@ -0,0 +1,74 @@ +--- +title: TranslationKey +description: Returns the translation key of the given page. +categories: [] +keywords: [] +action: + related: + - methods/page/Translations + - methods/page/AllTranslations + - methods/page/IsTranslated + returnType: string + signatures: [PAGE.TranslationKey] +--- + +The translation key creates a relationship between all translations of a given page. The translation key is derived from the file path, or from the `translationKey` parameter if defined in front matter. + +With this site configuration: + +{{< code-toggle file=hugo >}} +defaultContentLanguage = 'en' + +[languages.en] +contentDir = 'content/en' +languageCode = 'en-US' +languageName = 'English' +weight = 1 + +[languages.de] +contentDir = 'content/de' +languageCode = 'de-DE' +languageName = 'Deutsch' +weight = 2 +{{< /code-toggle >}} + +And this content: + +```text +content/ +├── de/ +│ ├── books/ +│ │ ├── buch-1.md +│ │ └── book-2.md +│ └── _index.md +├── en/ +│ ├── books/ +│ │ ├── book-1.md +│ │ └── book-2.md +│ └── _index.md +└── _index.md +``` + +And this front matter: + +{{< code-toggle file="content/en/books/book-1.md" fm=true >}} +title = 'Book 1' +translationKey = 'foo' +{{< /code-toggle >}} + +{{< code-toggle file="content/de/books/buch-1.md" fm=true >}} +title = 'Buch 1' +translationKey = 'foo' +{{< /code-toggle >}} + +When rendering either either of the pages above: + +```go-html-template +{{ .TranslationKey }} → page/foo +``` + +If the front matter of Book 2, in both languages, does not include a translation key: + +```go-html-template +{{ .TranslationKey }} → page/books/book-2 +``` diff --git a/content/en/methods/page/Translations.md b/content/en/methods/page/Translations.md new file mode 100644 index 00000000000..7aaee75ab7a --- /dev/null +++ b/content/en/methods/page/Translations.md @@ -0,0 +1,88 @@ +--- +title: Translations +description: Returns all translation of the given page, excluding the current language. +categories: [] +keywords: [] +action: + related: + - methods/page/AllTranslations + - methods/page/IsTranslated + - methods/page/TranslationKey + returnType: page.Pages + signatures: [PAGE.Translations] +--- + +With this site configuration: + +{{< code-toggle file=hugo >}} +defaultContentLanguage = 'en' + +[languages.en] +contentDir = 'content/en' +languageCode = 'en-US' +languageName = 'English' +weight = 1 + +[languages.de] +contentDir = 'content/de' +languageCode = 'de-DE' +languageName = 'Deutsch' +weight = 2 + +[languages.fr] +contentDir = 'content/fr' +languageCode = 'fr-FR' +languageName = 'Français' +weight = 3 +{{< /code-toggle >}} + +And this content: + +```text +content/ +├── de/ +│ ├── books/ +│ │ ├── book-1.md +│ │ └── book-2.md +│ └── _index.md +├── en/ +│ ├── books/ +│ │ ├── book-1.md +│ │ └── book-2.md +│ └── _index.md +├── fr/ +│ ├── books/ +│ │ └── book-1.md +│ └── _index.md +└── _index.md +``` + +And this template: + +```go-html-template +{{ with .Translations }} + +{{ end }} +``` + +Hugo will render this list on the "Book 1" page of the English site: + +```html + +``` + +Hugo will render this list on the "Book 2" page of the English site: + +```html + +``` diff --git a/content/en/methods/page/Truncated.md b/content/en/methods/page/Truncated.md new file mode 100644 index 00000000000..e6051f0cd4a --- /dev/null +++ b/content/en/methods/page/Truncated.md @@ -0,0 +1,35 @@ +--- +title: Truncated +description: Reports whether the content length exceeds the summary length. +categories: [] +keywords: [] +action: + related: + - methods/page/Summary + returnType: bool + signatures: [PAGE.Truncated] +--- + +There are three ways to define the [content summary]: + +1. Let Hugo create the summary based on the first 70 words. You can change the number of words by setting the `summaryLength` in your site configuration. +2. Manually split the content with a `<--more-->` tag in markdown. Everything before the tag is included in the summary. +3. Create a `summary` field in front matter. + +{{% note %}} +The `Truncated` method returns `false` if you define the summary in front matter. +{{% /note %}} + +The `Truncated` method returns `true` if the content length exceeds the summary length. This is useful for rendering a "read more" link: + +```go-html-template +{{ range .Pages }} +

{{ .LinkTitle }}

+ {{ .Summary }} + {{ if .Truncated }} + Read more... + {{ end }} +{{ end }} +``` + +[content summary]: /content-management/summaries diff --git a/content/en/methods/page/Type.md b/content/en/methods/page/Type.md new file mode 100644 index 00000000000..d504bc8b8a2 --- /dev/null +++ b/content/en/methods/page/Type.md @@ -0,0 +1,56 @@ +--- +title: Type +description: Returns the content type of the given page. +categories: [] +keywords: [] +action: + related: + - methods/page/Kind + - methods/page/Layout + - methods/page/Type + returnType: string + signatures: [PAGE.Type] +--- + +The `Type` method on a `Page` object returns the [content type] of the given page. The content type is defined by the `type` field in front matter, or inferred from the top-level directory name if the `type` field in front matter is not defined. + +With this content structure: + +```text +content/ +├── auction/ +│ ├── _index.md +│ ├── item-1.md +│ └── item-2.md <-- front matter: type = books +├── books/ +│ ├── _index.md +│ ├── book-1.md +│ └── book-2.md +├── films/ +│ ├── _index.md +│ ├── film-1.md +│ └── film-2.md +└── _index.md +``` + +To list the books, regardless of [section]: + +```go-html-template +{{ range where .Site.RegularPages.ByTitle "Type" "books" }} +

{{ .LinkTitle }}

+{{ end }} +``` + +Hugo renders this to; + +```html +

Book 1

+

Book 2

+

Item 2

+``` + +The `type` field in front matter is also useful for targeting a template. See [details]. + +[content type]: /getting-started/glossary/#content-type +[details]: /templates/lookup-order/#target-a-template +[section]: /getting-started/glossary/#section diff --git a/content/en/methods/page/Weight.md b/content/en/methods/page/Weight.md new file mode 100644 index 00000000000..75c75db86a9 --- /dev/null +++ b/content/en/methods/page/Weight.md @@ -0,0 +1,27 @@ +--- +title: Weight +description: Returns the weight of the given page as defined in front matter. +categories: [] +keywords: [] +action: + related: [] + returnType: int + signatures: [PAGE.Weight] +--- + +The `Weight` method on a `Page` object returns the [weight] of the given page as defined in front matter. + +[weight]: /getting-started/glossary/#weight + +{{< code-toggle file=content/recipes/sushi.md fm=true >}} +title = 'How to make spicy tuna hand rolls' +weight = 42 +{{< /code-toggle >}} + +Page weight controls the position of a page within a collection that is sorted by weight. Assign weights using non-zero integers. Lighter items float to the top, while heavier items sink to the bottom. Unweighted or zero-weighted elements are placed at the end of the collection. + +Although rarely used within a template, you can access the value with: + +```go-html-template +{{ .Weight }} → 42 +``` diff --git a/content/en/methods/page/WordCount.md b/content/en/methods/page/WordCount.md new file mode 100644 index 00000000000..bb1fdcf9414 --- /dev/null +++ b/content/en/methods/page/WordCount.md @@ -0,0 +1,20 @@ +--- +title: WordCount +description: Returns the number of words in the content of the given page. +categories: [] +keywords: [] +action: + related: + - methods/page/FuzzyWordCount + - methods/page/ReadingTime + returnType: int + signatures: [PAGE.WordCount] +--- + +```go-html-template +{{ .WordCount }} → 103 +``` + +To round up to nearest multiple of 100, use the [`FuzzyWordCount`] method. + +[`FuzzyWordCount`]: /methods/page/fuzzywordcount diff --git a/content/en/methods/page/_common/_index.md b/content/en/methods/page/_common/_index.md new file mode 100644 index 00000000000..47d5812fba5 --- /dev/null +++ b/content/en/methods/page/_common/_index.md @@ -0,0 +1,13 @@ +--- +cascade: + _build: + list: never + publishResources: false + render: never +--- + + diff --git a/content/en/methods/page/_common/definition-of-section.md b/content/en/methods/page/_common/definition-of-section.md new file mode 100644 index 00000000000..7dc60078939 --- /dev/null +++ b/content/en/methods/page/_common/definition-of-section.md @@ -0,0 +1,5 @@ +--- +# Do not remove front matter. +--- + +A _section_ is a top-level content directory, or any content directory with an _index.md file. diff --git a/content/en/methods/page/_common/output-format-definition.md b/content/en/methods/page/_common/output-format-definition.md new file mode 100644 index 00000000000..25944464a7c --- /dev/null +++ b/content/en/methods/page/_common/output-format-definition.md @@ -0,0 +1,11 @@ +--- +# Do not remove front matter. +--- + +Hugo generates one or more files per page when building a site. For example, when rendering home, [section], [taxonomy], and [term] pages, Hugo generates an HTML file and an RSS file. Both HTML and RSS are built-in _output formats_. Create multiple output formats, and control generation based on [page kind], or by enabling one or more output formats for one or more pages. See [details]. + +[section]: /getting-started/glossary/#section +[taxonomy]: /getting-started/glossary/#taxonomy +[term]: /getting-started/glossary/#term +[page kind]: /getting-started/glossary/#page-kind +[details]: /templates/output-formats diff --git a/content/en/methods/page/_common/output-format-methods.md b/content/en/methods/page/_common/output-format-methods.md new file mode 100644 index 00000000000..5e7111fe52b --- /dev/null +++ b/content/en/methods/page/_common/output-format-methods.md @@ -0,0 +1,27 @@ +--- +# Do not remove front matter. +--- + +Get IDENTIFIER +: (`any`) Returns the `OutputFormat` object with the given identifier. + +MediaType +: (`media.Type`) Returns the media type of the output format. + +MediaType.MainType +: (`string`) Returns the main type of the output format's media type. + +MediaType.SubType +: (`string`) Returns the subtype of the current format's media type. + +Name +: (`string`) Returns the output identifier of the output format. + +Permalink +: (`string`) Returns the permalink of the page generated by the current output format. + +Rel +: (`string`) Returns the `rel` value of the output format, either the default or as defined in the site configuration. + +RelPermalink +: (`string`) Returns the relative permalink of the page generated by the current output format. diff --git a/content/en/methods/page/_common/scratch-methods.md b/content/en/methods/page/_common/scratch-methods.md new file mode 100644 index 00000000000..c09b4aadc4b --- /dev/null +++ b/content/en/methods/page/_common/scratch-methods.md @@ -0,0 +1,79 @@ +--- +# Do not remove front matter. +--- + +## Methods + +Set +: Sets the value of a given key. + +```go-html-template +{{ .Scratch.Set "greeting" "Hello" }} +``` + +Get +: Gets the value of a given key. + +```go-html-template +{{ .Scratch.Set "greeting" "Hello" }} +{{ .Scratch.Get "greeting" }} → Hello +``` + +Add +: Adds a given value to existing value(s) of the given key. + +: For single values, `Add` accepts values that support Go's `+` operator. If the first `Add` for a key is an array or slice, the following adds will be appended to that list. + +```go-html-template +{{ .Scratch.Set "greeting" "Hello" }} +{{ .Scratch.Add "greeting" "Welcome" }} +{{ .Scratch.Get "greeting" }} → HelloWelcome +``` + +```go-html-template +{{ .Scratch.Set "total" 3 }} +{{ .Scratch.Add "total" 7 }} +{{ .Scratch.Get "total" }} → 10 +``` + +```go-html-template +{{ .Scratch.Set "greetings" (slice "Hello") }} +{{ .Scratch.Add "greetings" (slice "Welcome" "Cheers") }} +{{ .Scratch.Get "greetings" }} → [Hello Welcome Cheers] +``` + +SetInMap +: Takes a `key`, `mapKey` and `value` and adds a map of `mapKey` and `value` to the given `key`. + +```go-html-template +{{ .Scratch.SetInMap "greetings" "english" "Hello" }} +{{ .Scratch.SetInMap "greetings" "french" "Bonjour" }} +{{ .Scratch.Get "greetings" }} → map[english:Hello french:Bonjour] +``` + +DeleteInMap +: Takes a `key` and `mapKey` and removes the map of `mapKey` from the given `key`. + +```go-html-template +{{ .Scratch.SetInMap "greetings" "english" "Hello" }} +{{ .Scratch.SetInMap "greetings" "french" "Bonjour" }} +{{ .Scratch.DeleteInMap "greetings" "english" }} +{{ .Scratch.Get "greetings" }} → map[french:Bonjour] +``` + +GetSortedMapValues +: Returns an array of values from `key` sorted by `mapKey`. + +```go-html-template +{{ .Scratch.SetInMap "greetings" "english" "Hello" }} +{{ .Scratch.SetInMap "greetings" "french" "Bonjour" }} +{{ .Scratch.GetSortedMapValues "greetings" }} → [Hello Bonjour] +``` + +Delete +: Removes the given key. + +```go-html-template +{{ .Scratch.Set "greeting" "Hello" }} +{{ .Scratch.Delete "greeting" }} +``` diff --git a/content/en/methods/page/_index.md b/content/en/methods/page/_index.md new file mode 100644 index 00000000000..278541c5ab8 --- /dev/null +++ b/content/en/methods/page/_index.md @@ -0,0 +1,12 @@ +--- +title: Page methods +linkTitle: Page +description: Use these methods with a Page object. +categories: [] +keywords: [] +menu: + docs: + parent: methods +--- + +Use these methods with a Page object. diff --git a/content/en/methods/pages/ByDate.md b/content/en/methods/pages/ByDate.md new file mode 100644 index 00000000000..41d9b7da7b8 --- /dev/null +++ b/content/en/methods/pages/ByDate.md @@ -0,0 +1,31 @@ +--- +title: ByDate +description: Returns the given page collection sorted by date in ascending order. +categories: [] +keywords: [] +action: + related: + - methods/pages/ByExpiryDate + - methods/pages/ByLastMod + - methods/pages/ByPublishDate + returnType: page.Pages + signatures: [PAGES.ByDate] +--- + +When sorting by date, the value is determined by your [site configuration], defaulting to the `date` field in front matter. + +[site configuration]: /getting-started/configuration/#configure-dates + +```go-html-template +{{ range .Pages.ByDate }} +

{{ .Title }}

+{{ end }} +``` + +To sort in descending order: + +```go-html-template +{{ range .Pages.ByDate.Reverse }} +

{{ .Title }}

+{{ end }} +``` diff --git a/content/en/methods/pages/ByExpiryDate.md b/content/en/methods/pages/ByExpiryDate.md new file mode 100644 index 00000000000..cec8f975c4c --- /dev/null +++ b/content/en/methods/pages/ByExpiryDate.md @@ -0,0 +1,31 @@ +--- +title: ByExpiryDate +description: Returns the given page collection sorted by expiration date in ascending order. +categories: [] +keywords: [] +action: + related: + - methods/pages/ByDate + - methods/pages/ByLastMod + - methods/pages/ByPublishDate + returnType: page.Pages + signatures: [PAGES.ByExpiryDate] +--- + +When sorting by expiration date, the value is determined by your [site configuration], defaulting to the `expiryDate` field in front matter. + +[site configuration]: /getting-started/configuration/#configure-dates + +```go-html-template +{{ range .Pages.ByExpiryDate }} +

{{ .Title }}

+{{ end }} +``` + +To sort in descending order: + +```go-html-template +{{ range .Pages.ByExpiryDate.Reverse }} +

{{ .Title }}

+{{ end }} +``` diff --git a/content/en/methods/pages/ByLanguage.md b/content/en/methods/pages/ByLanguage.md new file mode 100644 index 00000000000..744a8efebce --- /dev/null +++ b/content/en/methods/pages/ByLanguage.md @@ -0,0 +1,24 @@ +--- +title: ByLanguage +description: Returns the given page collection sorted by language in ascending order. +categories: [] +keywords: [] +action: + related: [] + returnType: page.Pages + signatures: [PAGES.ByLanguage] +--- + +```go-html-template +{{ range .Site.AllPages.ByLanguage }} +

{{ .Title }}

+{{ end }} +``` + +To sort in descending order: + +```go-html-template +{{ range .Site.AllPages.ByLanguage.Reverse }} +

{{ .Title }}

+{{ end }} +``` diff --git a/content/en/methods/pages/ByLastmod.md b/content/en/methods/pages/ByLastmod.md new file mode 100644 index 00000000000..3c5a0347f52 --- /dev/null +++ b/content/en/methods/pages/ByLastmod.md @@ -0,0 +1,31 @@ +--- +title: ByLastmod +description: Returns the given page collection sorted by last modification date in ascending order. +categories: [] +keywords: [] +action: + related: + - methods/pages/ByDate + - methods/pages/ByExpiryDate + - methods/pages/ByPublishDate + returnType: page.Pages + signatures: [PAGES.ByLastmod] +--- + +When sorting by last modification date, the value is determined by your [site configuration], defaulting to the `lastmod` field in front matter. + +[site configuration]: /getting-started/configuration/#configure-dates + +```go-html-template +{{ range .Pages.ByLastmod }} +

{{ .Title }}

+{{ end }} +``` + +To sort in descending order: + +```go-html-template +{{ range .Pages.ByLastmod.Reverse }} +

{{ .Title }}

+{{ end }} +``` diff --git a/content/en/methods/pages/ByLength.md b/content/en/methods/pages/ByLength.md new file mode 100644 index 00000000000..7d20d034e43 --- /dev/null +++ b/content/en/methods/pages/ByLength.md @@ -0,0 +1,24 @@ +--- +title: ByLength +description: Returns the given page collection sorted by content length in ascending order. +categories: [] +keywords: [] +action: + related: [] + returnType: page.Pages + signatures: [PAGES.ByLength] +--- + +```go-html-template +{{ range .Pages.ByLength }} +

{{ .Title }}

+{{ end }} +``` + +To sort in descending order: + +```go-html-template +{{ range .Pages.ByLength.Reverse }} +

{{ .Title }}

+{{ end }} +``` diff --git a/content/en/methods/pages/ByLinkTitle.md b/content/en/methods/pages/ByLinkTitle.md new file mode 100644 index 00000000000..073b1e1ba33 --- /dev/null +++ b/content/en/methods/pages/ByLinkTitle.md @@ -0,0 +1,26 @@ +--- +title: ByLinkTitle +description: Returns the given page collection sorted by link title in ascending order, falling back to title if link title is not defined. +categories: [] +keywords: [] +action: + related: + - methods/pages/ByTitle + - methods/pages/ByParam + returnType: page.Pages + signatures: [PAGES.ByLinkTitle] +--- + +```go-html-template +{{ range .Pages.ByLinkTitle }} +

{{ .LinkTitle }}

+{{ end }} +``` + +To sort in descending order: + +```go-html-template +{{ range .Pages.ByLinkTitle.Reverse }} +

{{ .LinkTitle }}

+{{ end }} +``` diff --git a/content/en/methods/pages/ByParam.md b/content/en/methods/pages/ByParam.md new file mode 100644 index 00000000000..0fb9b48c099 --- /dev/null +++ b/content/en/methods/pages/ByParam.md @@ -0,0 +1,36 @@ +--- +title: ByParam +description: Returns the given page collection sorted by the given parameter in ascending order. +categories: [] +keywords: [] +action: + related: + - methods/pages/ByTitle + - methods/pages/ByLinkTitle + returnType: page.Pages + signatures: [PAGES.ByParam PARAM] +--- + +If the given parameter is not present in front matter, Hugo will use the matching parameter in your site configuration if present. + +```go-html-template +{{ range .Pages.ByParam "author" }} +

{{ .Title }}

+{{ end }} +``` + +To sort in descending order: + +```go-html-template +{{ range (.Pages.ByParam "author").Reverse }} +

{{ .Title }}

+{{ end }} +``` + +If the targeted parameter is nested, access the field using dot notation: + +```go-html-template +{{ range .Pages.ByParam "author.last_name" }} +

{{ .Title }}

+{{ end }} +``` diff --git a/content/en/methods/pages/ByPublishDate.md b/content/en/methods/pages/ByPublishDate.md new file mode 100644 index 00000000000..b7d026f5fce --- /dev/null +++ b/content/en/methods/pages/ByPublishDate.md @@ -0,0 +1,31 @@ +--- +title: ByPublishDate +description: Returns the given page collection sorted by publish date in ascending order. +categories: [] +keywords: [] +action: + related: + - methods/pages/ByDate + - methods/pages/ByExpiryDate + - methods/pages/ByLastMod + returnType: page.Pages + signatures: [PAGES.ByPublishDate] +--- + +When sorting by publish date, the value is determined by your [site configuration], defaulting to the `publishDate` field in front matter. + +[site configuration]: /getting-started/configuration/#configure-dates + +```go-html-template +{{ range .Pages.ByPublishDate }} +

{{ .Title }}

+{{ end }} +``` + +To sort in descending order: + +```go-html-template +{{ range .Pages.ByPublishDate.Reverse }} +

{{ .Title }}

+{{ end }} +``` diff --git a/content/en/methods/pages/ByTitle.md b/content/en/methods/pages/ByTitle.md new file mode 100644 index 00000000000..ee54d5cdaec --- /dev/null +++ b/content/en/methods/pages/ByTitle.md @@ -0,0 +1,26 @@ +--- +title: ByTitle +description: Returns the given page collection sorted by title in ascending order. +categories: [] +keywords: [] +action: + related: + - methods/pages/ByLinkTitle + - methods/pages/ByParam + returnType: page.Pages + signatures: [PAGES.ByTitle] +--- + +```go-html-template +{{ range .Pages.ByTitle }} +

{{ .Title }}

+{{ end }} +``` + +To sort in descending order: + +```go-html-template +{{ range .Pages.ByTitle.Reverse }} +

{{ .Title }}

+{{ end }} +``` diff --git a/content/en/methods/pages/ByWeight.md b/content/en/methods/pages/ByWeight.md new file mode 100644 index 00000000000..1e4de8e8047 --- /dev/null +++ b/content/en/methods/pages/ByWeight.md @@ -0,0 +1,28 @@ +--- +title: ByWeight +description: Returns the given page collection sorted by weight in ascending order. +categories: [] +keywords: [] +action: + related: [] + returnType: page.Pages + signatures: [PAGES.ByWeight] +--- + +Assign a [weight] to a page using the `weight` field in front matter. The weight must be a non-zero integer. Lighter items float to the top, while heavier items sink to the bottom. Unweighted or zero-weighted pages are placed at the end of the collection. + +[weight]: /getting-started/glossary/#weight + +```go-html-template +{{ range .Pages.ByWeight }} +

{{ .Title }}

+{{ end }} +``` + +To sort in descending order: + +```go-html-template +{{ range .Pages.ByWeight.Reverse }} +

{{ .Title }}

+{{ end }} +``` diff --git a/content/en/methods/pages/GroupBy.md b/content/en/methods/pages/GroupBy.md new file mode 100644 index 00000000000..e4e905ade27 --- /dev/null +++ b/content/en/methods/pages/GroupBy.md @@ -0,0 +1,36 @@ +--- +title: GroupBy +description: Returns the given page collection grouped by the given field in ascending order. +categories: [] +keywords: [] +action: + related: [] + returnType: page.PagesGroup + signatures: ['PAGES.GroupBy FIELD [SORT]'] +--- + +{{% include "methods/pages/_common/group-sort-order.md" %}} + +```go-html-template +{{ range .Pages.GroupBy "Section" }} +

{{ .Key }}

+ +{{ end }} +``` + +To sort the groups in descending order: + +```go-html-template +{{ range .Pages.GroupBy "Section" "desc" }} +

{{ .Key }}

+ +{{ end }} +``` diff --git a/content/en/methods/pages/GroupByDate.md b/content/en/methods/pages/GroupByDate.md new file mode 100644 index 00000000000..aa77f5881de --- /dev/null +++ b/content/en/methods/pages/GroupByDate.md @@ -0,0 +1,68 @@ +--- +title: GroupByDate +description: Returns the given page collection grouped by date in descending order. +categories: [] +keywords: [] +action: + related: + - methods/pages/GroupByExpiryDate + - methods/pages/GroupByLastMod + - methods/pages/GroupByParamDate + - methods/pages/GroupByPublishDate + returnType: page.PagesGroup + signatures: ['PAGES.GroupByDate LAYOUT [SORT]'] +--- + +When grouping by date, the value is determined by your [site configuration], defaulting to the `date` field in front matter. + +The [layout string] has the same format as the layout string for the [`time.Format`] function. The resulting group key is [localized] for language and region. + +[`time.Format`]: /functions/time/format/ +[layout string]: #layout-string +[localized]: /getting-started/glossary/#localization +[site configuration]: /getting-started/configuration/#configure-dates + +{{% include "methods/pages/_common/group-sort-order.md" %}} + +To group content by year and month: + +```go-html-template +{{ range .Pages.GroupByDate "January 2006" }} +

{{ .Key }}

+ +{{ end }} +``` + +To sort the groups in ascending order: + +```go-html-template +{{ range .Pages.GroupByDate "January 2006" "asc" }} +

{{ .Key }}

+ +{{ end }} +``` + +The pages within each group will also be sorted by date, either ascending or descending depending on the grouping option. To sort the pages within each group, use one of the sorting methods. For example, to sort the pages within each group by title: + +```go-html-template +{{ range .Pages.GroupByDate "January 2006" }} +

{{ .Key }}

+ +{{ end }} +``` + +## Layout string + +{{% include "functions/_common/time-layout-string.md" %}} diff --git a/content/en/methods/pages/GroupByExpiryDate.md b/content/en/methods/pages/GroupByExpiryDate.md new file mode 100644 index 00000000000..d398be77545 --- /dev/null +++ b/content/en/methods/pages/GroupByExpiryDate.md @@ -0,0 +1,68 @@ +--- +title: GroupByExpiryDate +description: Returns the given page collection grouped by expiration date in descending order. +categories: [] +keywords: [] +action: + related: + - methods/pages/GroupByDate + - methods/pages/GroupByLastMod + - methods/pages/GroupByParamDate + - methods/pages/GroupByPublishDate + returnType: page.PagesGroup + signatures: ['PAGES.GroupByExpiryDate LAYOUT [SORT]'] +--- + +When grouping by expiration date, the value is determined by your [site configuration], defaulting to the `expiryDate` field in front matter. + +The [layout string] has the same format as the layout string for the [`time.Format`] function. The resulting group key is [localized] for language and region. + +[`time.Format`]: /functions/time/format/ +[layout string]: #layout-string +[localized]: /getting-started/glossary/#localization +[site configuration]: /getting-started/configuration/#configure-dates + +{{% include "methods/pages/_common/group-sort-order.md" %}} + +To group content by year and month: + +```go-html-template +{{ range .Pages.GroupByExpiryDate "January 2006" }} +

{{ .Key }}

+ +{{ end }} +``` + +To sort the groups in ascending order: + +```go-html-template +{{ range .Pages.GroupByExpiryDate "January 2006" "asc" }} +

{{ .Key }}

+ +{{ end }} +``` + +The pages within each group will also be sorted by expiration date, either ascending or descending depending on your grouping option. To sort the pages within each group, use one of the sorting methods. For example, to sort the pages within each group by title: + +```go-html-template +{{ range .Pages.GroupByExpiryDate "January 2006" }} +

{{ .Key }}

+ +{{ end }} +``` + +## Layout string + +{{% include "functions/_common/time-layout-string.md" %}} diff --git a/content/en/methods/pages/GroupByLastmod.md b/content/en/methods/pages/GroupByLastmod.md new file mode 100644 index 00000000000..30f1431f7af --- /dev/null +++ b/content/en/methods/pages/GroupByLastmod.md @@ -0,0 +1,68 @@ +--- +title: GroupByLastmod +description: Returns the given page collection grouped by last modification date in descending order. +categories: [] +keywords: [] +action: + related: + - methods/pages/GroupByDate + - methods/pages/GroupByExpiryDate + - methods/pages/GroupByParamDate + - methods/pages/GroupByPublishDate + returnType: page.PagesGroup + signatures: ['PAGES.GroupByLastmod LAYOUT [SORT]'] +--- + +When grouping by last modification date, the value is determined by your [site configuration], defaulting to the `lastmod` field in front matter. + +The [layout string] has the same format as the layout string for the [`time.Format`] function. The resulting group key is [localized] for language and region. + +[`time.Format`]: /functions/time/format/ +[layout string]: #layout-string +[localized]: /getting-started/glossary/#localization +[site configuration]: /getting-started/configuration/#configure-dates + +{{% include "methods/pages/_common/group-sort-order.md" %}} + +To group content by year and month: + +```go-html-template +{{ range .Pages.GroupByLastmod "January 2006" }} +

{{ .Key }}

+ +{{ end }} +``` + +To sort the groups in ascending order: + +```go-html-template +{{ range .Pages.GroupByLastmod "January 2006" "asc" }} +

{{ .Key }}

+ +{{ end }} +``` + +The pages within each group will also be sorted by last modification date, either ascending or descending depending on your grouping option. To sort the pages within each group, use one of the sorting methods. For example, to sort the pages within each group by title: + +```go-html-template +{{ range .Pages.GroupByLastmod "January 2006" }} +

{{ .Key }}

+ +{{ end }} +``` + +## Layout string + +{{% include "functions/_common/time-layout-string.md" %}} diff --git a/content/en/methods/pages/GroupByParam.md b/content/en/methods/pages/GroupByParam.md new file mode 100644 index 00000000000..87570c0a5f0 --- /dev/null +++ b/content/en/methods/pages/GroupByParam.md @@ -0,0 +1,36 @@ +--- +title: GroupByParam +description: Returns the given page collection grouped by the given parameter in ascending order. +categories: [] +keywords: [] +action: + related: [] + returnType: page.PagesGroup + signatures: ['PAGES.GroupByParam PARAM [SORT]'] +--- + +{{% include "methods/pages/_common/group-sort-order.md" %}} + +```go-html-template +{{ range .Pages.GroupByParam "color" }} +

{{ .Key | title }}

+ +{{ end }} +``` + +To sort the groups in descending order: + +```go-html-template +{{ range .Pages.GroupByParam "color" "desc" }} +

{{ .Key | title }}

+ +{{ end }} +``` diff --git a/content/en/methods/pages/GroupByParamDate.md b/content/en/methods/pages/GroupByParamDate.md new file mode 100644 index 00000000000..03642fcbeec --- /dev/null +++ b/content/en/methods/pages/GroupByParamDate.md @@ -0,0 +1,65 @@ +--- +title: GroupByParamDate +description: Returns the given page collection grouped by the given date parameter in descending order. +categories: [] +keywords: [] +action: + related: + - methods/pages/GroupByDate + - methods/pages/GroupByExpiryDate + - methods/pages/GroupByLastMod + - methods/pages/GroupByPublishDate + returnType: page.PagesGroup + signatures: ['PAGES.GroupByParamDate PARAM LAYOUT [SORT]'] +--- + +The [layout string] has the same format as the layout string for the [`time.Format`] function. The resulting group key is [localized] for language and region. + +[`time.Format`]: /functions/time/format/ +[layout string]: #layout-string +[localized]: /getting-started/glossary/#localization + +{{% include "methods/pages/_common/group-sort-order.md" %}} + +To group content by year and month: + +```go-html-template +{{ range .Pages.GroupByParamDate "eventDate" "January 2006" }} +

{{ .Key }}

+ +{{ end }} +``` + +To sort the groups in ascending order: + +```go-html-template +{{ range .Pages.GroupByParamDate "eventDate" "January 2006" "asc" }} +

{{ .Key }}

+ +{{ end }} +``` + +The pages within each group will also be sorted by the parameter date, either ascending or descending depending on your grouping option. To sort the pages within each group, use one of the sorting methods. For example, to sort the pages within each group by title: + +```go-html-template +{{ range .Pages.GroupByParamDate "eventDate" "January 2006" }} +

{{ .Key }}

+ +{{ end }} +``` + +## Layout string + +{{% include "functions/_common/time-layout-string.md" %}} diff --git a/content/en/methods/pages/GroupByPublishDate.md b/content/en/methods/pages/GroupByPublishDate.md new file mode 100644 index 00000000000..9264ecae2c4 --- /dev/null +++ b/content/en/methods/pages/GroupByPublishDate.md @@ -0,0 +1,68 @@ +--- +title: GroupByPublishDate +description: Returns the given page collection grouped by publish date in descending order. +categories: [] +keywords: [] +action: + related: + - methods/pages/GroupByDate + - methods/pages/GroupByExpiryDate + - methods/pages/GroupByLastMod + - methods/pages/GroupByParamDate + returnType: page.PagesGroup + signatures: ['PAGES.GroupByPublishDate LAYOUT [SORT]'] +--- + +When grouping by publish date, the value is determined by your [site configuration], defaulting to the `publishDate` field in front matter. + +The [layout string] has the same format as the layout string for the [`time.Format`] function. The resulting group key is [localized] for language and region. + +[`time.Format`]: /functions/time/format/ +[layout string]: #layout-string +[localized]: /getting-started/glossary/#localization +[site configuration]: /getting-started/configuration/#configure-dates + +{{% include "methods/pages/_common/group-sort-order.md" %}} + +To group content by year and month: + +```go-html-template +{{ range .Pages.GroupByPublishDate "January 2006" }} +

{{ .Key }}

+ +{{ end }} +``` + +To sort the groups in ascending order: + +```go-html-template +{{ range .Pages.GroupByPublishDate "January 2006" "asc" }} +

{{ .Key }}

+ +{{ end }} +``` + +The pages within each group will also be sorted by publish date, either ascending or descending depending on your grouping option. To sort the pages within each group, use one of the sorting methods. For example, to sort the pages within each group by title: + +```go-html-template +{{ range .Pages.GroupByPublishDate "January 2006" }} +

{{ .Key }}

+ +{{ end }} +``` + +## Layout string + +{{% include "functions/_common/time-layout-string.md" %}} diff --git a/content/en/methods/pages/Len.md b/content/en/methods/pages/Len.md new file mode 100644 index 00000000000..0a5989a1a67 --- /dev/null +++ b/content/en/methods/pages/Len.md @@ -0,0 +1,14 @@ +--- +title: Len +description: Returns the number of pages in the given page collection. +categories: [] +keywords: [] +action: + related: [] + returnType: int + signatures: [PAGES.Len] +--- + +```go-html-template +{{ .Pages.Len }} → 42 +``` diff --git a/content/en/methods/pages/Limit.md b/content/en/methods/pages/Limit.md new file mode 100644 index 00000000000..0939add9f1c --- /dev/null +++ b/content/en/methods/pages/Limit.md @@ -0,0 +1,16 @@ +--- +title: Limit +description: Returns the first N pages from the given page collection. +categories: [] +keywords: [] +action: + related: [] + returnType: page.Pages + signatures: [PAGES.Limit NUMBER] +--- + +```go-html-template +{{ range .Pages.Limit 3 }} +

{{ .Title }}

+{{ end }} +``` diff --git a/content/en/methods/pages/Next.md b/content/en/methods/pages/Next.md new file mode 100644 index 00000000000..638822e5de2 --- /dev/null +++ b/content/en/methods/pages/Next.md @@ -0,0 +1,55 @@ +--- +title: Next +description: Returns the next page in a local page collection, relative to the given page. +categories: [] +keywords: [] +action: + related: + - methods/pages/Prev + - methods/page/Next + - methods/page/NextInSection + - methods/page/Prev + - methods/page/PrevInSection + returnType: hugolib.pageState + signatures: [PAGES.Next PAGE] +toc: true +--- + +The behavior of the `Prev` and `Next` methods on a `Pages` objects is probably the reverse of what you expect. + +With this content structure and the page collection sorted by weight in ascending order: + +```text +content/ +├── pages/ +│ ├── _index.md +│ ├── page-1.md <-- front matter: weight = 10 +│ ├── page-2.md <-- front matter: weight = 20 +│ └── page-3.md <-- front matter: weight = 30 +└── _index.md +``` + +When you visit page-2: + +- The `Prev` method points to page-3 +- The `Next` method points to page-1 + +{{% note %}} +Use the opposite label in your navigation links as shown in the example below. +{{% /note %}} + +```go-html-template +{{ $pages := where .Site.RegularPages.ByWeight "Section" "pages" }} + +{{ with $pages.Next . }} + Previous +{{ end }} + +{{ with $pages.Prev . }} + Next +{{ end }} +``` + +## Compare to Page methods + +{{% include "methods/_common/next-prev-on-page-vs-next-prev-on-pages.md" %}} diff --git a/content/en/methods/pages/Prev.md b/content/en/methods/pages/Prev.md new file mode 100644 index 00000000000..3a4dcfd1d68 --- /dev/null +++ b/content/en/methods/pages/Prev.md @@ -0,0 +1,55 @@ +--- +title: Prev +description: Returns the previous page in a local page collection, relative to the given page. +categories: [] +keywords: [] +action: + related: + - methods/pages/Next + - methods/page/Next + - methods/page/NextInSection + - methods/page/Prev + - methods/page/PrevInSection + returnType: hugolib.pageStates + signatures: [PAGES.Prev PAGE] +toc: true +--- + +The behavior of the `Prev` and `Next` methods on a `Pages` objects is probably the reverse of what you expect. + +With this content structure and the page collection sorted by weight in ascending order: + +```text +content/ +├── pages/ +│ ├── _index.md +│ ├── page-1.md <-- front matter: weight = 10 +│ ├── page-2.md <-- front matter: weight = 20 +│ └── page-3.md <-- front matter: weight = 30 +└── _index.md +``` + +When you visit page-2: + +- The `Prev` method points to page-3 +- The `Next` method points to page-1 + +{{% note %}} +Use the opposite label in your navigation links as shown in the example below. +{{% /note %}} + +```go-html-template +{{ $pages := where .Site.RegularPages.ByWeight "Section" "pages" }} + +{{ with $pages.Next . }} + Previous +{{ end }} + +{{ with $pages.Prev . }} + Next +{{ end }} +``` + +## Compare to Page methods + +{{% include "methods/_common/next-prev-on-page-vs-next-prev-on-pages.md" %}} diff --git a/content/en/methods/pages/Related.md b/content/en/methods/pages/Related.md new file mode 100644 index 00000000000..60ca4ca6743 --- /dev/null +++ b/content/en/methods/pages/Related.md @@ -0,0 +1,77 @@ +--- +title: Related +description: Returns a collection of pages related to the given page. +categories: [] +keywords: [] +action: + related: + - methods/page/HeadingsFiltered + - functions/collections/KeyVals + returnType: page.Pages + signatures: [PAGES.Related PAGE/OPTIONS] +--- + +Based on front matter, Hugo uses several factors to identify content related to the given page. Use the default [related content configuration], or tune the results to the desired indices and parameters. See [details]. + +The argument passed to the `Related` method may be a `Page` or an options map. For example, to pass the current page: + +{{< code file="layouts/_default/single.html" lang=go-html-template copy=false >}} +{{ with .Site.RegularPages.Related . | first 5 }} +

Related pages:

+ +{{ end }} +{{< /code >}} + +To pass an options map: + +{{< code file="layouts/_default/single.html" lang=go-html-template copy=false >}} +{{ $opts := dict + "document" . + "indices" (slice "tags" "keywords") +}} +{{ with .Site.RegularPages.Related $opts | first 5 }} +

Related pages:

+ +{{ end }} +{{< /code >}} + + +## Options + +indices +: (`slice`) The indices to search within. + +document +: (`page`) The page for which to find related content. Required when specifying an options map. + +namedSlices +: (`slice`) The keywords to search for, expressed as a slice of `KeyValues` using the [`keyVals`] function. + +[`keyVals`]: /functions/collections/keyvals/ + +fragments +: (`slice`) A list of special keywords that is used for indices configured as type "fragments". This will match the [fragment] identifiers of the documents. + +A contrived example using all of the above: + +```go-html-template +{{ $page := . }} +{{ $opts := dict + "indices" (slice "tags" "keywords") + "document" $page + "namedSlices" (slice (keyVals "tags" "hugo" "rocks") (keyVals "date" $page.Date)) + "fragments" (slice "heading-1" "heading-2") +}} +``` + +[details]: /content-management/related/ +[fragment]: /getting-started/glossary/#fragment +[related content configuration]: /content-management/related/ diff --git a/content/en/methods/pages/Reverse.md b/content/en/methods/pages/Reverse.md new file mode 100644 index 00000000000..b54cf451e64 --- /dev/null +++ b/content/en/methods/pages/Reverse.md @@ -0,0 +1,16 @@ +--- +title: Reverse +description: Returns the given page collection in reverse order. +categories: [] +keywords: [] +action: + related: [] + returnType: page.Pages + signatures: [PAGES.Reverse] +--- + +```go-html-template +{{ range .Pages.ByDate.Reverse }} +

{{ .Title }}

+{{ end }} +``` diff --git a/content/en/methods/pages/_common/_index.md b/content/en/methods/pages/_common/_index.md new file mode 100644 index 00000000000..47d5812fba5 --- /dev/null +++ b/content/en/methods/pages/_common/_index.md @@ -0,0 +1,13 @@ +--- +cascade: + _build: + list: never + publishResources: false + render: never +--- + + diff --git a/content/en/methods/pages/_common/group-sort-order.md b/content/en/methods/pages/_common/group-sort-order.md new file mode 100644 index 00000000000..bb5be82f6c3 --- /dev/null +++ b/content/en/methods/pages/_common/group-sort-order.md @@ -0,0 +1,5 @@ +--- +# Do not remove front matter. +--- + +For the optional sort order, specify either `asc` for ascending order, or `desc` for descending order. diff --git a/content/en/methods/pages/_index.md b/content/en/methods/pages/_index.md new file mode 100644 index 00000000000..d8a64f5d5f8 --- /dev/null +++ b/content/en/methods/pages/_index.md @@ -0,0 +1,13 @@ +--- +title: Pages methods +linkTitle: Pages +description: Use these methods with a collection of Page objects. +categories: [] +keywords: [] +menu: + docs: + parent: methods +aliases: [/variables/pages] +--- + +Use these methods with a collection of Page objects. diff --git a/content/en/methods/resource/Colors.md b/content/en/methods/resource/Colors.md new file mode 100644 index 00000000000..9eca058fe7e --- /dev/null +++ b/content/en/methods/resource/Colors.md @@ -0,0 +1,20 @@ +--- +title: Colors +description: Applicable to images, returns a slice of the most dominant colors using a simple histogram method. +categories: [] +keywords: [] +action: + related: [] + returnType: '[]string' + signatures: [RESOURCE.Colors] +--- + +```go-html-template +{{ with resources.Get "images/a.jpg" }} + {{ .Colors }} → [#bebebd #514947 #768a9a #647789 #90725e #a48974] +{{ end }} +``` + +This method is fast, but if you also scale down your images, it would be good for performance to extract the colors from the scaled image. + +{{% include "methods/resource/_common/global-page-remote-resources.md" %}} diff --git a/content/en/methods/resource/Content.md b/content/en/methods/resource/Content.md new file mode 100644 index 00000000000..f51abd32730 --- /dev/null +++ b/content/en/methods/resource/Content.md @@ -0,0 +1,61 @@ +--- +title: Content +description: Returns the content of the given resource. +categories: [] +keywords: [] +action: + related: [] + returnType: any + signatures: [RESOURCE.Content] +toc: +--- + +The `Content` method on a `Resource` object returns `template.HTML` when the resource type is `page`, otherwise it returns a `string`. + +[resource type]: /methods/resource/resourcetype + +{{< code file="assets/quotations/kipling.txt" >}} +He travels the fastest who travels alone. +{{< /code >}} + +To get the content: + +```go-html-template +{{ with resources.Get "quotations/kipling.txt" }} + {{ .Content }} → He travels the fastest who travels alone. +{{ end }} +``` + +To get the size in bytes: + +```go-html-template +{{ with resources.Get "quotations/kipling.txt" }} + {{ .Content | len }} → 42 +{{ end }} +``` + +To create an inline image: + +```go-html-template +{{ with resources.Get "images/a.jpg" }} + +{{ end }} +``` + +To create inline CSS: + +```go-html-template +{{ with resources.Get "css/style.css" }} + +{{ end }} +``` + +To create inline JavaScript: + +```go-html-template +{{ with resources.Get "js/script.js" }} + +{{ end }} +``` + +{{% include "methods/resource/_common/global-page-remote-resources.md" %}} diff --git a/content/en/methods/resource/Crop.md b/content/en/methods/resource/Crop.md new file mode 100644 index 00000000000..711fc07b0d8 --- /dev/null +++ b/content/en/methods/resource/Crop.md @@ -0,0 +1,48 @@ +--- +title: Crop +description: Applicable to images, returns an image resource cropped to the given dimensions without resizing. +categories: [] +keywords: [] +action: + related: + - methods/resource/Fit + - methods/resource/Fill + - methods/resource/Resize + - methods/resource/Process + - functions/images/Process + returnType: images.ImageResource + signatures: [RESOURCE.Crop SPEC] +toc: true +--- + +Crop an image to match the given dimensions without resizing. You must provide both width and height. + +```go-html-template +{{ with resources.Get "images/original.jpg" }} + {{ with .Crop "200x200" }} + + {{ end }} +{{ end }} +``` + +{{% include "methods/resource/_common/global-page-remote-resources.md" %}} + +{{% include "/methods/resource/_common/processing-spec.md" %}} + +## Example + +```go-html-template +{{ with resources.Get "images/original.jpg" }} + {{ with .Crop "200x200 topright webp q85 lanczos" }} + + {{ end }} +{{ end }} +``` + +{{< img + src="images/examples/zion-national-park.jpg" + alt="Zion National Park" + filter="Process" + filterArgs="crop 200x200 topright webp q85 lanczos" + example=true +>}} diff --git a/content/en/methods/resource/Data.md b/content/en/methods/resource/Data.md new file mode 100644 index 00000000000..0fbaf619906 --- /dev/null +++ b/content/en/methods/resource/Data.md @@ -0,0 +1,53 @@ +--- +title: Data +description: Applicable to resources returned by the resources.GetRemote function, returns information from the HTTP response. +categories: [] +keywords: [] +action: + related: + - functions/resources/GetRemote + - methods/resource/Err + returnType: map + signatures: [RESOURCE.Data] +--- + +The `Data` method on a resource returned by the [`resources.GetRemote`] function returns information from the HTTP response. + +[`resources.GetRemote`]: functions/resources/getremote + +```go-html-template +{{ $url := "https://example.org/images/a.jpg" }} +{{ with resources.GetRemote $url }} + {{ with .Err }} + {{ errorf "%s" . }} + {{ else }} + {{ with .Data }} + {{ .ContentLength }} → 42764 + {{ .ContentType }} → image/jpeg + {{ .Status }} → 200 OK + {{ .StatusCode }} → 200 + {{ .TransferEncoding }} → [] + {{ end }} + {{ end }} +{{ else }} + {{ errorf "Unable to get remote resource %q" $url }} +{{ end }} +``` + +ContentLength +: (`int`) The content length in bytes. + +ContentType +: (`string`) The content type. + +Status +: (`string`) The HTTP status text. + +StatusCode +: (`int`) The HTTP status code. + +TransferEncoding +: (`string`) The transfer encoding. + + +[`resources.GetRemote`]: functions/resources/getremote diff --git a/content/en/methods/resource/Err.md b/content/en/methods/resource/Err.md new file mode 100644 index 00000000000..f4b410aa7ab --- /dev/null +++ b/content/en/methods/resource/Err.md @@ -0,0 +1,56 @@ +--- +title: Err +description: Applicable to resources returned by the resources.GetRemote function, returns an error message if the HTTP request fails, else nil. +categories: [] +keywords: [] +action: + related: + - functions/resources/GetRemote + - methods/resource/Data + returnType: resource.resourceError + signatures: [RESOURCE.Err] +--- + +The `Err` method on a resource returned by the [`resources.GetRemote`] function returns an error message if the HTTP request fails, else nil. If you do not handle the error yourself, Hugo will fail the build. + +[`resources.GetRemote`]: functions/resources/getremote + +In this example we send an HTTP request to a nonexistent domain: + +```go-html-template +{{ $url := "https://broken-example.org/images/a.jpg" }} +{{ with resources.GetRemote $url }} + {{ with .Err }} + {{ errorf "%s" . }} + {{ else }} + + {{ end }} +{{ else }} + {{ errorf "Unable to get remote resource %q" $url }} +{{ end }} +``` + +The code above captures the error from the HTTP request, then fails the build: + +```text +ERROR error calling resources.GetRemote: Get "https://broken-example.org/images/a.jpg": dial tcp: lookup broken-example.org on 127.0.0.53:53: no such host +``` + +To log an error as a warning instead of an error: + +```go-html-template +{{ $url := "https://broken-example.org/images/a.jpg" }} +{{ with resources.GetRemote $url }} + {{ with .Err }} + {{ warnf "%s" . }} + {{ else }} + + {{ end }} +{{ else }} + {{ errorf "Unable to get remote resource %q" $url }} +{{ end }} +``` + +{{% note %}} +An HTTP response with a 404 status code is not an HTTP request error. To handle 404 status codes, code defensively using the nested `with-else-end` construct as shown above. +{{% /note %}} diff --git a/content/en/methods/resource/Exif.md b/content/en/methods/resource/Exif.md new file mode 100644 index 00000000000..bf18efd500f --- /dev/null +++ b/content/en/methods/resource/Exif.md @@ -0,0 +1,78 @@ +--- +title: Exif +description: Applicable to images, returns an EXIF object containing image metatdata. +categories: [] +keywords: [] +action: + related: [] + returnType: exif.ExifInfo + signatures: [RESOURCE.Exif] +toc: true +--- + +Applicable to images, the `Exif` method on an image `Resource` object returns an [EXIF] object containing image metatdata. + +## Methods + +Date +: (`time.Time`) Returns the image creation date/time. Format with the [`time.Format`]function. + +Lat +: (`float64`) Returns the GPS latitude in degrees. + +Long +: (`float64`) Returns the GPS longitude in degrees. + +Tags +: (`exif.Tags`) Returns a collection of the available EXIF tags for this image. You may include or exclude specific tags from this collection in the [site configuration]. + +## Examples + +To list the creation date, location, and EXIF tags: + +```go-html-template +{{ with resources.Get "images/a.jpg" }} + {{ with .Exif }} +

Date: {{ .Date }}

+

Lat/Long: {{ .Lat }}/{{ .Long }}

+ {{ with .Tags }} +

Tags

+ + + + + + {{ range $k, $v := . }} + + {{ end }} + +
TagValue
{{ $k }}{{ $v }}
+ {{ end }} + {{ end }} +{{ end }} +``` + +To list specific values: + +```go-html-template +{{ with resources.Get "images/a.jpg" }} + {{ with .Exif }} +
    + {{ with .Date }}
  • Date: {{ .Format "January 02, 2006" }}
  • {{ end }} + {{ with .Tags.ApertureValue }}
  • Aperture: {{ lang.FormatNumber 2 . }}
  • {{ end }} + {{ with .Tags.BrightnessValue }}
  • Brightness: {{ lang.FormatNumber 2 . }}
  • {{ end }} + {{ with .Tags.ExposureTime }}
  • Exposure Time: {{ . }}
  • {{ end }} + {{ with .Tags.FNumber }}
  • F Number: {{ . }}
  • {{ end }} + {{ with .Tags.FocalLength }}
  • Focal Length: {{ . }}
  • {{ end }} + {{ with .Tags.ISOSpeedRatings }}
  • ISO Speed Ratings: {{ . }}
  • {{ end }} + {{ with .Tags.LensModel }}
  • Lens Model: {{ . }}
  • {{ end }} +
+ {{ end }} +{{ end }} +``` + +{{% include "methods/resource/_common/global-page-remote-resources.md" %}} + +[exif]: https://en.wikipedia.org/wiki/Exif +[site configuration]: /content-management/image-processing/#exif-data +[`time.Format`]: /functions/time/format diff --git a/content/en/methods/resource/Fill.md b/content/en/methods/resource/Fill.md new file mode 100644 index 00000000000..8bbaf93eec5 --- /dev/null +++ b/content/en/methods/resource/Fill.md @@ -0,0 +1,48 @@ +--- +title: Fill +description: Applicable to images, returns an image resource cropped and resized to the given dimensions. +categories: [] +keywords: [] +action: + related: + - methods/resource/Crop + - methods/resource/Fit + - methods/resource/Resize + - methods/resource/Process + - functions/images/Process + returnType: images.ImageResource + signatures: [RESOURCE.Fill SPEC] +toc: true +--- + +Crop and resize an image to match the given dimensions. You must provide both width and height. + +```go-html-template +{{ with resources.Get "images/original.jpg" }} + {{ with .Fill "200x200" }} + + {{ end }} +{{ end }} +``` + +{{% include "methods/resource/_common/global-page-remote-resources.md" %}} + +{{% include "/methods/resource/_common/processing-spec.md" %}} + +## Example + +```go-html-template +{{ with resources.Get "images/original.jpg" }} + {{ with .Fill "200x200 top webp q85 lanczos" }} + + {{ end }} +{{ end }} +``` + +{{< img + src="images/examples/zion-national-park.jpg" + alt="Zion National Park" + filter="Process" + filterArgs="fill 200x200 top webp q85 lanczos" + example=true +>}} diff --git a/content/en/methods/resource/Filter.md b/content/en/methods/resource/Filter.md new file mode 100644 index 00000000000..329168da7c9 --- /dev/null +++ b/content/en/methods/resource/Filter.md @@ -0,0 +1,68 @@ +--- +title: Filter +description: Applicable to images, applies one or more image filters to the given image resource. +categories: [] +keywords: [] +action: + related: + - functions/images/Filter + returnType: resources.resourceAdapter + signatures: [RESOURCE.Filter FILTER...] +toc: true +--- + +Apply one or more [image filters](#image-filters) to the given image. + +To apply a single filter: + +```go-html-template +{{ with resources.Get "images/original.jpg" }} + {{ with .Filter images.Grayscale }} + + {{ end }} +{{ end }} +``` + +To apply two or more filters, executing from left to right: + +```go-html-template +{{ $filters := slice + images.Grayscale + (images.GaussianBlur 8) +}} +{{ with resources.Get "images/original.jpg" }} + {{ with .Filter $filters }} + + {{ end }} +{{ end }} +``` + +You can also apply image filters using the [`images.Filter`] function. + +[`images.Filter`]: /functions/images/filter + +{{% include "methods/resource/_common/global-page-remote-resources.md" %}} + +## Example + +```go-html-template +{{ with resources.Get "images/original.jpg" }} + {{ with .Filter images.Grayscale }} + + {{ end }} +{{ end }} +``` + +{{< img + src="images/examples/zion-national-park.jpg" + alt="Zion National Park" + filter="Grayscale" + filterArgs="" + example=true +>}} + +## Image filters + +Use any of these filters with the `Filter` method. + +{{< list-pages-in-section path=/functions/images filter=functions_images_no_filters filterType=exclude >}} diff --git a/content/en/methods/resource/Fit.md b/content/en/methods/resource/Fit.md new file mode 100644 index 00000000000..13354fe5a31 --- /dev/null +++ b/content/en/methods/resource/Fit.md @@ -0,0 +1,48 @@ +--- +title: Fit +description: Applicable to images, returns an image resource downscaled to fit the given dimensions while maintaining aspect ratio. +categories: [] +keywords: [] +action: + related: + - methods/resource/Crop + - methods/resource/Fill + - methods/resource/Resize + - methods/resource/Process + - functions/images/Process + returnType: images.ImageResource + signatures: [RESOURCE.Fit SPEC] +toc: true +--- + +Downscale an image to fit the given dimensions while maintaining aspect ratio. You must provide both width and height. + +```go-html-template +{{ with resources.Get "images/original.jpg" }} + {{ with .Fit "200x200" }} + + {{ end }} +{{ end }} +``` + +{{% include "methods/resource/_common/global-page-remote-resources.md" %}} + +{{% include "/methods/resource/_common/processing-spec.md" %}} + +## Example + +```go-html-template +{{ with resources.Get "images/original.jpg" }} + {{ with .Fit "300x175 webp q85 lanczos" }} + + {{ end }} +{{ end }} +``` + +{{< img + src="images/examples/zion-national-park.jpg" + alt="Zion National Park" + filter="Process" + filterArgs="fit 300x175 webp q85 lanczos" + example=true +>}} diff --git a/content/en/methods/resource/Height.md b/content/en/methods/resource/Height.md new file mode 100644 index 00000000000..dcaf6c514ff --- /dev/null +++ b/content/en/methods/resource/Height.md @@ -0,0 +1,27 @@ +--- +title: Height +description: Applicable to images, returns the height of the given resource. +categories: [] +keywords: [] +action: + related: + - methods/resource/Width + returnType: int + signatures: [RESOURCE.Height] +--- + +```go-html-template +{{ with resources.Get "images/a.jpg" }} + {{ .Height }} → 400 +{{ end }} +``` + +Use the `Width` and `Height` methods together when rendering an `img` element: + +```go-html-template +{{ with resources.Get "images/a.jpg" }} + +{{ end }} +``` + +{{% include "methods/resource/_common/global-page-remote-resources.md" %}} diff --git a/content/en/methods/resource/Key.md b/content/en/methods/resource/Key.md new file mode 100644 index 00000000000..747df414ce3 --- /dev/null +++ b/content/en/methods/resource/Key.md @@ -0,0 +1,45 @@ +--- +title: Key +description: Returns the unique key for the given resource, equivalent to its publishing path. +categories: [] +keywords: [] +action: + related: + - methods/resource/Permalink + - methods/resource/RelPermalink + - methods/resource/Publish + returnType: string + signatures: [RESOURCE.Key] +--- + +By way of example, consider this site configuration: + +{{< code-toggle file=hugo copy=false >}} +baseURL = 'https://example.org/docs/' +{{< /code-toggle >}} + +And this template: + +```go-html-template + {{ with resources.Get "images/a.jpg" }} + {{ with resources.Copy "foo/bar/b.jpg" . }} + {{ .Key }} → foo/bar/b.jpg + + {{ .Name }} → images/a.jpg + {{ .Title }} → images/a.jpg + + {{ .RelPermalink }} → /docs/foo/bar/b.jpg + {{ end }} + {{ end }} +``` + +We used the [`resources.Copy`] function to change the publishing path. The `Key` method returns the updated path, but note that it is different than the value returned by [`RelPermalink`]. The `RelPermalink` value includes the subdirectory segment of the `baseURL` in the site configuration. + +The `Key` method is useful if you need to get the resource's publishing path without publishing the resource. Unlike the `Permalink`, `RelPermalink`, or `Publish` methods, calling `Key` will not publish the resource. + + +{{% include "methods/resource/_common/global-page-remote-resources.md" %}} + +[`Permalink`]: /methods/resource/permalink +[`RelPermalink`]: /methods/resource/relpermalink +[`resources.Copy`]: /functions/resources/copy diff --git a/content/en/methods/resource/MediaType.md b/content/en/methods/resource/MediaType.md new file mode 100644 index 00000000000..6dea8706cd9 --- /dev/null +++ b/content/en/methods/resource/MediaType.md @@ -0,0 +1,52 @@ +--- +title: MediaType +description: Returns a media type object for the given resource. +categories: [] +keywords: [] +action: + related: [] + returnType: media.Type + signatures: [RESOURCE.MediaType] +--- + +The `MediaType` method on a `Resource` object returns an object with additional methods. + +## Methods + +Type +: (`string`) The resource's media type. + +```go-html-template +{{ with resources.Get "images/a.jpg" }} + {{ .MediaType.Type }} → image/jpeg +{{ end }} +``` + +MainType +: (`string`) The main type of the resource’s media type. + +```go-html-template +{{ with resources.Get "images/a.jpg" }} + {{ .MediaType.MainType }} → image +{{ end }} +``` + +SubType +: (`string`) The subtype of the resource’s media type. This may or may not correspond to the file suffix. + +```go-html-template +{{ with resources.Get "images/a.jpg" }} + {{ .MediaType.SubType }} → jpeg +{{ end }} +``` + +Suffixes +: (`slice`) A slice of possible file suffixes for the resource’s media type. + +```go-html-template +{{ with resources.Get "images/a.jpg" }} + {{ .MediaType.Suffixes }} → [jpg jpeg jpe jif jfif] +{{ end }} +``` + +{{% include "methods/resource/_common/global-page-remote-resources.md" %}} diff --git a/content/en/methods/resource/Name.md b/content/en/methods/resource/Name.md new file mode 100644 index 00000000000..1c584adadee --- /dev/null +++ b/content/en/methods/resource/Name.md @@ -0,0 +1,82 @@ +--- +title: Name +description: Returns the name of the given resource as optionally defined in front matter, falling back to a relative path or hashed file name depending on resource type. +categories: [] +keywords: [] +action: + related: + - methods/resource/Title + returnType: string + signatures: [RESOURCE.Name] +toc: true +--- + +The value returned by the `Name` method on a `Resource` object depends on the resource type. + +## Global resource + +With a [global resource], the `Name` method returns the path to the resource, relative to the assets directory. + +```text +assets/ +└── images/ + └── a.jpg +``` + +```go-html-template +{{ with resources.Get "images/a.jpg" }} + {{ .Name }} → images/a.jpg +{{ end }} +``` + +## Page resource + +With a [page resource], the `Name` method returns the path to the resource, relative to the page bundle. + +```text +content/ +├── posts/ +│ ├── post-1/ +│ │ ├── images/ +│ │ │ └── a.jpg +│ │ └── index.md +│ └── _index.md +└── _index.md +``` + +```go-html-template +{{ with .Resources.Get "images/a.jpg" }} + {{ .Name }} → images/a.jpg +{{ end }} +``` + +If you create an element in the `resources` array in front matter, the `Name` method returns the value of the `name` parameter: + +{{< code-toggle file="content/posts/post-1.md" fm=true >}} +title = 'Post 1' +[[resources]] +src = 'images/a.jpg' +name = 'cat' +title = 'Felix the cat' +[resources.params] +temperament = 'malicious' +{{< /code-toggle >}} + +```go-html-template +{{ with .Resources.Get "cat" }} + {{ .Name }} → cat +{{ end }} +``` +## Remote resource + +With a [remote resource], the `Name` method returns a hashed file name. + +```go-html-template +{{ with resources.GetRemote "https://example.org/images/a.jpg" }} + {{ .Name }} → a_18432433023265451104.jpg +{{ end }} +``` + +[global resource]: /getting-started/glossary/#global-resource +[page resource]: /getting-started/glossary/#page-resource +[remote resource]: /getting-started/glossary/#remote-resource diff --git a/content/en/methods/resource/Params.md b/content/en/methods/resource/Params.md new file mode 100644 index 00000000000..7a9ec89b3ab --- /dev/null +++ b/content/en/methods/resource/Params.md @@ -0,0 +1,65 @@ +--- +title: Params +description: Returns a map of resource parameters as defined in front matter. +categories: [] +keywords: [] +action: + related: [] + returnType: map + signatures: [RESOURCE.Params] +--- + +Use the `Params` method with [page resources]. It is not applicable to either [global] or [remote] resources. + +[global]: /getting-started/glossary/#global-resource +[page resources]: /getting-started/glossary/#page-resource +[remote]: /getting-started/glossary/#remote-resource + +With this content structure: + +```text +content/ +├── posts/ +│ ├── cats/ +│ │ ├── images/ +│ │ │ └── a.jpg +│ │ └── index.md +│ └── _index.md +└── _index.md +``` + +And this front matter: + +{{< code-toggle file=content/posts/cats.md fm=true copy=false >}} +title = 'Cats' +[[resources]] + src = 'images/a.jpg' + title = 'Felix the cat' + [resources.params] + alt = 'Photograph of black cat' + temperament = 'vicious' +{{< /code-toggle >}} + +And this template: + +```go-html-template +{{ with .Resources.Get "images/a.jpg" }} +
+ {{ .Params.alt }} +
{{ .Title }} is {{ .Params.temperament }}
+
+{{ end }} +``` + +Hugo renders: + +```html +
+ Photograph of black cat +
Felix the cat is vicious
+
+``` + +See the [page resources] section for more information. + +[page resources]: /content-management/page-resources diff --git a/content/en/methods/resource/Permalink.md b/content/en/methods/resource/Permalink.md new file mode 100644 index 00000000000..ab0ad41b0e4 --- /dev/null +++ b/content/en/methods/resource/Permalink.md @@ -0,0 +1,25 @@ +--- +title: Permalink +description: Publishes the given resource and returns its permalink. +categories: [] +keywords: [] +action: + related: + - methods/resource/RelPermalink + - methods/resource/Publish + - methods/resource/Key + returnType: string + signatures: [RESOURCE.Permalink] +--- + +The `Permalink` method on a `Resource` object writes the resource to the publish directory, typically `public`, and returns its [permalink]. + +[permalink]: /getting-started/glossary/#permalink + +```go-html-template +{{ with resources.Get "images/a.jpg" }} + {{ .Permalink }} → https://example.org/images/a.jpg +{{ end }} +``` + +{{% include "methods/resource/_common/global-page-remote-resources.md" %}} diff --git a/content/en/methods/resource/Process.md b/content/en/methods/resource/Process.md new file mode 100644 index 00000000000..c3f86feeea5 --- /dev/null +++ b/content/en/methods/resource/Process.md @@ -0,0 +1,66 @@ +--- +title: Process +description: Applicable to images, returns an image resource processed with the given specification. +categories: [] +keywords: [] +action: + related: + - methods/resource/Crop + - methods/resource/Fit + - methods/resource/Fill + - methods/resource/Resize + - functions/images/Process + returnType: images.ImageResource + signatures: [RESOURCE.Process SPEC] +toc: true +--- + +Process an image with the given specification. The specification can contain an optional action, one of `resize`, `crop`, `fit` or `fill`. This means that you can use this method instead of [`Crop`], [`Fill`], [`Fit`], or [`Resize`]. + +```go-html-template +{{ with resources.Get "images/original.jpg" }} + {{ with .Process "crop 200x200" }} + + {{ end }} +{{ end }} +``` + +You can also use this method to apply simple transformations such as rotation and conversion: + +```go-html-template +{{/* Rotate 90 degrees counter-clockwise. */}} +{{ $image := $image.Process "r90" }} + +{{/* Convert to WebP. */}} +{{ $image := $image.Process "webp" }} +``` + +The `Process` method is also available as a filter, which is more effective if you need to apply multiple filters to an image. See [`images.Process`]. + +{{% include "methods/resource/_common/global-page-remote-resources.md" %}} + +{{% include "/methods/resource/_common/processing-spec.md" %}} + +## Example + +```go-html-template +{{ with resources.Get "images/original.jpg" }} + {{ with .Process "crop 200x200 topright webp q85 lanczos" }} + + {{ end }} +{{ end }} +``` + +{{< img + src="images/examples/zion-national-park.jpg" + alt="Zion National Park" + filter="Process" + filterArgs="crop 200x200 topright webp q85 lanczos" + example=true +>}} + +[`Crop`]: /methods/resource/crop +[`Fill`]: /methods/resource/fill +[`Fit`]: /methods/resource/fit +[`Resize`]: /methods/resource/resize +[`images.Process`]: /functions/images/process diff --git a/content/en/methods/resource/Publish.md b/content/en/methods/resource/Publish.md new file mode 100644 index 00000000000..b090bfe5ab9 --- /dev/null +++ b/content/en/methods/resource/Publish.md @@ -0,0 +1,35 @@ +--- +title: Publish +description: Publishes the given resource. +categories: [] +keywords: [] +action: + related: + - methods/resource/Permalink + - methods/resource/RelPermalink + - methods/resource/Key + returnType: nil + signatures: [RESOURCE.Publish] +--- + +The `Publish` method on a `Resource` object writes the resource to the publish directory, typically `public`. + +```go-html-template +{{ with resources.Get "images/a.jpg" }} + {{ .Publish }} +{{ end }} +``` + +The `Permalink` and `RelPermalink` methods also publish a resource. `Publish` is a convenience method for publishing without a return value. For example, this: + +```go-html-template +{{ $resource.Publish }} +``` + +Instead of this: + +```go-html-template +{{ $noop := $resource.Permalink }} +``` + +{{% include "methods/resource/_common/global-page-remote-resources.md" %}} diff --git a/content/en/methods/resource/RelPermalink.md b/content/en/methods/resource/RelPermalink.md new file mode 100644 index 00000000000..2b96c35d739 --- /dev/null +++ b/content/en/methods/resource/RelPermalink.md @@ -0,0 +1,25 @@ +--- +title: RelPermalink +description: Publishes the given resource and returns its relative permalink. +categories: [] +keywords: [] +action: + related: + - methods/resource/Permalink + - methods/resource/Publish + - methods/resource/Key + returnType: string + signatures: [RESOURCE.RelPermalink] +--- + +The `Permalink` method on a `Resource` object writes the resource to the publish directory, typically `public`, and returns its [relative permalink]. + +[relative permalink]: /getting-started/glossary/#relative-permalink + +```go-html-template +{{ with resources.Get "images/a.jpg" }} + {{ .RelPermalink }} → /images/a.jpg +{{ end }} +``` + +{{% include "methods/resource/_common/global-page-remote-resources.md" %}} diff --git a/content/en/methods/resource/Resize.md b/content/en/methods/resource/Resize.md new file mode 100644 index 00000000000..4ba054bb502 --- /dev/null +++ b/content/en/methods/resource/Resize.md @@ -0,0 +1,49 @@ +--- +title: Resize +description: Applicable to images, returns an image resource resized to the given width and/or height. +categories: [] +keywords: [] +action: + related: + - methods/resource/Crop + - methods/resource/Fit + - methods/resource/Fill + - methods/resource/Process + - functions/images/Process + returnType: images.ImageResource + signatures: [RESOURCE.Resize SPEC] +--- + +Resize an image to the given width and/or height. + +If you specify both width and height, the resulting image will be disproportionally scaled unless the original image has the same aspect ratio. + +```go-html-template +{{ with resources.Get "images/original.jpg" }} + {{ with .Resize "300x" }} + + {{ end }} +{{ end }} +``` + +{{% include "methods/resource/_common/global-page-remote-resources.md" %}} + +{{% include "/methods/resource/_common/processing-spec.md" %}} + +## Example + +```go-html-template +{{ with resources.Get "images/original.jpg" }} + {{ with .Resize "300x webp q85 lanczos" }} + + {{ end }} +{{ end }} +``` + +{{< img + src="images/examples/zion-national-park.jpg" + alt="Zion National Park" + filter="Process" + filterArgs="resize 300x webp q85 lanczos" + example=true +>}} diff --git a/content/en/methods/resource/ResourceType.md b/content/en/methods/resource/ResourceType.md new file mode 100644 index 00000000000..1522b7d3294 --- /dev/null +++ b/content/en/methods/resource/ResourceType.md @@ -0,0 +1,43 @@ +--- +title: ResourceType +description: Returns the main type of the given resource's media type. +categories: [] +keywords: [] +action: + related: [] + returnType: string + signatures: [RESOURCE.ResourceType] +--- + +Common resource types include `audio`, `image`, `text`, and `video`. + +```go-html-template +{{ with resources.Get "image/a.jpg" }} + {{ .ResourceType }} → image + {{ .MediaType.MainType }} → image +{{ end }} +``` + +When working with content files, the resource type is `page`. + +```text +content/ +├── lessons/ +│ ├── lesson-1/ +│ │ ├── _objectives.md <-- resource type = page +│ │ ├── _topics.md <-- resource type = page +│ │ ├── _example.jpg <-- resource type = image +│ │ └── index.md +│ └── _index.md +└── _index.md +``` + +With the structure above, we can range through page resources of type `page` to build content: + +{{< code file="layouts/lessons/single.html" lang=go-html-template copy=false >}} +{{ range .Resources.ByType "page" }} + {{ .Content }} +{{ end }} +{{< /code >}} + +{{% include "methods/resource/_common/global-page-remote-resources.md" %}} diff --git a/content/en/methods/resource/Title.md b/content/en/methods/resource/Title.md new file mode 100644 index 00000000000..78540b20a4c --- /dev/null +++ b/content/en/methods/resource/Title.md @@ -0,0 +1,95 @@ +--- +title: Title +description: Returns the title of the given resource as optionally defined in front matter, falling back to a relative path or hashed file name depending on resource type. +categories: [] +keywords: [] +action: + related: + - methods/resource/Name + returnType: string + signatures: [RESOURCE.Title] +toc: true +--- + +The value returned by the `Title` method on a `Resource` object depends on the resource type. + +## Global resource + +With a [global resource], the `Title` method returns the path to the resource, relative to the assets directory. + +```text +assets/ +└── images/ + └── a.jpg +``` + +```go-html-template +{{ with resources.Get "images/a.jpg" }} + {{ .Title }} → images/a.jpg +{{ end }} +``` + +## Page resource + +With a [page resource], the `Title` method returns the path to the resource, relative to the page bundle. + +```text +content/ +├── posts/ +│ ├── post-1/ +│ │ ├── images/ +│ │ │ └── a.jpg +│ │ └── index.md +│ └── _index.md +└── _index.md +``` + +```go-html-template +{{ with .Resources.Get "images/a.jpg" }} + {{ .Title }} → images/a.jpg +{{ end }} +``` + +If you create an element in the `resources` array in front matter, the `Title` method returns the value of the `title` parameter: + +{{< code-toggle file="content/posts/post-1.md" fm=true >}} +title = 'Post 1' +[[resources]] +src = 'images/a.jpg' +name = 'cat' +title = 'Felix the cat' +[resources.params] +temperament = 'malicious' +{{< /code-toggle >}} + +```go-html-template +{{ with .Resources.Get "cat" }} + {{ .Title }} → Felix the cat +{{ end }} +``` + +If the page resource is a content file, the `Title` methods return the `title` field as defined in front matter. + +```text +content/ +├── lessons/ +│ ├── lesson-1/ +│ │ ├── _objectives.md <-- resource type = page +│ │ └── index.md +│ └── _index.md +└── _index.md +``` + +## Remote resource + +With a [remote resource], the `Title` method returns a hashed file name. + +```go-html-template +{{ with resources.GetRemote "https://example.org/images/a.jpg" }} + {{ .Title }} → a_18432433023265451104.jpg +{{ end }} +``` + +[global resource]: /getting-started/glossary/#global-resource +[page resource]: /getting-started/glossary/#page-resource +[remote resource]: /getting-started/glossary/#remote-resource diff --git a/content/en/methods/resource/Width.md b/content/en/methods/resource/Width.md new file mode 100644 index 00000000000..8b96c95e842 --- /dev/null +++ b/content/en/methods/resource/Width.md @@ -0,0 +1,27 @@ +--- +title: Width +description: Applicable to images, returns the width of the given resource. +categories: [] +keywords: [] +action: + related: + - methods/resource/Height + returnType: int + signatures: [RESOURCE.Width] +--- + +```go-html-template +{{ with resources.Get "images/a.jpg" }} + {{ .Width }} → 600 +{{ end }} +``` + +Use the `Width` and `Height` methods together when rendering an `img` element: + +```go-html-template +{{ with resources.Get "images/a.jpg" }} + +{{ end }} +``` + +{{% include "methods/resource/_common/global-page-remote-resources.md" %}} diff --git a/content/en/methods/resource/_common/_index.md b/content/en/methods/resource/_common/_index.md new file mode 100644 index 00000000000..47d5812fba5 --- /dev/null +++ b/content/en/methods/resource/_common/_index.md @@ -0,0 +1,13 @@ +--- +cascade: + _build: + list: never + publishResources: false + render: never +--- + + diff --git a/content/en/methods/resource/_common/global-page-remote-resources.md b/content/en/methods/resource/_common/global-page-remote-resources.md new file mode 100644 index 00000000000..4ea4d1b8745 --- /dev/null +++ b/content/en/methods/resource/_common/global-page-remote-resources.md @@ -0,0 +1,13 @@ +--- +# Do not remove front matter. +--- + +{{% note %}} + +Use this method with [global], [page], or [remote] resources. + +[global]: /getting-started/glossary/#global-resource +[page]: /getting-started/glossary/#page-resource +[remote]: /getting-started/glossary/#remote-resource + +{{% /note %}} diff --git a/content/en/methods/resource/_common/processing-spec.md b/content/en/methods/resource/_common/processing-spec.md new file mode 100644 index 00000000000..1a3ed887dfd --- /dev/null +++ b/content/en/methods/resource/_common/processing-spec.md @@ -0,0 +1,34 @@ +--- +# Do not remove front matter. +--- + +## Process specification + +The process specification is a space-delimited, case-insensitive list of one or more of the following in any sequence: + +action +: Applicable to the [`Process`](/methods/resource/process) method only. Specify zero or one of `resize`, `fit`, `fill`, or `crop`. If you specify an action you must also provide dimensions. + +dimensions +: Provide width _or_ height when using the [`Resize`](/methods/resource/resize) method, else provide both width _and_ height. See [details](/content-management/image-processing/#dimensions). + +anchor +: Use with the [`Crop`](/methods/resource/crop) and [`Fill`](/methods/resource/fill) methods. Specify zero or one of `TopLeft`, `Top`, `TopRight`, `Left`, `Center`, `Right`, `BottomLeft`, `Bottom`, `BottomRight`, or `Smart`. Default is `Smart`. See [details](/content-management/image-processing/#anchor). + +rotation +: Typically specify zero or one of `r90`, `r180`, or `r270`. Also supports arbitrary rotation angles. See [details](/content-management/image-processing/#rotation). + +target format +: Specify zero or one of `gif`, `jpeg`, `png`, `tiff`, or `webp`. See [details](/content-management/image-processing/#target-format). + +quality +: Applicable to JPEG and WebP images. Optionally specify `qN` where `N` is an integer in the range [0, 100]. Default is `75`. See [details](/content-management/image-processing/#quality). + +hint +: Applicable to WebP images. Specify zero or one of `drawing`, `icon`, `photo`, `picture`, or `text`. Default is `photo`. See [details](/content-management/image-processing/#hint). + +background color +: When converting a PNG or WebP with transparency to a format that does not support transparency, optionally specify a background color using a 3-digit or a 6-digit hexadecimal color code. Default is `#ffffff` (white). See [details](/content-management/image-processing/#background-color). + +resampling filter +: Typically specify zero or one of `Box`, `Lanczos`, `CatmullRom`, `MitchellNetravali`, `Linear`, or `NearestNeighbor`. Other resampling filters are available. See [details](/content-management/image-processing/#resampling-filter). diff --git a/content/en/methods/resource/_index.md b/content/en/methods/resource/_index.md new file mode 100644 index 00000000000..e9426e1a59b --- /dev/null +++ b/content/en/methods/resource/_index.md @@ -0,0 +1,12 @@ +--- +title: Resource methods +linkTitle: Resource +description: Use these methods with global, page, and remote Resource objects. +categories: [] +keywords: [] +menu: + docs: + parent: methods +--- + +Use these methods with global, page, and remote Resource objects. diff --git a/content/en/methods/shortcode/Get.md b/content/en/methods/shortcode/Get.md new file mode 100644 index 00000000000..5674f71a2b1 --- /dev/null +++ b/content/en/methods/shortcode/Get.md @@ -0,0 +1,50 @@ +--- +title: Get +description: Returns the value of the given parameter. +categories: [] +keywords: [] +action: + related: + - methods/shortcode/IsNamedParams + returnType: any + signatures: [SHORTCODE.Get PARAM] +toc: true +--- + +Specify the parameter by position or by name. When calling a shortcode within markdown, use either positional or named parameters, but not both. + +{{% note %}} +Some shortcodes support positional parameters, some support named parameters, and others support both. Refer to the shortcode's documentation for usage details. +{{% /note %}} + +## Positional parameters + +This shortcode call uses positional parameters: + +{{< code file="content/about.md" lang=md copy=false >}} +{{}} +{{< /code >}} + +To retrieve parameters by position: + +{{< code file="layouts/shortcodes/myshortcode.html" lang=go-html-template copy=false >}} +{{ printf "%s %s." (.Get 0) (.Get 1) }} → Hello world. +{{< /code >}} + +## Named parameters + +This shortcode call uses named parameters: + +{{< code file="content/about.md" lang=md copy=false >}} +{{}} +{{< /code >}} + +To retrieve parameters by name: + +{{< code file="layouts/shortcodes/myshortcode.html" lang=go-html-template copy=false >}} +{{ printf "%s %s." (.Get "greeting") (.Get "firstName") }} → Hello world. +{{< /code >}} + +{{% note %}} +Parameter names are case-sensitive. +{{% /note %}} diff --git a/content/en/methods/shortcode/Inner.md b/content/en/methods/shortcode/Inner.md new file mode 100644 index 00000000000..98e97b5ea06 --- /dev/null +++ b/content/en/methods/shortcode/Inner.md @@ -0,0 +1,153 @@ +--- +title: Inner +description: Returns the content between opening and closing shortcode tags, applicable when the shortcode call includes a closing tag. +categories: [] +keywords: [] +action: + related: + - functions/strings/Trim + - methods/page/RenderString + - functions/transform/Markdownify + - methods/shortcode/InnerDeindent + returnType: template.HTML + signatures: [SHORTCODE.Inner] +--- + +This content: + +{{< code file="content/services.md" lang=md copy=false >}} +{{}} +We design the **best** widgets in the world. +{{}} +{{< /code >}} + +With this shortcode: + +{{< code file="layouts/shortcodes/card.html" lang=go-html-template copy=false >}} +
+ {{ with .Get "title" }} +
{{ . }}
+ {{ end }} +
+ {{ trim .Inner "\r\n" }} +
+
+{{< /code >}} + +Is rendered to: + +```html +
+
Product Design
+
+ We design the **best** widgets in the world. +
+
+``` + +{{% note %}} +Content between opening and closing shortcode tags may include leading and/or trailing newlines, depending on placement within the markdown. Use the [`trim`] function as shown above to remove both carriage returns and newlines. + +[`trim`]: /functions/strings/trim +{{% /note %}} + +{{% note %}} +In the example above, the value returned by `Inner` is markdown, but it was rendered as plain text. Use either of the following approaches to render markdown to HTML. +{{% /note %}} + + +## Use the RenderString method + +Let's modify the example above to pass the value returned by `Inner` through the [`RenderString`] method on the `Page` object: + +[`RenderString`]: /methods/page/renderstring + +{{< code file="layouts/shortcodes/card.html" lang=go-html-template copy=false >}} +
+ {{ with .Get "title" }} +
{{ . }}
+ {{ end }} +
+ {{ trim .Inner "\r\n" | .Page.RenderString }} +
+
+{{< /code >}} + +Hugo renders this to: + +```html +
+
Product design
+
+ We produce the best widgets in the world. +
+
+``` + +You can use the [`markdownify`] function instead of the `RenderString` method, but the latter is more flexible. See [details]. + +[details]: /methods/page/renderstring +[`markdownify`]: /functions/transform/markdownify + +## Use alternate notation + +Instead of calling the shortcode with the `{{}}` notation, use the `{{%/* */%}}` notation: + +{{< code file="content/services.md" lang=md copy=false >}} +{{%/* card title="Product Design" */%}} +We design the **best** widgets in the world. +{{%/* /card */%}} +{{< /code >}} + +When you use the `{{%/* */%}}` notation, Hugo renders the entire shortcode as markdown, requiring the following changes. + +First, configure the renderer to allow raw HTML within markdown: + +{{< code-toggle file=hugo copy=false >}} +[markup.goldmark.renderer] +unsafe = true +{{< /code-toggle >}} + +This configuration is not unsafe if _you_ control the content. Read more about Hugo's [security model]. + +Second, because we are rendering the entire shortcode as markdown, we must adhere to the rules governing [indentation] and inclusion of [raw HTML blocks] as provided in the [CommonMark] specification. + +{{< code file="layouts/shortcodes/card.html" lang=go-html-template copy=false >}} +
+ {{ with .Get "title" }} +
{{ . }}
+ {{ end }} +
+ + {{ trim .Inner "\r\n" }} +
+
+{{< /code >}} + +The difference between this and the previous example is subtle but required. Note the change in indentation, the addition of a blank line, and removal of the `RenderString` method. + +```diff +--- layouts/shortcodes/a.html ++++ layouts/shortcodes/b.html +@@ -1,8 +1,9 @@ +
+ {{ with .Get "title" }} +-
{{ . }}
++
{{ . }}
+ {{ end }} +
+- {{ trim .Inner "\r\n" | .Page.RenderString }} ++ ++ {{ trim .Inner "\r\n" }} +
+
+``` + +{{% note %}} +When using the `{{%/* */%}}` notation, do not pass the value returned by `Inner` through the `RenderString` method or the `markdownify` function. +{{% /note %}} + +[commonmark]: https://commonmark.org/ +[indentation]: https://spec.commonmark.org/0.30/#indented-code-blocks +[raw html blocks]: https://spec.commonmark.org/0.30/#html-blocks +[security model]: /about/security-model/ diff --git a/content/en/methods/shortcode/InnerDeindent.md b/content/en/methods/shortcode/InnerDeindent.md new file mode 100644 index 00000000000..34191df1fb1 --- /dev/null +++ b/content/en/methods/shortcode/InnerDeindent.md @@ -0,0 +1,99 @@ +--- +title: InnerDeindent +description: Returns the content between opening and closing shortcode tags, with indentation removed, applicable when the shortcode call includes a closing tag. +categories: [] +keywords: [] +action: + related: + - methods/shortcode/Inner + returnType: template.HTML + signatures: [SHORTCODE.InnerDeindent] +--- + +Similar to the [`Inner`] method, `InnerDeindent` returns the content between opening and closing shortcode tags. However, with `InnerDeindent`, indentation before the content is removed. + +This allows us to effectively bypass the rules governing [indentation] as provided in the [CommonMark] specification. + +Consider this markdown, an unordered list with a small gallery of thumbnail images within each list item: + +{{< code file="content/about.md" lang=md copy=false >}} +- Gallery one + + {{}} + ![kitten a](thumbnails/a.jpg) + ![kitten b](thumbnails/b.jpg) + {{}} + +- Gallery two + + {{}} + ![kitten c](thumbnails/c.jpg) + ![kitten d](thumbnails/d.jpg) + {{}} +{{< /code >}} + +In the example above, notice that the content between the opening and closing shortcode tags is indented by four spaces. Per the CommonMark specification, this is treated as an indented code block. + +With this shortcode, calling `Inner` instead of `InnerDeindent`: + +{{< code file="layouts/shortcodes/gallery.html" lang=go-html-template copy=false >}} + +{{< /code >}} + +Hugo renders the markdown to: + +```html +
    +
  • +

    Gallery one

    + +
  • +
  • +

    Gallery two

    + +
  • +
+``` + +Although technically correct per the CommonMark specification, this is not what we want. If we remove the indentation using the `InnerDeindent` method: + +{{< code file="layouts/shortcodes/gallery.html" lang=go-html-template copy=false >}} + +{{< /code >}} + +Hugo renders the markdown to: + +```html +
    +
  • +

    Gallery one

    + +
  • +
  • +

    Gallery two

    + +
  • +
+``` + +[commonmark]: https://commonmark.org/ +[indentation]: https://spec.commonmark.org/0.30/#indented-code-blocks +[`Inner`]: /methods/shortcode/inner diff --git a/content/en/methods/shortcode/IsNamedParams.md b/content/en/methods/shortcode/IsNamedParams.md new file mode 100644 index 00000000000..eac3fc8127c --- /dev/null +++ b/content/en/methods/shortcode/IsNamedParams.md @@ -0,0 +1,30 @@ +--- +title: IsNamedParams +description: Reports whether the shortcode call specifies named or positional parameters. +categories: [] +keywords: [] +action: + related: + - methods/shortcode/Get + returnType: bool + signatures: [SHORTCODE.IsNamedParams] +--- + +To support both positional and named parameters when calling a shortcode, use the `IsNamedParams` method to determine how the shortcode was called. + +With this shortcode template: + +{{< code file="layouts/shortcodes/myshortcode.html" lang=go-html-template copy=false >}} +{{ if .IsNamedParams }} + {{ printf "%s %s." (.Get "greeting") (.Get "firstName") }} +{{ else }} + {{ printf "%s %s." (.Get 0) (.Get 1) }} +{{ end }} +{{< /code >}} + +Both of these calls return the same value: + +{{< code file="content/about.md" lang=md copy=false >}} +{{}} +{{}} +{{< /code >}} diff --git a/content/en/methods/shortcode/Name.md b/content/en/methods/shortcode/Name.md new file mode 100644 index 00000000000..011138e003c --- /dev/null +++ b/content/en/methods/shortcode/Name.md @@ -0,0 +1,29 @@ +--- +title: Name +description: Returns the shortcode file name, excluding the file extension. +categories: [] +keywords: [] +action: + related: + - methods/shortcode/Position + - functions/fmt/Errorf + returnType: string + signatures: [SHORTCODE.Name] +--- + +The `Name` method is useful for error reporting. For example, if your shortcode requires a "greeting" parameter: + +{{< code file="layouts/shortcodes/myshortcode.html" lang=go-html-template copy=false >}} +{{ $greeting := "" }} +{{ with .Get "greeting" }} + {{ $greeting = . }} +{{ else }} + {{ errorf "The %q shortcode requires a 'greeting' parameter. See %s" .Name .Position }} +{{ end }} +{{< /code >}} + +In the absence of a "greeting" parameter, Hugo will throw an error message and fail the build: + +```text +ERROR The "myshortcode" shortcode requires a 'greeting' parameter. See "/home/user/project/content/about.md:11:1" +``` diff --git a/content/en/methods/shortcode/Ordinal.md b/content/en/methods/shortcode/Ordinal.md new file mode 100644 index 00000000000..50cd2b0d31c --- /dev/null +++ b/content/en/methods/shortcode/Ordinal.md @@ -0,0 +1,50 @@ +--- +title: Ordinal +description: Returns the zero-based ordinal of the shortcode in relation to its parent. +categories: [] +keywords: [] +action: + related: [] + returnType: int + signatures: [SHORTCODE.Ordinal] +--- + +The `Ordinal` method returns the zero-based ordinal of the shortcode in relation to its parent. If the parent is the page itself, the ordinal represents the position of this shortcode in the page content. + +This method is useful for, among other things, assigning unique element IDs when a shortcode is called two or more times from the same page. For example: + +{{< code file="content/about.md" lang=md copy=false >}} +{{}} + +{{}} +{{< /code >}} + +This shortcode performs error checking, then renders an HTML `img` element with a unique `id` attribute: + +{{< code file="layouts/shortcodes/img.html" lang=go-html-template copy=false >}} +{{ $src := "" }} +{{ with .Get "src" }} + {{ $src = . }} + {{ with resources.Get $src }} + {{ $id := printf "img-%03d" $.Ordinal }} + + {{ else }} + {{ errorf "The %q shortcode was unable to find %s. See %s" $.Name $src $.Position }} + {{ end }} +{{ else }} + {{ errorf "The %q shortcode requires a 'src' parameter. See %s" .Name .Position }} +{{ end }} +{{< /code >}} + +Hugo renders the page to: + +```html + + +``` + +{{% note %}} +In the shortcode template above, the [`with`] statement is used to create conditional blocks. Remember that the `with` statement binds context (the dot) to its expression. Inside of a `with` block, preface shortcode method calls with a `$` to access the top level context passed into the template. + +[`with`]: /functions/go-template/with +{{% /note %}} diff --git a/content/en/methods/shortcode/Page.md b/content/en/methods/shortcode/Page.md new file mode 100644 index 00000000000..83d414599d1 --- /dev/null +++ b/content/en/methods/shortcode/Page.md @@ -0,0 +1,36 @@ +--- +title: Page +description: Returns the Page object from which the shortcode was called. +categories: [] +keywords: [] +action: + related: [] + returnType: hugolib.pageForShortcode + signatures: [SHORTCODE.Page] +--- + +With this content: + +{{< code-toggle file=content/books/les-miserables.md copy=false fm=true >}} +title = 'Les Misérables' +author = 'Victor Hugo' +published = 1862 +isbn = '978-0451419439' +{{< /code-toggle >}} + +Calling this shortcode: + +```text +{{}} +``` + +We can access the front matter values using the `Page` method: + +{{< code file="layouts/shortcodes/book-details.html" lang=go-html-template copy=false >}} +
    +
  • Title: {{ .Page.Title }}
  • +
  • Author: {{ .Page.Params.author }}
  • +
  • Published: {{ .Page.Params.publication_year }}
  • +
  • ISBN: {{ .Page.Params.isbn }}
  • +
+{{< /code >}} diff --git a/content/en/methods/shortcode/Params.md b/content/en/methods/shortcode/Params.md new file mode 100644 index 00000000000..58da547072d --- /dev/null +++ b/content/en/methods/shortcode/Params.md @@ -0,0 +1,32 @@ +--- +title: Params +description: Returns a collection of the shortcode parameters. +categories: [] +keywords: [] +action: + related: [] + returnType: any + signatures: [SHORTCODE.Params] +--- + +When you call a shortcode using positional parameters, the `Params` method returns a slice. + +{{< code file="content/about.md" lang=md copy=false >}} +{{}} +{{< /code >}} + +{{< code file="layouts/shortcodes/myshortcode.html" lang=go-html-template copy=false >}} +{{ index .Params 0 }} → Hello +{{ index .Params 1 }} → world +{{< /code >}} + +When you call a shortcode using named parameters, the `Params` method returns a map. + +{{< code file="content/about.md" lang=md copy=false >}} +{{}} +{{< /code >}} + +{{< code file="layouts/shortcodes/myshortcode.html" lang=go-html-template copy=false >}} +{{ .Params.greeting }} → Hello +{{ .Params.name }} → world +{{< /code >}} diff --git a/content/en/methods/shortcode/Parent.md b/content/en/methods/shortcode/Parent.md new file mode 100644 index 00000000000..f40148bc5b4 --- /dev/null +++ b/content/en/methods/shortcode/Parent.md @@ -0,0 +1,50 @@ +--- +title: Parent +description: Returns the parent shortcode context in nested shortcodes. +categories: [] +keywords: [] +action: + related: [] + returnType: hugolib.ShortcodeWithPage + signatures: [SHORTCODE.Parent] +--- + +This is useful for inheritance of common shortcode parameters from the root. + +In this contrived example, the "greeting" shortcode is the parent, and the "now" shortcode is child. + +{{< code file="content/welcome.md" lang=md copy=false >}} +{{}} +Welcome. Today is {{}}. +{{}} +{{< /code >}} + +{{< code file="layouts/shortcodes/greeting.html" lang=go-html-template copy=false >}} +
+ {{ trim .Inner "\r\n" | .Page.RenderString }} +
+{{< /code >}} + +{{< code file="layouts/shortcodes/now.html" lang=go-html-template copy=false >}} +{{- $dateFormat := "January 2, 2006 15:04:05" }} + +{{- with .Params }} + {{- with .dateFormat }} + {{- $dateFormat = . }} + {{- end }} +{{- else }} + {{- with .Parent.Params }} + {{- with .dateFormat }} + {{- $dateFormat = . }} + {{- end }} + {{- end }} +{{- end }} + +{{- now | time.Format $dateFormat -}} +{{< /code >}} + +The "now" shortcode formats the current time using: + +1. The `dateFormat` parameter passed to the "now" shortcode, if present +2. The `dateFormat` parameter passed to the "greeting" shortcode, if present +3. The default layout string defined at the top of the shortcode diff --git a/content/en/methods/shortcode/Position.md b/content/en/methods/shortcode/Position.md new file mode 100644 index 00000000000..181a13e4d1c --- /dev/null +++ b/content/en/methods/shortcode/Position.md @@ -0,0 +1,33 @@ +--- +title: Position +description: Returns the filename and position from which the shortcode was called. +categories: [] +keywords: [] +action: + related: + - methods/shortcode/Name + - functions/fmt/Errorf + returnType: text.Position + signatures: [SHORTCODE.Position] +--- + +The `Position` method is useful for error reporting. For example, if your shortcode requires a "greeting" parameter: + +{{< code file="layouts/shortcodes/myshortcode.html" lang=go-html-template copy=false >}} +{{ $greeting := "" }} +{{ with .Get "greeting" }} + {{ $greeting = . }} +{{ else }} + {{ errorf "The %q shortcode requires a 'greeting' parameter. See %s" .Name .Position }} +{{ end }} +{{< /code >}} + +In the absence of a "greeting" parameter, Hugo will throw an error message and fail the build: + +```text +ERROR The "myshortcode" shortcode requires a 'greeting' parameter. See "/home/user/project/content/about.md:11:1" +``` + +{{% note %}} +The position can relatively expensive to calculate. Limit its use to error reporting. +{{% /note %}} diff --git a/content/en/methods/shortcode/Scratch.md b/content/en/methods/shortcode/Scratch.md new file mode 100644 index 00000000000..76586abe275 --- /dev/null +++ b/content/en/methods/shortcode/Scratch.md @@ -0,0 +1,24 @@ +--- +title: Scratch +description: Creates a "scratch pad" scoped to the shortcode to store and manipulate data. +categories: [] +keywords: [] +action: + related: + - functions/collections/NewScratch + returnType: maps.Scratch + signatures: [SHORTCODE.Scratch] +--- + +The `Scratch` method within a shortcode creates a [scratch pad] to store and manipulate data. The scratch pad is scoped to the shortcode, and is reset on server rebuilds. + +{{% note %}} +With the introduction of the [`newScratch`] function, and the ability to [assign values to template variables] after initialization, the `Scratch` method within a shortcode is obsolete. + +[assign values to variables]: https://go.dev/doc/go1.11#text/template +[`newScratch`]: functions/collections/newscratch +{{% /note %}} + +[scratch pad]: /getting-started/glossary/#scratch-pad + +{{% include "methods/page/_common/scratch-methods.md" %}} diff --git a/content/en/methods/shortcode/_index.md b/content/en/methods/shortcode/_index.md new file mode 100644 index 00000000000..d26366844fd --- /dev/null +++ b/content/en/methods/shortcode/_index.md @@ -0,0 +1,12 @@ +--- +title: Shortcode methods +linkTitle: Shortcode +description: Use these methods in your shortcode templates. +categories: [] +keywords: [] +menu: + docs: + parent: methods +--- + +Use these methods in your shortcode templates. diff --git a/content/en/methods/site/AllPages.md b/content/en/methods/site/AllPages.md new file mode 100644 index 00000000000..8df6348f9b5 --- /dev/null +++ b/content/en/methods/site/AllPages.md @@ -0,0 +1,26 @@ +--- +title: AllPages +description: Returns a collection of all pages in all languages. +categories: [] +keywords: [] +action: + related: + - methods/site/Pages + - methods/site/RegularPages + - methods/site/Sections + returnType: page.Pages + signatures: [SITE.AllPages] +--- + +This method returns all page [kinds] in all languages. That includes the home page, section pages, taxonomy pages, term pages, and regular pages. + +In most cases you should use the [`RegularPages`] method instead. + +[`RegularPages`]: methods/site/regularpages +[kinds]: /getting-started/glossary/#page-kind + +```go-html-template +{{ range .Site.AllPages }} +

{{ .LinkTitle }}

+{{ end }} +``` diff --git a/content/en/methods/site/BaseURL.md b/content/en/methods/site/BaseURL.md new file mode 100644 index 00000000000..f9c43bca357 --- /dev/null +++ b/content/en/methods/site/BaseURL.md @@ -0,0 +1,37 @@ +--- +title: BaseURL +description: Returns the base URL as defined in the site configuration. +categories: [] +keywords: [] +action: + related: + - functions/urls/AbsURL + - functions/urls/AbsLangURL + - functions/urls/RelURL + - functions/urls/RelLangURL + returnType: string + signatures: [SITE.BaseURL] +--- + +Site configuration: + +{{< code-toggle file=hugo >}} +baseURL = 'https://example.org/docs/' +{{< /code-toggle >}} + +Template: + +```go-html-template +{{ .Site.BaseURL }} → https://example.org/docs/ +``` + +{{% note %}} +There is almost never a good reason to use this method in your templates. Its usage tends to be fragile due to misconfiguration. + +Use the [`absURL`], [`absLangURL`], [`relURL`], or [`relLangURL`] functions instead. + +[`absURL`]: /functions/urls/absURL +[`absLangURL`]: /functions/urls/absLangURL +[`relURL`]: /functions/urls/relURL +[`relLangURL`]: /functions/urls/relLangURL +{{% /note %}} diff --git a/content/en/methods/site/BuildDrafts.md b/content/en/methods/site/BuildDrafts.md new file mode 100644 index 00000000000..0d85c78fda3 --- /dev/null +++ b/content/en/methods/site/BuildDrafts.md @@ -0,0 +1,28 @@ +--- +title: BuildDrafts +description: Reports whether the current build includes draft pages. +categories: [] +keywords: [] +action: + related: [] + returnType: bool + signatures: [SITE.BuildDrafts] +--- + +By default, draft pages are not published when building a site. You can change this behavior with a command line flag: + +```sh +hugo --buildDrafts +``` + +Or by setting `buildDrafts` to `true` in your site configuration: + +{{< code-toggle file=hugo >}} +buildDrafts = true +{{< /code-toggle >}} + +Use the `BuildDrafts` method on a `Site` object to determine the current configuration: + +```go-html-template +{{ .Site.BuildDrafts }} → true +``` diff --git a/content/en/methods/site/Config.md b/content/en/methods/site/Config.md new file mode 100644 index 00000000000..b18371d086e --- /dev/null +++ b/content/en/methods/site/Config.md @@ -0,0 +1,57 @@ +--- +title: Config +description: Returns a subset of the site configuration. +categories: [] +keywords: [] +action: + related: [] + returnType: page.SiteConfig + signatures: [SITE.Config] +toc: true +--- + +The `Config` method on a `Site` object provides access to a subset of the site configuration, specifically the `services` and `privacy` keys. + +## Services + +These are the default service settings, typically used by Hugo's built-in templates and shortcodes. + +{{< code-toggle config="services" />}} + +For example, to use Hugo's built-in Google Analytics template you must add a [Google tag ID]: + +[Google tag ID]: https://support.google.com/tagmanager/answer/12326985?hl=en + +{{< code-toggle file=hugo >}} +[services.googleAnalytics] +id = 'G-XXXXXXXXX' +{{< /code-toggle >}} + +To access this value from a template: + +```go-html-template +{{ .Site.Config.Services.GoogleAnalytics.ID }} → G-XXXXXXXXX +``` + +You must capitalize each identifier as shown above. + +## Privacy + +These are the default privacy settings, typically used by Hugo's built-in templates and shortcodes: + +{{< code-toggle config="privacy" />}} + +For example, to disable usage of the built-in YouTube shortcode: + +{{< code-toggle file=hugo >}} +[privacy.youtube] +disable = true +{{< /code-toggle >}} + +To access this value from a template: + +```go-html-template +{{ .Site.Config.Privacy.YouTube.Disable }} → true +``` + +You must capitalize each identifier as shown above. diff --git a/content/en/methods/site/Copyright.md b/content/en/methods/site/Copyright.md new file mode 100644 index 00000000000..e2ae7d2a5a3 --- /dev/null +++ b/content/en/methods/site/Copyright.md @@ -0,0 +1,22 @@ +--- +title: Copyright +description: Returns the copyright notice as defined in the site configuration. +categories: [] +keywords: [] +action: + related: [] + returnType: string + signatures: [SITE.Copyright] +--- + +Site configuration: + +{{< code-toggle file=hugo >}} +copyright = '© 2023 ABC Widgets, Inc.' +{{< /code-toggle >}} + +Template: + +```go-html-template +{{ .Site.Copyright }} → © 2023 ABC Widgets, Inc. +``` diff --git a/content/en/methods/site/Data.md b/content/en/methods/site/Data.md new file mode 100644 index 00000000000..f6b38b5973b --- /dev/null +++ b/content/en/methods/site/Data.md @@ -0,0 +1,108 @@ +--- +title: Data +description: Returns a data structure composed from the files in the data directory. +categories: [] +keywords: [] +action: + related: + - functions/collections/IndexFunction + - functions/transform/Unmarshal + - functions/collections/Where + - functions/collections/Sort + returnType: map + signatures: [SITE.Data] +--- + +Use the `Data` method on a `Site` object to access data within the data directory, or within any directory [mounted] to the data directory. Supported data formats include JSON, TOML, YAML, and XML. + +[mounted]: /hugo-modules/configuration/#module-configuration-mounts + +{{% note %}} +Although Hugo can unmarshal CSV files with the [`transform.Unmarshal`] function, do not place CSV files in the data directory. You cannot access data within CSV files using this method. + +[`transform.Unmarshal`]: /functions/transform/unmarshal +{{% /note %}} + +Consider this data directory: + +```text +data/ +├── books/ +│ ├── fiction.yaml +│ └── nonfiction.yaml +├── films.json +├── paintings.xml +└── sculptures.toml +``` + +And these data files: + +{{< code file="data/books/fiction.yaml" lang=yaml >}} +- title: The Hunchback of Notre Dame + author: Victor Hugo + isbn: 978-0140443530 +- title: Les Misérables + author: Victor Hugo + isbn: 978-0451419439 +{{< /code >}} + +{{< code file="data/books/nonfiction.yaml" lang=yaml >}} +- title: The Ancien Régime and the Revolution + author: Alexis de Tocqueville + isbn: 978-0141441641 +- title: Interpreting the French Revolution + author: François Furet + isbn: 978-0521280495 +{{< /code >}} + +Access the data by [chaining] the [identifiers]: + +```go-html-template +{{ range $category, $books := .Site.Data.books }} +

{{ $category | title }}

+
    + {{ range $books }} +
  • {{ .title }} ({{ .isbn }})
  • + {{ end }} +
+{{ end }} +``` + +Hugo renders this to: + +```html +

Fiction

+
    +
  • The Hunchback of Notre Dame (978-0140443530)
  • +
  • Les Misérables (978-0451419439)
  • +
+

Nonfiction

+
    +
  • The Ancien Régime and the Revolution (978-0141441641)
  • +
  • Interpreting the French Revolution (978-0521280495)
  • +
+``` + +To limit the listing to fiction, and sort by title: + +```go-html-template +
    + {{ range sort .Site.Data.books.fiction "title" }} +
  • {{ .title }} ({{ .author }})
  • + {{ end }} +
+``` + +To find a fiction book by ISBN: + +```go-html-template +{{ range where .Site.Data.books.fiction "isbn" "978-0140443530" }} +
  • {{ .title }} ({{ .author }})
  • +{{ end }} +``` + +In the template examples above, each of the keys is a valid identifier. For example, none of the keys contains a hyphen. To access a key that is not a valid identifier, use the [`index`] function: + +[`index`]: /functions/collections/indexfunction +[chaining]: /getting-started/glossary/#chain +[identifiers]: /getting-started/glossary/#identifier diff --git a/content/en/methods/site/GetPage.md b/content/en/methods/site/GetPage.md new file mode 100644 index 00000000000..1e949f37c92 --- /dev/null +++ b/content/en/methods/site/GetPage.md @@ -0,0 +1,109 @@ +--- +title: GetPage +description: Returns a Page object from the given path. +categories: [] +keywords: [] +action: + related: + - methods/page/GetPage + returnType: hugolib.pageState + signatures: [SITE.GetPage PATH] +toc: true +--- + +The `GetPage` method is also available on `Page` objects, allowing you to specify a path relative to the current page. See [details]. + +[details]: /methods/page/getpage + +When using the `GetPage` method on a `Site` object, specify a path relative to the content directory. + +If Hugo cannot resolve the path to a page, the method returns nil. + +Consider this content structure: + +```text +content/ +├── works/ +│ ├── paintings/ +│ │ ├── _index.md +│ │ ├── starry-night.md +│ │ └── the-mona-lisa.md +│ ├── sculptures/ +│ │ ├── _index.md +│ │ ├── david.md +│ │ └── the-thinker.md +│ └── _index.md +└── _index.md +``` + +This home page template: + +```go-html-template +{{ with .Site.GetPage "/works/paintings" }} +
      + {{ range .Pages }} +
    • {{ .Title }} by {{ .Params.artist }}
    • + {{ end }} +
    +{{ end }} +``` + +Is rendered to: + +```html +
      +
    • Starry Night by Vincent van Gogh
    • +
    • The Mona Lisa by Leonardo da Vinci
    • +
    +``` + +To get a regular page instead of a section page: + +```go-html-template +{{ with .Site.GetPage "/works/paintings/starry-night" }} + {{ .Title }} → Starry Night + {{ .Params.artist }} → Vincent van Gogh +{{ end }} +``` + +## Multilingual projects + +With multilingual projects, the `GetPage` method on a `Site` object resolves the given path to a page in the current language. + +To get a page from a different language, query the `Sites` object: + +```go-html-template +{{ with where .Site.Sites "Language.Lang" "eq" "de" }} + {{ with index . 0 }} + {{ with .GetPage "/works/paintings/starry-night" }} + {{ .Title }} → Sternenklare Nacht + {{ end }} + {{ end }} +{{ end }} +``` + +## Page bundles + +Consider this content structure: + +```text +content/ +├── headless/ +│ ├── a.jpg +│ ├── b.jpg +│ ├── c.jpg +│ └── index.md <-- front matter: headless = true +└── _index.md +``` + +In the home page template, use the `GetPage` method on a `Site` object to render all the images in the headless [page bundle]: + +```go-html-template +{{ with .Site.GetPage "/headless" }} + {{ range .Resources.ByType "image" }} + + {{ end }} +{{ end }} +``` + +[page bundle]: /getting-started/glossary/#page-bundle diff --git a/content/en/methods/site/Home.md b/content/en/methods/site/Home.md new file mode 100644 index 00000000000..52612dd12ed --- /dev/null +++ b/content/en/methods/site/Home.md @@ -0,0 +1,25 @@ +--- +title: Home +description: Returns the home Page object for the given site. +categories: [] +keywords: [] +action: + related: [] + returnType: hugolib.pageState + signatures: [SITE.Home] +--- + +This method is useful for obtaining a link to the home page. + +Site configuration: + +{{< code-toggle file=hugo >}} +baseURL = 'https://example.org/docs/' +{{< /code-toggle >}} + +Template: + +```go-html-template +{{ .Site.Home.Permalink }} → https://example.org/docs/ +{{ .Site.Home.RelPermalink }} → /docs/ +``` diff --git a/content/en/methods/site/IsMultiLingual.md b/content/en/methods/site/IsMultiLingual.md new file mode 100644 index 00000000000..61cc5e46240 --- /dev/null +++ b/content/en/methods/site/IsMultiLingual.md @@ -0,0 +1,34 @@ +--- +title: IsMultiLingual +description: Reports whether the site is multilingual. +categories: [] +keywords: [] +action: + related: [] + returnType: bool + signatures: [SITE.IsMultiLingual] +--- + +Site configuration: + +{{< code-toggle file=hugo >}} +defaultContentLanguage = 'de' +defaultContentLanguageInSubdir = true +[languages] + [languages.de] + languageCode = 'de-DE' + languageName = 'Deutsch' + title = 'Projekt Dokumentation' + weight = 1 + [languages.en] + languageCode = 'en-US' + languageName = 'English' + title = 'Project Documentation' + weight = 2 +{{< /code-toggle >}} + +Template: + +```go-html-template +{{ .Site.IsMultiLingual }} → true +``` diff --git a/content/en/methods/site/Language.md b/content/en/methods/site/Language.md new file mode 100644 index 00000000000..1babc099bbd --- /dev/null +++ b/content/en/methods/site/Language.md @@ -0,0 +1,83 @@ +--- +title: Language +description: Returns the language object for the given site. +categories: [] +keywords: [] +action: + related: + - methods/page/language + returnType: langs.Language + signatures: [SITE.Language] +toc: true +--- + +The `Language` method on a `Site` object returns the language object for the given site. The language object points to the language definition in the site configuration. + +You can also use the `Language` method on a `Page` object. See [details]. + +## Methods + +The examples below assume the following in your site configuration: + +{{< code-toggle file=hugo >}} +[languages.de] +languageCode = 'de-DE' +languageDirection = 'ltr' +languageName = 'Deutsch' +weight = 1 +{{< /code-toggle >}} + +Lang +: (`string`) The language tag as defined by [RFC 5646]. + +```go-html-template +{{ .Site.Language.Lang }} → de +``` + +LanguageCode +: (`string`) The language code from the site configuration. + +```go-html-template +{{ .Site.Language.LanguageCode }} → de-DE +``` + +LanguageDirection +: (`string`) The language direction from the site configuration, either `ltr` or `rtl`. + +```go-html-template +{{ .Site.Language.LanguageDirection }} → ltr +``` + +LanguageName +: (`string`) The language name from the site configuration. + +```go-html-template +{{ .Site.Language.LanguageName }} → Deutsch +``` + +Weight +: (`int`) The language weight from the site configuration which determines its order in the slice of languages returned by the `Languages` method on a `Site` object. + +```go-html-template +{{ .Site.Language.Weight }} → 1 +``` + +## Example + +Some of the methods above are commonly used in a base template as attributes for the `html` element. + +```go-html-template +{{ jsonify (dict "indent" " ") .Site.Languages }} +``` + +With this site configuration: + +{{< code-toggle file=hugo >}} +defaultContentLanguage = 'de' +defaultContentLanguageInSubdir = false + +[languages.de] +languageCode = 'de-DE' +languageDirection = 'ltr' +languageName = 'Deutsch' +title = 'Projekt Dokumentation' +weight = 1 + +[languages.en] +languageCode = 'en-US' +languageDirection = 'ltr' +languageName = 'English' +title = 'Project Documentation' +weight = 2 +{{< /code-toggle >}} + +This template: + +```go-html-template +
      + {{ range .Site.Languages }} +
    • {{ .Title }} ({{ .LanguageName }})
    • + {{ end }} +
    +``` + +Is rendered to: + +```html +
      +
    • Projekt Dokumentation (Deutsch)
    • +
    • Project Documentation (English)
    • +
    +``` diff --git a/content/en/methods/site/LastChange.md b/content/en/methods/site/LastChange.md new file mode 100644 index 00000000000..aceee691d02 --- /dev/null +++ b/content/en/methods/site/LastChange.md @@ -0,0 +1,21 @@ +--- +title: LastChange +description: Returns the last modification date of site content. +categories: [] +keywords: [] +action: + related: [] + returnType: time.Time + signatures: [SITE.LastChange] +--- + +The `LastChange` method on a `Site` object returns a [`time.Time`] value. Use this with time [functions] and [methods]. For example: + +```go-html-template +{{ .Site.LastChange | time.Format ":date_long" }} → October 16, 2023 + +``` + +[`time.Time`]: https://pkg.go.dev/time#Time +[functions]: /functions/time +[methods]: /methods/time diff --git a/content/en/methods/site/MainSections.md b/content/en/methods/site/MainSections.md new file mode 100644 index 00000000000..251fe1a977c --- /dev/null +++ b/content/en/methods/site/MainSections.md @@ -0,0 +1,55 @@ +--- +title: MainSections +description: Returns a slice of the main section names as defined in the site configuration, falling back to the top level section with the most pages. +categories: [] +keywords: [] +action: + related: [] + returnType: '[]string' + signatures: [SITE.MainSections] +--- + +Site configuration: + +{{< code-toggle file=hugo >}} +[params] +mainSections = ['books','films'] +{{< /code-toggle >}} + +Template: + +```go-html-template +{{ .Site.MainSections }} → [books films] +``` + +If `params.mainSections` is not defined in the site configuration, this method returns a slice with one element---the top level section with the most pages. + +With this content structure, the "films" section has the most pages: + +```text +content/ +├── books/ +│ ├── book-1.md +│ └── book-2.md +├── films/ +│ ├── film-1.md +│ ├── film-2.md +│ └── film-3.md +└── _index.md +``` + +Template: + +```go-html-template +{{ .Site.MainSections }} → [films] +``` + +When creating a theme, instead of hardcoding section names when listing the most relevant pages on the front page, instruct site authors to set `params.mainSections` in their site configuration. + +Then your home page template can do something like this: + +```go-html-template +{{ range where .Site.RegularPages "Section" "in" .Site.MainSections }} +

    {{ .LinkTitle }}

    +{{ end }} +``` diff --git a/content/en/methods/site/Menus.md b/content/en/methods/site/Menus.md new file mode 100644 index 00000000000..1967a921126 --- /dev/null +++ b/content/en/methods/site/Menus.md @@ -0,0 +1,94 @@ +--- +title: Menus +description: Returns a collection of menu objects for the given site. +categories: [] +keywords: [] +action: + related: + - methods/page/IsMenuCurrent + - methods/page/HasMenuCurrent + returnType: navigation.Menus + signatures: [SITE.Menus] +--- + +The `Menus` method on a `Site` object returns a collection of menus, where each menu contains one or more entries, either flat or nested. Each entry points to a page within the site, or to an external resource. + +{{% note %}} +Menus can be defined and localized in several ways. Please see the [menus] section for a complete explanation and examples. + +[menus]: /content-management/menus/ +{{% /note %}} + +A site can have multiple menus. For example, a main menu and a footer menu: + +{{< code-toggle file=hugo >}} +[[menu.main]] +name = 'Home' +pageRef = '/' +weight = 10 + +[[menu.main]] +name = 'Books' +pageRef = '/books' +weight = 20 + +[[menu.main]] +name = 'Films' +pageRef = '/films' +weight = 30 + +[[menu.footer]] +name = 'Legal' +pageRef = '/legal' +weight = 10 + +[[menu.footer]] +name = 'Privacy' +pageRef = '/privacy' +weight = 20 +{{< /code-toggle >}} + +This template renders the main menu: + +```go-html-template +{{ with site.Menus.main }} + +{{ end }} +``` + +When viewing the home page, the result is: + +```html + +``` + +When viewing the "books" page, the result is: + +```html + +``` + +You will typically render a menu using a partial template. As the active menu entry will be different on each page, use the [`partial`] function to call the template. Do not use the [`partialCached`] function. + +The example above is simplistic. Please see the [menu templates] section for more information. + +[menu templates]: /templates/menu-templates + +[`partial`]: /functions/partials/include +[`partialCached`]: /functions/partials/includecached diff --git a/content/en/methods/site/Pages.md b/content/en/methods/site/Pages.md new file mode 100644 index 00000000000..583e98c11c9 --- /dev/null +++ b/content/en/methods/site/Pages.md @@ -0,0 +1,26 @@ +--- +title: Pages +description: Returns a collection of all pages. +categories: [] +keywords: [] +action: + related: + - methods/site/AllPages + - methods/site/RegularPages + - methods/site/Sections + returnType: page.Pages + signatures: [SITE.Pages] +--- + +This method returns all page [kinds] in the current language. That includes the home page, section pages, taxonomy pages, term pages, and regular pages. + +In most cases you should use the [`RegularPages`] method instead. + +[`RegularPages`]: methods/site/regularpages +[kinds]: /getting-started/glossary/#page-kind + +```go-html-template +{{ range .Site.Pages }} +

    {{ .LinkTitle }}

    +{{ end }} +``` diff --git a/content/en/methods/site/Params.md b/content/en/methods/site/Params.md new file mode 100644 index 00000000000..518d93bf3e5 --- /dev/null +++ b/content/en/methods/site/Params.md @@ -0,0 +1,47 @@ +--- +title: Params +description: Returns a map of custom parameters as defined in the site configuration. +categories: [] +keywords: [] +action: + related: + - functions/collections/indexFunction + - methods/page/Params + - methods/page/Param + returnType: maps.Params + signatures: [SITE.Params] +--- + +With this site configuration: + +{{< code-toggle file=hugo >}} +[params] + subtitle = 'The Best Widgets on Earth' + copyright-year = '2023' + [params.author] + email = 'jsmith@example.org' + name = 'John Smith' + [params.layouts] + rfc_1123 = 'Mon, 02 Jan 2006 15:04:05 MST' + rfc_3339 = '2006-01-02T15:04:05-07:00' +{{< /code-toggle >}} + +Access the custom parameters by [chaining] the [identifiers]: + +```go-html-template +{{ .Site.Params.subtitle }} → The Best Widgets on Earth +{{ .Site.Params.author.name }} → John Smith + +{{ $layout := .Site.Params.layouts.rfc_1123 }} +{{ .Site.LastChange.Format $layout }} → Tue, 17 Oct 2023 13:21:02 PDT +``` + +In the template example above, each of the keys is a valid identifier. For example, none of the keys contains a hyphen. To access a key that is not a valid identifier, use the [`index`] function: + +```go-html-template +{{ index .Site.Params "copyright-year" }} → 2023 +``` + +[`index`]: /functions/collections/indexfunction +[chaining]: /getting-started/glossary/#chain +[identifiers]: /getting-started/glossary/#identifier diff --git a/content/en/methods/site/RegularPages.md b/content/en/methods/site/RegularPages.md new file mode 100644 index 00000000000..744eb8a1391 --- /dev/null +++ b/content/en/methods/site/RegularPages.md @@ -0,0 +1,38 @@ +--- +title: RegularPages +description: Returns a collection of all regular pages. +categories: [] +keywords: [] +action: + related: + - methods/site/AllPages + - methods/site/RegularPages + - methods/site/Sections + returnType: page.Pages + signatures: [SITE.RegularPages] +--- + +```go-html-template +{{ range .Site.RegularPages }} +

    {{ .LinkTitle }}

    +{{ end }} +``` + +By default, Hugo sorts page collections by: + +1. The page `weight` as defined in front matter +1. The page `date` as defined in front matter +1. The page `linkTitle` as defined in front matter +1. The file path + +If the `linkTitle` is not defined, Hugo evaluates the `title` instead. + +To change the sort order, use any of the `Pages` [sorting methods]. For example: + +```go-html-template +{{ range .Site.RegularPages.ByTitle }} +

    {{ .LinkTitle }}

    +{{ end }} +``` + +[sorting methods]: /methods/pages/ diff --git a/content/en/methods/site/Sections.md b/content/en/methods/site/Sections.md new file mode 100644 index 00000000000..a397c5926b4 --- /dev/null +++ b/content/en/methods/site/Sections.md @@ -0,0 +1,41 @@ +--- +title: Sections +description: Returns a collection of first level section pages. +categories: [] +keywords: [] +action: + related: + - methods/site/AllPages + - methods/site/Pages + - methods/site/RegularPages + returnType: page.Pages + signatures: [SITE.Sections] +--- + +Given this content structure: + +```text +content/ +├── books/ +│ ├── book-1.md +│ └── book-2.md +├── films/ +│ ├── film-1.md +│ └── film-2.md +└── _index.md +``` + +This template: + +```go-html-template +{{ range .Site.Sections }} +

    {{ .LinkTitle }}

    +{{ end }} +``` + +Is rendered to: + +```html +

    Books

    +

    Films

    +``` diff --git a/content/en/methods/site/Sites.md b/content/en/methods/site/Sites.md new file mode 100644 index 00000000000..f7bafd3ed2c --- /dev/null +++ b/content/en/methods/site/Sites.md @@ -0,0 +1,66 @@ +--- +title: Sites +description: Returns a collection of all Site objects, one for each language, ordered by language weight. +categories: [] +keywords: [] +action: + related: [] + returnType: page.Sites + signatures: [SITE.Sites] +--- + +With this site configuration: + +{{< code-toggle file=hugo >}} +defaultContentLanguage = 'de' +defaultContentLanguageInSubdir = false + +[languages.de] +languageCode = 'de-DE' +languageDirection = 'ltr' +languageName = 'Deutsch' +title = 'Projekt Dokumentation' +weight = 1 + +[languages.en] +languageCode = 'en-US' +languageDirection = 'ltr' +languageName = 'English' +title = 'Project Documentation' +weight = 2 +{{< /code-toggle >}} + +This template: + +```go-html-template + +``` + +Produces a list of links to each home page: + +```html + +``` + +To render a link to home page of the primary (first) language: + +```go-html-template +{{ with .Site.Sites.First }} + {{ .Title }} +{{ end }} +``` + +This is equivalent to: + +```go-html-template +{{ with index .Site.Sites 0 }} + {{ .Title }} +{{ end }} +``` diff --git a/content/en/methods/site/Taxonomies.md b/content/en/methods/site/Taxonomies.md new file mode 100644 index 00000000000..72bfc75d56c --- /dev/null +++ b/content/en/methods/site/Taxonomies.md @@ -0,0 +1,99 @@ +--- +title: Taxonomies +description: Returns a data structure containing the site's taxonomy objects, the terms within each taxonomy object, and the pages to which the terms are assigned. +categories: [] +keywords: [] +action: + related: [] + returnType: page.TaxonomyList + signatures: [SITE.Taxonomies] +--- + +Conceptually, the `Taxonomies` method on a `Site` object returns a data structure such as: + +{{< code-toggle >}} +taxonomy a: + - term 1: + - page 1 + - page 2 + - term 2: + - page 1 +taxonomy b: + - term 1: + - page 2 + - term 2: + - page 1 + - page 2 +{{< /code-toggle >}} + +For example, on a book review site you might create two taxonomies; one for genres and another for authors. + +With this site configuration: + +{{< code-toggle file=hugo >}} +[taxonomies] +genre = 'genres' +author = 'authors' +{{< /code-toggle >}} + +And this content structure: + +```text +content/ +├── books/ +│ ├── and-then-there-were-none.md --> genres: suspense +│ ├── death-on-the-nile.md --> genres: suspense +│ └── jamaica-inn.md --> genres: suspense, romance +│ └── pride-and-prejudice.md --> genres: romance +└── _index.md +``` + +Conceptually, the taxonomies data structure looks like: + +{{< code-toggle >}} +genres: + - suspense: + - And Then There Were None + - Death on the Nile + - Jamaica Inn + - romance: + - Jamaica Inn + - Pride and Prejudice +authors: + - achristie: + - And Then There Were None + - Death on the Nile + - ddmaurier: + - Jamaica Inn + - jausten: + - Pride and Prejudice +{{< /code-toggle >}} + + +To list the "suspense" books: + +```go-html-template +
      + {{ range .Site.Taxonomies.genres.suspense }} +
    • {{ .LinkTitle }}
    • + {{ end }} +
    +``` + +Hugo renders this to: + +```html + +``` + +{{% note %}} +Hugo's taxonomy system is powerful, allowing you to classify content and create relationships between pages. + +Please see the [taxonomies] section for a complete explanation and examples. + +[taxonomies]: content-management/taxonomies/ +{{% /note %}} diff --git a/content/en/methods/site/Title.md b/content/en/methods/site/Title.md new file mode 100644 index 00000000000..a357286c1f8 --- /dev/null +++ b/content/en/methods/site/Title.md @@ -0,0 +1,22 @@ +--- +title: Title +description: Returns the title as defined in the site configuration. +categories: [] +keywords: [] +action: + related: [] + returnType: string + signatures: [SITE.Title] +--- + +Site configuration: + +{{< code-toggle file=hugo >}} +title = 'My Documentation Site' +{{< /code-toggle >}} + +Template: + +```go-html-template +{{ .Site.Title }} → My Documentation Site +``` diff --git a/content/en/methods/site/_index.md b/content/en/methods/site/_index.md new file mode 100644 index 00000000000..39f66f308bc --- /dev/null +++ b/content/en/methods/site/_index.md @@ -0,0 +1,12 @@ +--- +title: Site methods +linkTitle: Site +description: Use these methods with Site objects. +categories: [] +keywords: [] +menu: + docs: + parent: methods +--- + +Use these methods with Site objects. A multilingual project will have two or more sites, one for each language. diff --git a/content/en/methods/taxonomy/Alphabetical.md b/content/en/methods/taxonomy/Alphabetical.md new file mode 100644 index 00000000000..2c07fbef4ae --- /dev/null +++ b/content/en/methods/taxonomy/Alphabetical.md @@ -0,0 +1,78 @@ +--- +title: Alphabetical +description: Returns an ordered taxonomy, sorted alphabetically by term. +categories: [] +keywords: [] +action: + related: + - methods/taxonomy/ByCount + returnType: page.OrderedTaxonomy + signatures: [TAXONOMY.Alphabetical] +toc: true +--- + +The `Alphabetical` method on a `Taxonomy` object returns an [ordered taxonomy], sorted alphabetically by [term]. + +While a `Taxonomy` object is a [map], an ordered taxonomy is a [slice], where each element is an object that contains the term and a slice of its [weighted pages]. + +{{% include "methods/taxonomy/_common/get-a-taxonomy-object.md" %}} + +## Get the ordered taxonomy + +Now that we have captured the “genres” Taxonomy object, let’s get the ordered taxonomy sorted alphabetically by term: + +```go-html-template +{{ $taxonomyObject.Alphabetical }} +``` + +To reverse the sort order: + +```go-html-template +{{ $taxonomyObject.Alphabetical.Reverse }} +``` + +To inspect the data structure: + +```go-html-template +
    {{ jsonify (dict "indent" "  ") $taxonomyObject.Alphabetical }}
    +``` + +{{% include "methods/taxonomy/_common/ordered-taxonomy-element-methods.md" %}} + +## Example + +With this template: + +```go-html-template +{{ range $taxonomyObject.Alphabetical }} +

    {{ .Page.LinkTitle }} ({{ .Count }})

    + +{{ end }} +``` + +Hugo renders: + +```html +

    romance (2)

    + +

    suspense (3)

    + +``` + +[ordered taxonomy]: /getting-started/glossary/#ordered-taxonomy +[term]: /getting-started/glossary/#term +[map]: /getting-started/glossary/#map +[slice]: /getting-started/glossary/#slice +[term]: /getting-started/glossary/#term +[weighted pages]: /getting-started/glossary/#weighted-page diff --git a/content/en/methods/taxonomy/ByCount.md b/content/en/methods/taxonomy/ByCount.md new file mode 100644 index 00000000000..d0caa7d2c8e --- /dev/null +++ b/content/en/methods/taxonomy/ByCount.md @@ -0,0 +1,78 @@ +--- +title: ByCount +description: Returns an ordered taxonomy, sorted by the number of pages associated with each term. +categories: [] +keywords: [] +action: + related: + - methods/taxonomy/Alphabetical + returnType: page.OrderedTaxonomy + signatures: [TAXONOMY.ByCount] +toc: true +--- + +The `ByCount` method on a `Taxonomy` object returns an [ordered taxonomy], sorted by the number of pages associated with each [term]. + +While a `Taxonomy` object is a [map], an ordered taxonomy is a [slice], where each element is an object that contains the term and a slice of its [weighted pages]. + +{{% include "methods/taxonomy/_common/get-a-taxonomy-object.md" %}} + +## Get the ordered taxonomy + +Now that we have captured the “genres” Taxonomy object, let’s get the ordered taxonomy sorted by the number of pages associated with each term: + +```go-html-template +{{ $taxonomyObject.ByCount }} +``` + +To reverse the sort order: + +```go-html-template +{{ $taxonomyObject.ByCount.Reverse }} +``` + +To inspect the data structure: + +```go-html-template +
    {{ jsonify (dict "indent" "  ") $taxonomyObject.ByCount }}
    +``` + +{{% include "methods/taxonomy/_common/ordered-taxonomy-element-methods.md" %}} + +## Example + +With this template: + +```go-html-template +{{ range $taxonomyObject.ByCount }} +

    {{ .Page.LinkTitle }} ({{ .Count }})

    + +{{ end }} +``` + +Hugo renders: + +```html +

    suspense (3)

    + +

    romance (2)

    + +``` + +[ordered taxonomy]: /getting-started/glossary/#ordered-taxonomy +[term]: /getting-started/glossary/#term +[map]: /getting-started/glossary/#map +[slice]: /getting-started/glossary/#slice +[term]: /getting-started/glossary/#term +[weighted pages]: /getting-started/glossary/#weighted-page diff --git a/content/en/methods/taxonomy/Count.md b/content/en/methods/taxonomy/Count.md new file mode 100644 index 00000000000..50f705ec9b1 --- /dev/null +++ b/content/en/methods/taxonomy/Count.md @@ -0,0 +1,26 @@ +--- +title: Count +description: Returns the number of number of weighted pages to which the given term has been assigned. +categories: [] +keywords: [] +action: + related: [] + returnType: int + signatures: [TAXONOMY.Count TERM] +toc: true +--- + +The `Count` method on a `Taxonomy` object returns the number of number of [weighted pages] to which the given [term] has been assigned. + +{{% include "methods/taxonomy/_common/get-a-taxonomy-object.md" %}} + +## Count the weighted pages + +Now that we have captured the "genres" `Taxonomy` object, let's count the number of weighted pages to which the "suspense" term has been assigned: + +```go-html-template +{{ $taxonomyObject.Count "suspense" }} → 3 +``` + +[weighted pages]: /getting-started/glossary/#weighted-page +[term]: /getting-started/glossary/#term diff --git a/content/en/methods/taxonomy/Get.md b/content/en/methods/taxonomy/Get.md new file mode 100644 index 00000000000..3bac86f08a3 --- /dev/null +++ b/content/en/methods/taxonomy/Get.md @@ -0,0 +1,72 @@ +--- +title: Get +description: Returns a slice of weighted pages to which the given term has been assigned. +categories: [] +keywords: [] +action: + related: [] + returnType: page.WeightedPages + signatures: [TAXONOMY.Get TERM] +toc: true +--- + +The `Get` method on a `Taxonomy` object returns a slice of [weighted pages] to which the given [term] has been assigned. + +{{% include "methods/taxonomy/_common/get-a-taxonomy-object.md" %}} + +## Get the weighted pages + +Now that we have captured the "genres" `Taxonomy` object, let's get the weighted pages to which the "suspense" term has been assigned: + +```go-html-template +{{ $weightedPages := $taxonomyObject.Get "suspense" }} +``` + +The above is equivalent to: + +```go-html-template +{{ $weightedPages := $taxonomyObject.suspense }} +``` + +But, if the term is not a valid [identifier], you cannot use the [chaining] syntax. For example, this will throw an error because the identifier contains a hyphen: + +```go-html-template +{{ $weightedPages := $taxonomyObject.my-genre }} +``` + +You could also use the [`index`] function, but the syntax is more verbose: + +```go-html-template +{{ $weightedPages := index $taxonomyObject "my-genre" }} +``` + +To inspect the data structure: + +```go-html-template +
    {{ jsonify (dict "indent" "  ") $weightedPages }}
    +``` + +## Example + +With this template: + +```go-html-template +{{ $weightedPages := $taxonomyObject.Get "suspense" }} +{{ range $weightedPages }} +

    {{ .LinkTitle }}

    +{{ end }} +``` + +Hugo renders: + +```html +

    Jamaica inn

    +

    Death on the nile

    +

    And then there were none

    +``` + +[chaining]: /getting-started/glossary/#chain +[`index`]: /functions/collections/indexfunction +[identifier]: /getting-started/glossary/#identifier +[term]: /getting-started/glossary/#term +[weighted pages]: /getting-started/glossary/#weighted-page diff --git a/content/en/methods/taxonomy/_common/_index.md b/content/en/methods/taxonomy/_common/_index.md new file mode 100644 index 00000000000..47d5812fba5 --- /dev/null +++ b/content/en/methods/taxonomy/_common/_index.md @@ -0,0 +1,13 @@ +--- +cascade: + _build: + list: never + publishResources: false + render: never +--- + + diff --git a/content/en/methods/taxonomy/_common/get-a-taxonomy-object.md b/content/en/methods/taxonomy/_common/get-a-taxonomy-object.md new file mode 100644 index 00000000000..d9bd1436471 --- /dev/null +++ b/content/en/methods/taxonomy/_common/get-a-taxonomy-object.md @@ -0,0 +1,68 @@ +--- +# Do not remove front matter. +--- + +Before we can use a `Taxonomy` method, we need to capture a `Taxonomy` object. + +## Capture a taxonomy object + +Consider this site configuration: + +{{< code-toggle file=hugo >}} +[taxonomies] +genre = 'genres' +author = 'authors' +{{< /code-toggle >}} + +And this content structure: + +```text +content/ +├── books/ +│ ├── and-then-there-were-none.md --> genres: suspense +│ ├── death-on-the-nile.md --> genres: suspense +│ └── jamaica-inn.md --> genres: suspense, romance +│ └── pride-and-prejudice.md --> genres: romance +└── _index.md +``` + +To capture the "genres" taxonomy object from within any template, use the [`Taxonomies`] method on a `Site` object. + +```go-html-template +{{ $taxonomyObject := .Site.Taxonomies.genres }} +``` + +To capture the "genres" taxonomy object when rendering its page with a taxonomy template, use the [`Terms`] method on the page's [`Data`] object: + +{{< code file="layouts/_default/taxonomy.html" lang=go-html-template >}} +{{ $taxonomyObject := .Data.Terms }} +{{< /code >}} + +To inspect the data structure: + +```go-html-template +
    {{ jsonify (dict "indent" "  ") $taxonomyObject }}
    +``` + +Although the [`Alphabetical`] and [`ByCount`] methods provide a better data structure for ranging through the taxonomy, you can render the weighted pages by term directly from the `Taxonomy` object: + +```go-html-template +{{ range $term, $weightedPages := $taxonomyObject }} +

    {{ .Page.LinkTitle }}

    + +{{ end }} +``` + +In the example above, the first anchor element is a link to the term page. + + +[`Alphabetical`]: /methods/taxonomy/alphabetical +[`ByCount`]: /methods/taxonomy/bycount + +[`data`]: /methods/page/data +[`terms`]: /methods/page/data/#in-a-taxonomy-template +[`taxonomies`]: /methods/site/taxonomies diff --git a/content/en/methods/taxonomy/_common/ordered-taxonomy-element-methods.md b/content/en/methods/taxonomy/_common/ordered-taxonomy-element-methods.md new file mode 100644 index 00000000000..57c9e8e2937 --- /dev/null +++ b/content/en/methods/taxonomy/_common/ordered-taxonomy-element-methods.md @@ -0,0 +1,25 @@ +--- +# Do not remove front matter. +--- + +An ordered taxonomy is a slice, where each element is an object that contains the term and a slice of its weighted pages. + +Each element of the slice provides these methods: + +Count +: (`int`) Returns the number of pages to which the term is assigned. + +Page +: (`hugolib.pageState`) Returns the term's `Page` object, useful for linking to the term page. + +Pages +: (`page.Pages`) Returns a `Pages` object containing the `Page` objects to which the term is assigned, sorted by [taxonomic weight]. To sort or group, use any of the [methods] available to the `Pages` object. For example, sort by the last modification date. + +Term +: (`string`) Returns the term name. + +WeightedPages +: (`page.WeightedPages`) Returns a slice of weighted pages to which the term is assigned, sorted by [taxonomic weight]. The `Pages` method above is more flexible, allowing you to sort and group. + +[methods]: /methods/pages +[taxonomic weight]: /getting-started/glossary/#taxonomic-weight diff --git a/content/en/methods/taxonomy/_index.md b/content/en/methods/taxonomy/_index.md new file mode 100644 index 00000000000..e7eb57834c5 --- /dev/null +++ b/content/en/methods/taxonomy/_index.md @@ -0,0 +1,12 @@ +--- +title: Taxonomy methods +linkTitle: Taxonomy +description: Use these methods with Taxonomy objects. +keywords: [] +menu: + docs: + identifier: + parent: methods +--- + +Use these methods with Taxonomy objects. diff --git a/content/en/methods/time/Add.md b/content/en/methods/time/Add.md new file mode 100644 index 00000000000..8fd755244a2 --- /dev/null +++ b/content/en/methods/time/Add.md @@ -0,0 +1,23 @@ +--- +title: Add +description: Returns the given time plus the given duration. +categories: [] +keywords: [] +action: + related: + - functions/time/AsTime + - functions/time/Duration + - functions/time/ParseDuration + returnType: time.Time + signatures: [TIME.Add DURATION] +--- + +```go-html-template +{{ $t := time.AsTime "2023-01-27T23:44:58-08:00" }} + +{{ $d1 = time.ParseDuration "3h20m10s" }} +{{ $d2 = time.ParseDuration "-3h20m10s" }} + +{{ $t.Add $d1 }} → 2023-01-28 03:05:08 -0800 PST +{{ $t.Add $d2 }} → 2023-01-27 20:24:48 -0800 PST +``` diff --git a/content/en/functions/AddDate.md b/content/en/methods/time/AddDate.md similarity index 88% rename from content/en/functions/AddDate.md rename to content/en/methods/time/AddDate.md index 7f5b39b160a..8537d6e25fc 100644 --- a/content/en/functions/AddDate.md +++ b/content/en/methods/time/AddDate.md @@ -1,16 +1,14 @@ --- -title: .AddDate +title: AddDate description: Returns the time corresponding to adding the given number of years, months, and days to the given time.Time value. -categories: [functions] +categories: [] keywords: [] -menu: - docs: - parent: functions -function: +action: aliases: [] + related: [] returnType: time.Time - signatures: [.AddDate YEARS MONTHS DAYS] -relatedFunctions: [] + signatures: [TIME.AddDate YEARS MONTHS DAYS] +aliases: [/functions/adddate] --- ```go-html-template diff --git a/content/en/methods/time/After.md b/content/en/methods/time/After.md new file mode 100644 index 00000000000..0aeeb38d82e --- /dev/null +++ b/content/en/methods/time/After.md @@ -0,0 +1,20 @@ +--- +title: After +description: Reports whether TIME1 is after TIME2. +categories: [] +keywords: [] +action: + related: + - methods/time/Before + - methods/time/After + - functions/time/AsTime + returnType: bool + signatures: [TIME1.After TIME2] +--- + +```go-html-template +{{ $t1 := time.AsTime "2023-01-01T17:00:00-08:00" }} +{{ $t2 := time.AsTime "2010-01-01T17:00:00-08:00" }} + +{{ $t1.After $t2 }} → true +``` diff --git a/content/en/methods/time/Before.md b/content/en/methods/time/Before.md new file mode 100644 index 00000000000..c3d582860ca --- /dev/null +++ b/content/en/methods/time/Before.md @@ -0,0 +1,19 @@ +--- +title: Before +description: Reports whether TIME1 is before TIME2. +categories: [] +keywords: [] +action: + related: + - methods/time/After + - methods/time/Equal + - functions/time/AsTime + returnType: bool + signatures: [TIME1.Before TIME2] +--- + +```go-html-template +{{ $t1 := time.AsTime "2023-01-01T17:00:00-08:00" }} +{{ $t2 := time.AsTime "2030-01-01T17:00:00-08:00" }} + +{{ $t1.Before $t2 }} → true diff --git a/content/en/methods/time/Day.md b/content/en/methods/time/Day.md new file mode 100644 index 00000000000..1173b8489aa --- /dev/null +++ b/content/en/methods/time/Day.md @@ -0,0 +1,21 @@ +--- +title: Day +description: Returns the day of the month of the given time.Time value. +categories: [] +keywords: [] +action: + related: + - methods/time/Year + - methods/time/Month + - methods/time/Hour + - methods/time/Minute + - methods/time/Second + - functions/time/AsTime + returnType: int + signatures: [TIME.Day] +--- + +```go-html-template +{{ $t := time.AsTime "2023-01-27T23:44:58-08:00" }} +{{ $t.Day }} → 27 +``` diff --git a/content/en/methods/time/Equal.md b/content/en/methods/time/Equal.md new file mode 100644 index 00000000000..4d45a3ada03 --- /dev/null +++ b/content/en/methods/time/Equal.md @@ -0,0 +1,20 @@ +--- +title: Equal +description: Reports whether TIME1 is equal to TIME2. +categories: [] +keywords: [] +action: + related: + - methods/time/After + - methods/time/Before + - functions/time/AsTime + returnType: bool + signatures: [TIME1.Equal TIME2] +--- + +```go-html-template +{{ $t1 := time.AsTime "2023-01-01T17:00:00-08:00" }} +{{ $t2 := time.AsTime "2023-01-01T20:00:00-05:00" }} + +{{ $t1.Equal $t2 }} → true +``` diff --git a/content/en/functions/Format.md b/content/en/methods/time/Format.md similarity index 87% rename from content/en/functions/Format.md rename to content/en/methods/time/Format.md index e679cb2c45b..0137144f9af 100644 --- a/content/en/functions/Format.md +++ b/content/en/methods/time/Format.md @@ -1,17 +1,18 @@ --- -title: .Format +title: Format description: Returns a formatted time.Time value. -categories: [functions] +categories: [] keywords: [] -menu: - docs: - parent: functions -function: +action: aliases: [] + related: + - functions/time/AsTime + - methods/time/UTC + - methods/time/Local returnType: string - signatures: [.Format LAYOUT] -relatedFunctions: [] + signatures: [TIME.Format LAYOUT] toc: true +aliases: [/methods/time/format] --- ```go-template @@ -41,13 +42,13 @@ Use the `.Format` method with any `time.Time` value, including the four predefin ## Layout string -{{% readfile file="/functions/_common/time-layout-string.md" %}} +{{% include "functions/_common/time-layout-string.md" %}} ## Examples Given this front matter: -{{< code-toggle fm=true copy=false >}} +{{< code-toggle fm=true >}} title = "About time" date = 2023-01-27T23:44:58-08:00 {{< /code-toggle >}} diff --git a/content/en/methods/time/Hour.md b/content/en/methods/time/Hour.md new file mode 100644 index 00000000000..58ed0026058 --- /dev/null +++ b/content/en/methods/time/Hour.md @@ -0,0 +1,21 @@ +--- +title: Hour +description: Returns the hour within the day of the given time.Time value, in the range [0, 23]. +categories: [] +keywords: [] +action: + related: + - methods/time/Year + - methods/time/Month + - methods/time/Day + - methods/time/Minute + - methods/time/Second + - functions/time/AsTime + returnType: int + signatures: [TIME.Hour] +--- + +```go-html-template +{{ $t := time.AsTime "2023-01-27T23:44:58-08:00" }} +{{ $t.Hour }} → 23 +``` diff --git a/content/en/methods/time/IsDST.md b/content/en/methods/time/IsDST.md new file mode 100644 index 00000000000..df2b84cae89 --- /dev/null +++ b/content/en/methods/time/IsDST.md @@ -0,0 +1,19 @@ +--- +title: IsDST +description: Reports whether the given time.Time value is in Daylight Savings Time. +categories: [] +keywords: [] +action: + related: + - functions/time/AsTime + returnType: bool + signatures: [TIME.IsDST] +--- + +```go-html-template +{{ $t1 := time.AsTime "2023-01-01T00:00:00-08:00" }} +{{ $t2 := time.AsTime "2023-07-01T00:00:00-07:00" }} + +{{ $t1.IsDST }} → false +{{ $t2.IsDST }} → true +``` diff --git a/content/en/methods/time/IsZero.md b/content/en/methods/time/IsZero.md new file mode 100644 index 00000000000..2026f3b2eef --- /dev/null +++ b/content/en/methods/time/IsZero.md @@ -0,0 +1,19 @@ +--- +title: IsZero +description: Reports whether the given time.Time value represents the zero time instant, January 1, year 1, 00:00:00 UTC. +categories: [] +keywords: [] +action: + related: + - functions/time/AsTime + returnType: bool + signatures: [TIME.IsZero] +--- + +````go-html-template +{{ $t1 := time.AsTime "2023-01-01T00:00:00-08:00" }} +{{ $t2 := time.AsTime "0001-01-01T00:00:00-00:00" }} + +{{ $t1.IsZero }} → false +{{ $t2.IsZero }} → true +``` diff --git a/content/en/methods/time/Local.md b/content/en/methods/time/Local.md new file mode 100644 index 00000000000..bd40e3a4420 --- /dev/null +++ b/content/en/methods/time/Local.md @@ -0,0 +1,17 @@ +--- +title: Local +description: Returns the given time.Time value with the location set to local time. +categories: [] +keywords: [] +action: + related: + - methods/time/UTC + - functions/time/AsTime + returnType: time.Time + signatures: [TIME.Local] +--- + +```go-html-template +{{ $t := time.AsTime "2023-01-28T07:44:58+00:00" }} +{{ $t.Local }} → 2023-01-27 23:44:58 -0800 PST +``` diff --git a/content/en/methods/time/Minute.md b/content/en/methods/time/Minute.md new file mode 100644 index 00000000000..d482fab5db4 --- /dev/null +++ b/content/en/methods/time/Minute.md @@ -0,0 +1,21 @@ +--- +title: Minute +description: Returns the minute offset within the hour of the given time.Time value, in the range [0, 59]. +categories: [] +keywords: [] +action: + related: + - methods/time/Year + - methods/time/Month + - methods/time/Day + - methods/time/Hour + - methods/time/Second + - functions/time/AsTime + returnType: int + signatures: [TIME.Minute] +--- + +```go-html-template +{{ $t := time.AsTime "2023-01-27T23:44:58-08:00" }} +{{ $t.Minute }} → 44 +``` diff --git a/content/en/methods/time/Month.md b/content/en/methods/time/Month.md new file mode 100644 index 00000000000..08815afbd3a --- /dev/null +++ b/content/en/methods/time/Month.md @@ -0,0 +1,30 @@ +--- +title: Month +description: Returns the month of the year of the given time.Time value. +categories: [] +keywords: [] +action: + related: + - methods/time/Year + - methods/time/Day + - methods/time/Hour + - methods/time/Minute + - methods/time/Second + - functions/time/AsTime + returnType: time.Month + signatures: [TIME.Month] +--- + +To convert the time.Month value to a string: + +```go-html-template +{{ $t := time.AsTime "2023-01-27T23:44:58-08:00" }} +{{ $t.Month.String }} → January +``` + +To convert the time.Month value to an integer. + +```go-html-template +{{ $t := time.AsTime "2023-01-27T23:44:58-08:00" }} +{{ $t.Month | int }} → 1 +``` diff --git a/content/en/methods/time/Nanosecond.md b/content/en/methods/time/Nanosecond.md new file mode 100644 index 00000000000..60614313967 --- /dev/null +++ b/content/en/methods/time/Nanosecond.md @@ -0,0 +1,16 @@ +--- +title: Nanosecond +description: Returns the nanosecond offset within the second of the given time.Time value, in the range [0, 999999999]. +categories: [] +keywords: [] +action: + related: + - functions/time/AsTime + returnType: int + signatures: [TIME.Nanosecond] +--- + +```go-html-template +{{ $t := time.AsTime "2023-01-27T23:44:58-08:00" }} +{{ $t.Nanosecond }} → 0 +``` diff --git a/content/en/methods/time/Second.md b/content/en/methods/time/Second.md new file mode 100644 index 00000000000..e326c64bce3 --- /dev/null +++ b/content/en/methods/time/Second.md @@ -0,0 +1,21 @@ +--- +title: Second +description: Returns the second offset within the minute of the given time.Time value, in the range [0, 59]. +categories: [] +keywords: [] +action: + related: + - methods/time/Year + - methods/time/Month + - methods/time/Day + - methods/time/Hour + - methods/time/Minute + - functions/time/AsTime + returnType: int + signatures: [TIME.Second] +--- + +```go-html-template +{{ $t := time.AsTime "2023-01-27T23:44:58-08:00" }} +{{ $t.Second }} → 58 +``` diff --git a/content/en/methods/time/Sub.md b/content/en/methods/time/Sub.md new file mode 100644 index 00000000000..9678365eb13 --- /dev/null +++ b/content/en/methods/time/Sub.md @@ -0,0 +1,18 @@ +--- +title: Sub +description: Returns the duration computed by subtracting TIME2 from TIME1. +categories: [] +keywords: [] +action: + related: + - functions/time/AsTime + returnType: time.Duration + signatures: [TIME1.Sub TIME2] +--- + +```go-html-template +{{ $t1 := time.AsTime "2023-01-27T23:44:58-08:00" }} +{{ $t2 := time.AsTime "2023-01-26T22:34:38-08:00" }} + +{{ $t1.Sub $t2 }} → 25h10m20s +``` diff --git a/content/en/methods/time/UTC.md b/content/en/methods/time/UTC.md new file mode 100644 index 00000000000..6fd7b526dd1 --- /dev/null +++ b/content/en/methods/time/UTC.md @@ -0,0 +1,16 @@ +--- +title: UTC +description: Returns the given time.Time value with the location set to UTC. +categories: [] +keywords: [] +action: + related: + - methods/time/Local + - functions/time/AsTime + returnType: time.Time + signatures: [TIME.UTC] +--- + +```go-html-template +{{ $t := time.AsTime "2023-01-27T23:44:58-08:00" }} +{{ $t.UTC }} → 2023-01-28 07:44:58 +0000 UTC diff --git a/content/en/methods/time/Unix.md b/content/en/methods/time/Unix.md new file mode 100644 index 00000000000..fcfc661fe85 --- /dev/null +++ b/content/en/methods/time/Unix.md @@ -0,0 +1,21 @@ +--- +title: Unix +description: Returns the given time.Time value expressed as the number of seconds elapsed since January 1, 1970 UTC. +categories: [] +action: + related: + - methods/time/UnixMilli + - methods/time/UnixMicro + - methods/time/UnixNano + - functions/time/AsTime + returnType: int64 + signatures: [TIME.Unix] +aliases: [/functions/unix] +--- + +See [Unix epoch](https://en.wikipedia.org/wiki/Unix_time). + +```go-html-template +{{ $t := time.AsTime "2023-01-27T23:44:58-08:00" }} +{{ $t.Unix }} → 1674891898 +``` diff --git a/content/en/methods/time/UnixMicro.md b/content/en/methods/time/UnixMicro.md new file mode 100644 index 00000000000..150497cd357 --- /dev/null +++ b/content/en/methods/time/UnixMicro.md @@ -0,0 +1,21 @@ +--- +title: UnixMicro +description: Returns the given time.Time value expressed as the number of microseconds elapsed since January 1, 1970 UTC. +categories: [] +keywords: [] +action: + related: + - methods/time/Unix + - methods/time/UnixMilli + - methods/time/UnixNano + - functions/time/AsTime + returnType: int64 + signatures: [TIME.UnixMicro] +--- + +See [Unix epoch](https://en.wikipedia.org/wiki/Unix_time). + +```go-html-template +{{ $t := time.AsTime "2023-01-27T23:44:58-08:00" }} +{{ $t.UnixMicro }} → 1674891898000000 +``` diff --git a/content/en/methods/time/UnixMilli.md b/content/en/methods/time/UnixMilli.md new file mode 100644 index 00000000000..e5e90ba25f9 --- /dev/null +++ b/content/en/methods/time/UnixMilli.md @@ -0,0 +1,21 @@ +--- +title: UnixMilli +description: Returns the given time.Time value expressed as the number of milliseconds elapsed since January 1, 1970 UTC. +categories: [] +keywords: [] +action: + related: + - methods/time/Unix + - methods/time/UnixMicro + - methods/time/UnixNano + - functions/time/AsTime + returnType: int64 + signatures: [TIME.UnixMilli] +--- + +See [Unix epoch](https://en.wikipedia.org/wiki/Unix_time). + +```go-html-template +{{ $t := time.AsTime "2023-01-27T23:44:58-08:00" }} +{{ $t.UnixMilli }} → 1674891898000 +``` diff --git a/content/en/methods/time/UnixNano.md b/content/en/methods/time/UnixNano.md new file mode 100644 index 00000000000..63db320a350 --- /dev/null +++ b/content/en/methods/time/UnixNano.md @@ -0,0 +1,21 @@ +--- +title: UnixNano +description: Returns the given time.Time value expressed as the number of nanoseconds elapsed since January 1, 1970 UTC. +categories: [] +keywords: [] +action: + related: + - methods/time/Unix + - methods/time/UnixMilli + - methods/time/UnixMicro + - functions/time/AsTime + returnType: int64 + signatures: [TIME.UnixNano] +--- + +See [Unix epoch](https://en.wikipedia.org/wiki/Unix_time). + +```go-html-template +{{ $t := time.AsTime "2023-01-27T23:44:58-08:00" }} +{{ $t.UnixNano }} → 1674891898000000000 +``` diff --git a/content/en/methods/time/Weekday.md b/content/en/methods/time/Weekday.md new file mode 100644 index 00000000000..9e2349c8060 --- /dev/null +++ b/content/en/methods/time/Weekday.md @@ -0,0 +1,24 @@ +--- +title: Weekday +description: Returns the day of the week of the given time.Time value. +categories: [] +keywords: [] +action: + related: + - functions/time/AsTime + returnType: time.Weekday + signatures: [TIME.Weekday] +--- + +To convert the time.Weekday value to a string: + +```go-html-template +{{ $t := time.AsTime "2023-01-27T23:44:58-08:00" }} +{{ $t.Weekday.String }} → Friday +``` + +To convert the time.Weekday value to an integer. + +```go-html-template +{{ $t := time.AsTime "2023-01-27T23:44:58-08:00" }} +{{ $t.Weekday | int }} → 5 diff --git a/content/en/methods/time/Year.md b/content/en/methods/time/Year.md new file mode 100644 index 00000000000..b046896f456 --- /dev/null +++ b/content/en/methods/time/Year.md @@ -0,0 +1,21 @@ +--- +title: Year +description: Returns the year of the given time.Time value. +categories: [] +keywords: [] +action: + related: + - methods/time/Month + - methods/time/Day + - methods/time/Hour + - methods/time/Minute + - methods/time/Second + - functions/time/AsTime + returnType: int + signatures: [TIME.Year] +--- + +```go-html-template +{{ $t := time.AsTime "2023-01-27T23:44:58-08:00" }} +{{ $t.Year }} → 2023 +``` diff --git a/content/en/methods/time/YearDay.md b/content/en/methods/time/YearDay.md new file mode 100644 index 00000000000..40d7d6aabc5 --- /dev/null +++ b/content/en/methods/time/YearDay.md @@ -0,0 +1,15 @@ +--- +title: YearDay +description: Returns the day of the year of the given time.Time value, in the range [1, 365] for non-leap years, and [1,366] in leap years. +categories: [] +keywords: [] +action: + related: [] + returnType: int + signatures: [TIME.YearDay] +--- + +```go-html-template +{{ $t := time.AsTime "2023-01-27T23:44:58-08:00" }} +{{ $t.YearDay }} → 27 +``` diff --git a/content/en/methods/time/_index.md b/content/en/methods/time/_index.md new file mode 100644 index 00000000000..81d4690e0e4 --- /dev/null +++ b/content/en/methods/time/_index.md @@ -0,0 +1,13 @@ +--- +title: Time methods +linkTitle: Time +description: Use these methods with time.Time values. +categories: [] +keywords: [] +menu: + docs: + identifier: time-methods + parent: methods +--- + +Use these methods with time.Time values. diff --git a/content/en/myshowcase/bio.md b/content/en/myshowcase/bio.md index 7d1b30895ba..6b8f7e1a913 100644 --- a/content/en/myshowcase/bio.md +++ b/content/en/myshowcase/bio.md @@ -3,6 +3,5 @@ Add some **general info** about Myshowcase here. The site is built by: -* [Person 1](https://example.com) -* [Person 1](https://example.com) - +* [Person 1](https://example.org) +* [Person 1](https://example.org) diff --git a/content/en/showcase/1password-support/bio.md b/content/en/showcase/1password-support/bio.md index 9187908d987..3e15adc9f7c 100644 --- a/content/en/showcase/1password-support/bio.md +++ b/content/en/showcase/1password-support/bio.md @@ -1,5 +1,4 @@ **1Password** is a password manager that keeps you safe online. It protects your secure information behind the one password only you know. - The [1Password Support](https://support.1password.com/) website was built from scratch with **Hugo** and enhanced with **React** and **Elasticsearch** to give us the best of both worlds: The simplicity and performance of a static site, with the richness of a hosted web app. diff --git a/content/en/showcase/1password-support/index.md b/content/en/showcase/1password-support/index.md index 2bcbff3fd17..ed44053c817 100644 --- a/content/en/showcase/1password-support/index.md +++ b/content/en/showcase/1password-support/index.md @@ -34,6 +34,6 @@ Finding a tool that will make your customers, writers, designers, _and_ DevOps t * [1Password Support](https://support.1password.com) uses Hugo with a custom theme. It shares styles and some template code with [1Password.com](https://1password.com), which we also moved to Hugo in 2016. * Code and articles live in a private GitHub repository, which is deployed to a static content server using Git hooks. * Writers build and preview the site on their computers and contribute content using pull requests. - * We use Hugo's [multilingual support](/content-management/multilingual/) to build the site in English, Spanish, French, Italian, German, and Russian. With the help of Hugo, 1Password Support became our very first site in multiple languages. +* We use Hugo's [multilingual support](/content-management/multilingual/) to build the site in English, Spanish, French, Italian, German, and Russian. With the help of Hugo, 1Password Support became our very first site in multiple languages. * Our [contact form](https://support.1password.com/contact) is a single-page React app. We were able to integrate it with Hugo seamlessly thanks to its support for static files. * The one part of the support site which is not static is our search engine, which we developed with Elasticsearch and host on AWS. diff --git a/content/en/showcase/alora-labs/bio.md b/content/en/showcase/alora-labs/bio.md index d304cf191ab..f14a90b753f 100644 --- a/content/en/showcase/alora-labs/bio.md +++ b/content/en/showcase/alora-labs/bio.md @@ -1,3 +1,3 @@ -**Alora Labs** is a product development consultancy headquartered in Toronto, Canada. +**Alora Labs** is a product development consultancy headquartered in Toronto, Canada. We help companies build software and IoT products and were recently recognized as one of the [**top IoT development firms**](https://aloralabs.com/insights/alora-labs-receives-clutch-2021-top-iot-agency-award) in Toronto. diff --git a/content/en/showcase/alora-labs/index.md b/content/en/showcase/alora-labs/index.md index 5e6e1813137..79a3c5416e6 100644 --- a/content/en/showcase/alora-labs/index.md +++ b/content/en/showcase/alora-labs/index.md @@ -9,10 +9,10 @@ aliases: [/showcase/aloralabs/] At Alora Labs we always have an eye open for new tools and technology that we can utilize to the benefit of our customers or internal projects like our website. -The previous iteration of our site was built with Jekyll, which served us well at first. However as time went on, we became frustrated with the number of dependencies we had to rely on, that would often break at the most inconvenient times. +The previous iteration of our site was built with Jekyll, which served us well at first. However as time went on, we became frustrated with the number of dependencies we had to rely on, that would often break at the most inconvenient times. Hugo was a breath of fresh air in this regard, a single binary that works equally well on Windows as it did on macOS or Linux. We no longer need additional tools for image optimization, Sass compilation or JavaScript bundling. Everything just works, and with a substantial performance boost too. -Hugo has become a favorite tool in the toolbelt and the foundation for many client projects. We couldn't be happier with the switch and we are optimistic about recommending Hugo for many years to come. +Hugo has become a favorite tool in the toolbelt and the foundation for many client projects. We couldn't be happier with the switch and we are optimistic about recommending Hugo for many years to come. -Thank you to the vibrant community and talented development team for all the hard work in making Hugo a success. As excellent as Hugo is now, we cannot wait to see what the release notes have in store for us next. \ No newline at end of file +Thank you to the vibrant community and talented development team for all the hard work in making Hugo a success. As excellent as Hugo is now, we cannot wait to see what the release notes have in store for us next. diff --git a/content/en/showcase/ampio-help/index.md b/content/en/showcase/ampio-help/index.md index 3d21192b814..9b685038537 100644 --- a/content/en/showcase/ampio-help/index.md +++ b/content/en/showcase/ampio-help/index.md @@ -47,7 +47,7 @@ We even implemented a simple REST API generated by Hugo! We have yet to find som When talking about Hugo, we cannot forget about the speed. At the beginning we were not considering it a killer feature, but as our document base grew bigger, we appreciated it more and more. Dry-runs are not so common---most of the time we are working on one of the documents with cache already built during one of the previous Hugo runs. In such a scenario, Hugo rebuilds the site in about a second and we consider it a very good result. -``` +```text | EN | PL -------------------+-----+------ Pages | 483 | 486 diff --git a/content/en/showcase/bypasscensorship/bio.md b/content/en/showcase/bypasscensorship/bio.md index 6563e13ca6a..0a847df1ec5 100644 --- a/content/en/showcase/bypasscensorship/bio.md +++ b/content/en/showcase/bypasscensorship/bio.md @@ -3,4 +3,4 @@ Bypass Censorship find and promote tools that provide Internet access to everyon The site is built by: * [Leyla Avsar](https://www.leylaavsar.com/) (designer) -* [Fredrik Jonsson](https://xdeb.net/) (dev) \ No newline at end of file +* [Fredrik Jonsson](https://xdeb.net/) (dev) diff --git a/content/en/showcase/bypasscensorship/index.md b/content/en/showcase/bypasscensorship/index.md index a266797ea04..8cbda9aa65d 100644 --- a/content/en/showcase/bypasscensorship/index.md +++ b/content/en/showcase/bypasscensorship/index.md @@ -21,4 +21,4 @@ It's a simply site, basically one page in seven languages. I had no problems get Thanks to the design by [Leyla Avsar](https://www.leylaavsar.com/) the site also looks good. I used the [Hugo Zen theme](https://github.com/frjo/hugo-theme-zen) with a few custom templates and the needed CSS. -The editors can maintain content via [Forestry.io CMS](https://forestry.io/) or directly via Git. Forestry does unfortunately not have multilingual support. All the language versions are in one pile making it harder to find the right file to edit, but it works. \ No newline at end of file +The editors can maintain content via [Forestry.io CMS](https://forestry.io/) or directly via Git. Forestry does unfortunately not have multilingual support. All the language versions are in one pile making it harder to find the right file to edit, but it works. diff --git a/content/en/showcase/digitalgov/index.md b/content/en/showcase/digitalgov/index.md index c0a927edb83..4376bbf03ce 100644 --- a/content/en/showcase/digitalgov/index.md +++ b/content/en/showcase/digitalgov/index.md @@ -12,11 +12,11 @@ Through collaboration in our communities of practice, Digital.gov is a window in Digital.gov is built using the [U.S. Web Design System](https://designsystem.digital.gov/) (USWDS) and have followed the [design principles](https://designsystem.digital.gov/maturity-model/) in building out our new site: -- **Start with real user needs** — We used human-centered design methods to inform our product decisions (like qualitative user research), and gathered feedback from real users. We also continually test our assumptions with small experiments. -- **Earn trust** —We recognize that trust has to be earned every time. We are including all [required links and content](https://digital.gov/resources/required-web-content-and-links/) on our site, clearly identifying as a government site, building with modern best practices, and using HTTPS. -- **Embrace accessibility** — [Accessibility](https://digital.gov/resources/intro-accessibility/) affects everybody, and we built it into every decision. We’re continually working to conform to Section 508 requirements, use user experience best practices, and support a wide range of devices. -- **Promote continuity** — We started from shared solutions like USWDS and [Federalist](https://federalist.18f.gov/). We designed our site to clearly identify as a government site by including USWDS’s .gov banner, common colors and patterns, and built with modern best practices. -- **Listen** — We actively collect user feedback and web metrics. We use the [Digital Analytics Program](https://digital.gov/services/dap/) (DAP) and analyze the data to discover actionable insights. We make small, incremental changes to continuously improve our website by listening to readers and learning from what we hear. +- **Start with real user needs** — We used human-centered design methods to inform our product decisions (like qualitative user research), and gathered feedback from real users. We also continually test our assumptions with small experiments. +- **Earn trust** —We recognize that trust has to be earned every time. We are including all [required links and content](https://digital.gov/resources/required-web-content-and-links/) on our site, clearly identifying as a government site, building with modern best practices, and using HTTPS. +- **Embrace accessibility** — [Accessibility](https://digital.gov/resources/intro-accessibility/) affects everybody, and we built it into every decision. We’re continually working to conform to Section 508 requirements, use user experience best practices, and support a wide range of devices. +- **Promote continuity** — We started from shared solutions like USWDS and [Federalist](https://federalist.18f.gov/). We designed our site to clearly identify as a government site by including USWDS’s .gov banner, common colors and patterns, and built with modern best practices. +- **Listen** — We actively collect user feedback and web metrics. We use the [Digital Analytics Program](https://digital.gov/services/dap/) (DAP) and analyze the data to discover actionable insights. We make small, incremental changes to continuously improve our website by listening to readers and learning from what we hear. _More on the [USWDS maturity model »](https://designsystem.digital.gov/maturity-model/)_ @@ -34,7 +34,7 @@ At the moment, it takes around `32 seconds` to build close to `~10,000` pages! Take a look: -```bash +```text | EN -------------------+------- @@ -48,19 +48,17 @@ Take a look: Cleaned | 0 Built in 32.427 seconds - ``` In addition to Hugo, we are proudly using a number of other tools and services, all built by government are free to use: -- [Federalist](https://federalist.18f.gov/) -- [Search.gov](https://www.search.gov/) — A free, hosted search platform for federal websites. -- [Cloud.gov](https://www.cloud.gov/) — helps teams build, run, and authorize cloud-ready or legacy government systems quickly and cheaply. -- [Federal CrowdSource Mobile Testing Program](https://digital.gov/services/mobile-application-testing-program/) — Free mobile compatibility testing by feds, for feds. -- [Digital Analytics Program](https://digital.gov/services/dap/) (DAP) — A free analytics tool for measuring digital services in the federal government -- [Section508.gov](https://www.section508.gov/) and [PlainLanguage.gov](https://www.plainlanguage.gov/) resources -- [API.data.gov](https://api.data.gov/) — a free API management service for federal agencies -- [U.S. Digital Registry](https://digital.gov/services/u-s-digital-registry/) — A resource for confirming the official status of government social media accounts, mobile apps, and mobile websites. - +- [Federalist](https://federalist.18f.gov/) +- [Search.gov](https://www.search.gov/) — A free, hosted search platform for federal websites. +- [Cloud.gov](https://www.cloud.gov/) — helps teams build, run, and authorize cloud-ready or legacy government systems quickly and cheaply. +- [Federal CrowdSource Mobile Testing Program](https://digital.gov/services/mobile-application-testing-program/) — Free mobile compatibility testing by feds, for feds. +- [Digital Analytics Program](https://digital.gov/services/dap/) (DAP) — A free analytics tool for measuring digital services in the federal government +- [Section508.gov](https://www.section508.gov/) and [PlainLanguage.gov](https://www.plainlanguage.gov/) resources +- [API.data.gov](https://api.data.gov/) — a free API management service for federal agencies +- [U.S. Digital Registry](https://digital.gov/services/u-s-digital-registry/) — A resource for confirming the official status of government social media accounts, mobile apps, and mobile websites. **Questions or feedback?** [Submit an issue](https://github.com/GSA/digitalgov.gov/issues) or send us an email to [digitalgov@gsa.gov](mailto:digitalgov@gsa.gov) :heart: diff --git a/content/en/showcase/forestry/index.md b/content/en/showcase/forestry/index.md index fbfe9b96147..32a932a7a86 100644 --- a/content/en/showcase/forestry/index.md +++ b/content/en/showcase/forestry/index.md @@ -14,7 +14,7 @@ In our early research we looked at Ionic’s [site](https://github.com/ionic-tea We knew Hugo was fast but we did [some additional benchmarking](https://forestry.io/blog/hugo-vs-jekyll-benchmark/) before making our decision. Seeing Hugo in action is a whole different world of awesome. Hugo takes less than one second to build our 150-page site! Take a look: -```bash +```text | EN +------------------+-----+ Pages | 141 @@ -41,7 +41,7 @@ Lastly, we want to take the opportunity to give some love to other amazing tools * Chris can’t think of modern web development without [**Gulp**](https://gulpjs.com/) & [**Webpack**](https://webpack.js.org/). We used them to add additional build steps such as Browsersync, CSS, JS and SVG optimization. * Speaking about adding steps to our build, our lives would be much harder without [**CircleCI**](https://circleci.com/) for continuous deployment and automated testing purposes. * We can’t stop raving about [**Algolia**](https://www.algolia.com/). Chris loves it and even wrote a tutorial on [how to implement Algolia](https://forestry.io/blog/search-with-algolia-in-hugo/) into static sites using Hugo’s [Custom Outputs](/templates/output-formats/). -* [**Cloudinary**](https://cloudinary.com/) is probably one of the easiest ways to get responsive images into your website. +* [**Cloudinary**](https://cloudinary.com/) is probably one of the easiest ways to get responsive images into your website. * We might be a little biased on this one - We think [**Forestry.io**](https://forestry.io/) is a great way to add a content management system with a clean UI on top of your site without interrupting your experience as a developer. * For hosting purposes we use the almighty [**AWS**](https://aws.amazon.com/). * [**Formspree.io**](https://formspree.io/) is managing our support and enterprise requests. diff --git a/content/en/showcase/godot-tutorials/index.md b/content/en/showcase/godot-tutorials/index.md index 56a8dfc4d15..3b71fd8bcb9 100644 --- a/content/en/showcase/godot-tutorials/index.md +++ b/content/en/showcase/godot-tutorials/index.md @@ -13,7 +13,6 @@ byline: "[Godot Tutorials](https://godottutorials.com), Web Developer & Game Pro --- - [Godot Tutorials](https://godottutorials.com) started as a way to teach beginners game programming and game development. As I created videos, I ran into a problem; if I made a mistake with a YouTube video, it was difficult to correct errors. @@ -22,4 +21,4 @@ I discovered that blogging episodes and having articles that teach on top of my As I researched blogging platforms, I came across two solutions; however, I chose [Hugo](https://gohugo.io) because it's built with Markdown in mind and simplified my workflow. In a sense, with [Hugo](https://gohugo.io) programmed the right way, I can focus **more time on planning, creating, and editing** -my videos and **less time maintaining and fixing** my website. \ No newline at end of file +my videos and **less time maintaining and fixing** my website. diff --git a/content/en/showcase/letsencrypt/index.md b/content/en/showcase/letsencrypt/index.md index 18221ed6881..6ad4b7840fb 100644 --- a/content/en/showcase/letsencrypt/index.md +++ b/content/en/showcase/letsencrypt/index.md @@ -17,5 +17,4 @@ That site is bookmarked in many browsers, so preserving the URLs was a must. Hug The lessons learned from this also lead to [disableLanguages](/content-management/multilingual/#disable-a-language) in Hugo `0.34` (a way to turn off languages during translation). And I also registered [this issue](https://github.com/gohugoio/hugo/issues/4463). Once fixed it will make it easier to handle partially translated sites. - [^1]: The work on getting the content translated is in progress. diff --git a/content/en/showcase/pharmaseal/index.md b/content/en/showcase/pharmaseal/index.md index 64e9960a397..2110b439632 100644 --- a/content/en/showcase/pharmaseal/index.md +++ b/content/en/showcase/pharmaseal/index.md @@ -11,13 +11,11 @@ siteURL: https://pharmaseal.co/ # Link to the site's Hugo source code if public and you can/want to share. # Remove or leave blank if not needed/wanted. - # Add credit to the article author. Leave blank or remove if not needed/wanted. byline: "[Roboto Studio](https://roboto.studio), Jonathan Alford" --- - We wanted to shake the status quo with PHARMASEAL, opting for a fast and scalable website built with Hugo instead of slower monolithic systems the competitors were using. We had two goals: @@ -34,4 +32,4 @@ We're big fans of simplicity, and that's what we delivered with the Forestry bui PHARMASEAL have found Forestry CMS combined with HUGO to be so effective at producing fast, purpose driven pages, that we have worked with them to add even more blocks in a scalable, modular fashion. -**TLDR:** We're blown away with HUGO, the sheer speed, scalability and deployment possibilities with Netlify is the 💣 \ No newline at end of file +**TLDR:** We're blown away with HUGO, the sheer speed, scalability and deployment possibilities with Netlify is the 💣 diff --git a/content/en/showcase/quiply-employee-communications-app/bio.md b/content/en/showcase/quiply-employee-communications-app/bio.md index f72a6255499..f79677a1a51 100644 --- a/content/en/showcase/quiply-employee-communications-app/bio.md +++ b/content/en/showcase/quiply-employee-communications-app/bio.md @@ -1,4 +1,4 @@ **Quiply** is an employee communications app enabling mobile collaboration across an entire organization. -Our customers get their own branded app enabling them to communicate fast and effectively with all employees, also non-desk and shift workers. +Our customers get their own branded app enabling them to communicate fast and effectively with all employees, also non-desk and shift workers. As the Quiply app's build process is based on **Gulp**, we have started to build our company and product website using **Gulp + Hugo** which is super-fast and gives us exactly the flexibility we need. diff --git a/content/en/showcase/template/bio.md b/content/en/showcase/template/bio.md index 59716334002..de92878986b 100644 --- a/content/en/showcase/template/bio.md +++ b/content/en/showcase/template/bio.md @@ -3,6 +3,5 @@ Add some **general info** about the site here. The site is built by: -* [Person 1](https://example.com) -* [Person 1](https://example.com) - +* [Person 1](https://example.org) +* [Person 1](https://example.org) diff --git a/content/en/templates/data-templates.md b/content/en/templates/data-templates.md index cf835af4440..97da22db53e 100644 --- a/content/en/templates/data-templates.md +++ b/content/en/templates/data-templates.md @@ -12,20 +12,18 @@ aliases: [/extras/datafiles/,/extras/datadrivencontent/,/doc/datafiles/] toc: true --- - - Hugo supports loading data from YAML, JSON, XML, and TOML files located in the `data` directory at the root of your Hugo project. {{< youtube FyPgSuwIMWQ >}} -## The data folder +## The data directory -The `data` folder should store additional data for Hugo to use when generating your site. +The `data` directory should store additional data for Hugo to use when generating your site. Data files are not for generating standalone pages. They should supplement content files by: -- extending the content when the front matter fields grow out of control, or -- showing a larger dataset in a template (see the example below). +- Extending the content when the front matter fields grow out of control, or +- Showing a larger dataset in a template (see the example below). In both cases, it's a good idea to outsource the data in their (own) files. @@ -58,7 +56,9 @@ The keys in the map created with data templates from data files will be a dot-ch This is best explained with an example: -## Example: Jaco Pastorius' Solo Discography +## Examples + +### Jaco Pastorius' Solo Discography [Jaco Pastorius](https://en.wikipedia.org/wiki/Jaco_Pastorius_discography) was a great bass player, but his solo discography is short enough to use as an example. [John Patitucci](https://en.wikipedia.org/wiki/John_Patitucci) is another bass giant. @@ -111,7 +111,7 @@ And then in the `partials/artist.html`: Discover a new favorite bass player? Just add another `.toml` file in the same directory. -## Example: accessing named values in a data file +### Accessing named values in a data file Assume you have the following data structure in your `User0123.[yml|toml|xml|json]` data file located directly in `data/`: @@ -132,109 +132,19 @@ You can use the following code to render the `Short Description` in your layout: Note the use of the [`markdownify`] function. This will send the description through the Markdown rendering engine. -## Get remote data - -Use `getJSON` or `getCSV` to get remote data: - -```go-html-template -{{ $dataJ := getJSON "url" }} -{{ $dataC := getCSV "separator" "url" }} -``` - -If you use a prefix or postfix for the URL, the functions accept [variadic arguments][variadic]: - -```go-html-template -{{ $dataJ := getJSON "url prefix" "arg1" "arg2" "arg n" }} -{{ $dataC := getCSV "separator" "url prefix" "arg1" "arg2" "arg n" }} -``` - -The separator for `getCSV` must be put in the first position and can only be one character long. - -All passed arguments will be joined to the final URL: - -```go-html-template -{{ $urlPre := "https://api.github.com" }} -{{ $gistJ := getJSON $urlPre "/users/GITHUB_USERNAME/gists" }} -``` - -This will resolve internally to the following: - -```go-html-template -{{ $gistJ := getJSON "https://api.github.com/users/GITHUB_USERNAME/gists" }} -``` - -### Add HTTP headers - -Both `getJSON` and `getCSV` takes an optional map as the last argument, e.g.: - -```go-html-template -{{ $data := getJSON "https://example.org/api" (dict "Authorization" "Bearer abcd") }} -``` - -If you need multiple values for the same header key, use a slice: - -```go-html-template -{{ $data := getJSON "https://example.org/api" (dict "X-List" (slice "a" "b" "c")) }} -``` - -### Example for CSV files - -For `getCSV`, the one-character-long separator must be placed in the first position followed by the URL. The following is an example of creating an HTML table in a [partial template][partials] from a published CSV: - -{{< code file="layouts/partials/get-csv.html" >}} - - - - - - - - - - {{ $url := "https://example.com/finance/employee-salaries.csv" }} - {{ $sep := "," }} - {{ range $i, $r := getCSV $sep $url }} - - - - - - {{ end }} - -
    NamePositionSalary
    {{ index $r 0 }}{{ index $r 1 }}{{ index $r 2 }}
    -{{< /code >}} - -The expression `{{ index $r number }}` must be used to output the nth-column from the current row. - -### Cache URLs +## Remote data -Each downloaded URL will be cached in the default folder `$TMPDIR/hugo_cache_$USER/`. The variable `$TMPDIR` will be resolved to your system-dependent temporary directory. +Retrieve remote data using these template functions: -With the command-line flag `--cacheDir`, you can specify any folder on your system as a caching directory. - -You can also set `cacheDir` in the [main configuration file][config]. - -If you don't like caching at all, you can fully disable caching with the command-line flag `--ignoreCache`. - -### Authentication when using REST URLs - -Currently, you can only use those authentication methods that can be put into an URL. [OAuth] and other authentication methods are not implemented. - -## Load local files - -To load local files with `getJSON` and `getCSV`, the source files must reside within Hugo's working directory. The file extension does not matter, but the content does. - -It applies the same output logic as above in [Get Remote Data](#get-remote-data). - -{{% note %}} -The local CSV files to be loaded using `getCSV` must be located **outside** the `data` directory. -{{% /note %}} +- [`resources.GetRemote`](/functions/resources/getremote) (recommended) +- [`data.GetCSV`](/functions/data/getcsv) +- [`data.GetJSON`](/functions/data/getjson) ## LiveReload with data files There is no chance to trigger a [LiveReload] when the content of a URL changes. However, when a *local* file changes (i.e., `data/*` and `themes//data/*`), a LiveReload will be triggered. Symlinks are not supported. Note too that because downloading data takes a while, Hugo stops processing your Markdown files until the data download has been completed. -{{% warning "URL Data and LiveReload" %}} +{{% note %}} If you change any local file and the LiveReload is triggered, Hugo will read the data-driven (URL) content from the cache. If you have disabled the cache (i.e., by running the server with `hugo server --ignoreCache`), Hugo will re-download the content every time LiveReload triggers. This can create *huge* traffic. You may reach API limits quickly. {{% /note %}} diff --git a/content/en/templates/internal.md b/content/en/templates/internal.md index a73ecde4c88..6610f6e4779 100644 --- a/content/en/templates/internal.md +++ b/content/en/templates/internal.md @@ -31,9 +31,10 @@ Hugo ships with an internal template supporting [Google Analytics 4][GA4] (GA4). Provide your tracking ID in your configuration file: **Google Analytics 4 (gtag.js)** -{{< code-toggle file="hugo" >}} + +{{< code-toggle file=hugo >}} [services.googleAnalytics] -ID = "G-MEASUREMENT_ID" +googleAnalytics = "G-MEASUREMENT_ID" {{}} ### Use the Google Analytics template @@ -54,7 +55,7 @@ Hugo also ships with an internal template for [Disqus comments][disqus], a popul To use Hugo's Disqus template, first set up a single configuration value: -{{< code-toggle file="hugo" >}} +{{< code-toggle file=hugo >}} [services.disqus] shortname = 'your-disqus-shortname' {{}} @@ -122,7 +123,7 @@ This format is used for Facebook and some other sites. Hugo's Open Graph template is configured using a mix of configuration variables and [front-matter](/content-management/front-matter/) on individual pages. -{{< code-toggle file="hugo" >}} +{{< code-toggle file=hugo >}} [params] title = "My cool site" images = ["site-feature-image.jpg"] @@ -172,7 +173,7 @@ metadata used to attach rich media to Tweets linking to your site. Hugo's Twitter Card template is configured using a mix of configuration variables and [front-matter](/content-management/front-matter/) on individual pages. -{{< code-toggle file="hugo" >}} +{{< code-toggle file=hugo >}} [params] images = ["site-feature-image.jpg"] description = "Text about my cool site" @@ -192,7 +193,7 @@ Hugo uses the page title and description for the card's title and description fi Set the value of `twitter:site` in your site configuration: -{{< code-toggle file="hugo" copy=false >}} +{{< code-toggle file=hugo >}} [params.social] twitter = "GoHugoIO" {{}} diff --git a/content/en/templates/introduction.md b/content/en/templates/introduction.md index 2eb436f8227..54946699249 100644 --- a/content/en/templates/introduction.md +++ b/content/en/templates/introduction.md @@ -29,7 +29,6 @@ A _predefined variable_ could be a variable already existing in the current scope (like the `.Title` example in the [Variables](#variables) section below) or a custom variable (like the `$address` example in that same section). - ```go-html-template {{ .Title }} {{ $address }} @@ -276,7 +275,6 @@ It skips the block if the variable is absent, or if it evaluates to Below snippet uses the "description" front-matter parameter's value if set, else uses the default `.Summary` [Page variable][pagevars]: - ```go-html-template {{ with .Param "description" }} {{ . }} @@ -350,7 +348,6 @@ The following two examples are functionally the same: {{ shuffle (seq 1 5) }} ``` - ```go-html-template {{ (seq 1 5) | shuffle }} ``` @@ -429,7 +426,7 @@ Notice how once we have entered the loop (i.e. `range`), the value of `{{ . }}` {{< /code >}} -{{% warning "Don't Redefine the Dot" %}} +{{% note %}} The built-in magic of `$` would cease to work if someone were to mischievously redefine the special character; e.g. `{{ $ := .Site }}`. *Don't do it.* You may, of course, recover from this mischief by using `{{ $ := . }}` in a global context to reset `$` to its default value. {{% /note %}} @@ -535,7 +532,7 @@ An example of this is used in the Hugo docs. Most of the pages benefit from havi Here is the example front matter: -{{< code-toggle file="content/example.md" fm=true copy=false >}} +{{< code-toggle file="content/example.md" fm=true >}} title: Example notoc: true {{< /code-toggle >}} @@ -564,7 +561,7 @@ You can arbitrarily define as many site-level parameters as you want in your [si For instance, you might declare the following: -{{< code-toggle file="hugo" >}} +{{< code-toggle file=hugo >}} params: copyrighthtml: "Copyright © 2017 John Doe. All Rights Reserved." twitteruser: "spf13" @@ -617,7 +614,7 @@ content/ └── event-3.md ``` -{{< code-toggle file="content/events/event-1.md" copy=false >}} +{{< code-toggle file="content/events/event-1.md" >}} title = 'Event 1' date = 2021-12-06T10:37:16-08:00 draft = false @@ -634,7 +631,7 @@ This [partial template][partials] renders future events: {{ if gt (.Params.start_date | time.AsTime) now }} {{ $startDate := .Params.start_date | time.Format ":date_medium" }}
  • - {{ .LinkTitle }} - {{ $startDate }} + {{ .Title }} - {{ $startDate }}
  • {{ end }} {{ end }} @@ -649,7 +646,7 @@ If you restrict front matter to the TOML format, and omit quotation marks surrou {{ range where (where site.RegularPages "Type" "events") "Params.start_date" "gt" now }} {{ $startDate := .Params.start_date | time.Format ":date_medium" }}
  • - {{ .LinkTitle }} - {{ $startDate }} + {{ .Title }} - {{ $startDate }}
  • {{ end }} @@ -666,9 +663,9 @@ If you restrict front matter to the TOML format, and omit quotation marks surrou [internal templates]: /templates/internal [math]: /functions/math [pagevars]: /variables/page -[param]: /functions/param +[param]: /methods/page/param [partials]: /templates/partials -[relpermalink]: /variables/page#page-variables +[relpermalink]: /variables/page [`safehtml`]: /functions/safe/html [sitevars]: /variables/site [variables]: /variables diff --git a/content/en/templates/lists/index.md b/content/en/templates/lists/index.md index b48969e96be..0de261e583f 100644 --- a/content/en/templates/lists/index.md +++ b/content/en/templates/lists/index.md @@ -110,7 +110,7 @@ You can now access this `_index.md`'s' content in your list template: This above will output the following HTML: -{{< code file="example.com/posts/index.html" copy=false >}} +{{< code file="example.com/posts/index.html" >}}
    @@ -134,7 +134,7 @@ You do *not* have to create an `_index.md` file for every list page (i.e. sectio Using this same `layouts/_default/list.html` template and applying it to the `quotes` section above will render the following output. Note that `quotes` does not have an `_index.md` file to pull from: -{{< code file="example.com/quote/index.html" copy=false >}} +{{< code file="example.com/quote/index.html" >}}
    @@ -144,8 +144,8 @@ Using this same `layouts/_default/list.html` template and applying it to the `qu
    @@ -316,7 +316,7 @@ Lower weight gets higher precedence. So content with lower weight will come firs {{ range .Pages.ByLinkTitle }}
  • -

    {{ .LinkTitle }}

    +

    {{ .Title }}

  • {{ end }} @@ -575,9 +575,9 @@ See the documentation on [`where`] and [base]: /templates/base/ [bepsays]: https://bepsays.com/en/2016/12/19/hugo-018/ [directorystructure]: /getting-started/directory-structure/ -[`Format` function]: /functions/format/ +[`Format` function]: /methods/time/format/ [front matter]: /content-management/front-matter/ -[getpage]: /functions/getpage/ +[getpage]: /methods/page/getpage/ [homepage]: /templates/homepage/ [mentalmodel]: https://webstyleguide.com/wsg3/3-information-architecture/3-site-structure.html [pagevars]: /variables/page/ @@ -592,6 +592,6 @@ See the documentation on [`where`] and [taxvars]: /variables/taxonomy/ [views]: /templates/views/ [`where`]: /functions/collections/where -[`first`]: /functions/first/ +[`first`]: /functions/collections/first/ [main sections]: /functions/collections/where#mainsections [`time.Format`]: /functions/time/format diff --git a/content/en/templates/lookup-order.md b/content/en/templates/lookup-order.md index c36956cb96b..b49c795242e 100644 --- a/content/en/templates/lookup-order.md +++ b/content/en/templates/lookup-order.md @@ -63,7 +63,7 @@ layouts/ But the contact page probably has a form and requires a different template. In the front matter specify `layout`: -{{< code-toggle file=content/contact.md copy=false >}} +{{< code-toggle file=content/contact.md >}} title = 'Contact' layout = 'contact' {{< /code-toggle >}} @@ -79,12 +79,12 @@ layouts/ As a content type, the word `page` is vague. Perhaps `miscellaneous` would be better. Add `type` to the front matter of each page: -{{< code-toggle file=content/about.md copy=false >}} +{{< code-toggle file=content/about.md >}} title = 'About' type = 'miscellaneous' {{< /code-toggle >}} -{{< code-toggle file=content/contact.md copy=false >}} +{{< code-toggle file=content/contact.md >}} title = 'Contact' type = 'miscellaneous' layout = 'contact' @@ -117,7 +117,7 @@ A section page is a list of pages within a given section. A taxonomy page is a list of terms within a given taxonomy. The examples below assume the following site configuration: -{{< code-toggle file=hugo copy=false >}} +{{< code-toggle file=hugo >}} [taxonomies] category = 'categories' {{< /code-toggle >}} @@ -128,7 +128,7 @@ category = 'categories' A term page is a list of pages associated with a given term. The examples below assume the following site configuration: -{{< code-toggle file=hugo copy=false >}} +{{< code-toggle file=hugo >}} [taxonomies] category = 'categories' {{< /code-toggle >}} diff --git a/content/en/templates/menu-templates.md b/content/en/templates/menu-templates.md index e2d141f06f3..6b7532e29fc 100644 --- a/content/en/templates/menu-templates.md +++ b/content/en/templates/menu-templates.md @@ -28,7 +28,7 @@ The example below handles every combination. This partial template recursively "walks" a menu structure, rendering a localized, accessible nested list. -{{< code file="layouts/partials/menu.html" >}} +{{< code file="layouts/partials/menu.html" copy=true >}} {{- $page := .page }} {{- $menuID := .menuID }} @@ -128,5 +128,5 @@ Hugo provides two methods to localize your menu entries. See [multilingual]. [localize the menu entries]: /content-management/multilingual/#menus [menu entry defined in front matter]: /content-management/menus/#example-front-matter [menu entry defined in site configuration]: /content-management/menus/#example-site-configuration -[menu variables and methods]: /variables/menus/ +[menu variables and methods]: /variables/menu-entry/ [multilingual]: /content-management/multilingual/#menus diff --git a/content/en/templates/output-formats.md b/content/en/templates/output-formats.md index 5f2abdc2a14..0fad84da4a6 100644 --- a/content/en/templates/output-formats.md +++ b/content/en/templates/output-formats.md @@ -31,7 +31,7 @@ This is the full set of built-in media types in Hugo: To add or modify a media type, define it in a `mediaTypes` section in your [site configuration], either for all sites or for a given language. -{{< code-toggle file="hugo" >}} +{{< code-toggle file=hugo >}} [mediaTypes] [mediaTypes."text/enriched"] suffixes = ["enr"] @@ -43,7 +43,7 @@ The above example adds one new media type, `text/enriched`, and changes the suff **Note:** these media types are configured for **your output formats**. If you want to redefine one of Hugo's default output formats (e.g. `HTML`), you also need to redefine the media type. So, if you want to change the suffix of the `HTML` output format from `html` (default) to `htm`: -{{< code-toggle file="hugo" >}} +{{< code-toggle file=hugo >}} [mediaTypes] [mediaTypes."text/html"] suffixes = ["htm"] @@ -69,7 +69,7 @@ This is the full set of Hugo's built-in output formats: To add or modify an output format, define it in an `outputFormats` section in your site's [configuration file](/getting-started/configuration/), either for all sites or for a given language. -{{< code-toggle file="hugo" >}} +{{< code-toggle file=hugo >}} [outputFormats.MyEnrichedFormat] mediaType = "text/enriched" baseName = "myindex" @@ -139,7 +139,7 @@ per language). Example from site configuration file: -{{< code-toggle file="hugo" >}} +{{< code-toggle file=hugo >}} [outputs] home = ["html", "amp", "rss"] page = ["html"] @@ -153,7 +153,7 @@ Note that in the above examples, the _output formats_ for `section`, The following is an example of front matter in a content file that defines output formats for the rendered `Page`: -{{< code-toggle file="content/example.md" fm=true copy=false >}} +{{< code-toggle file="content/example.md" fm=true >}} title: Example outputs: - html diff --git a/content/en/templates/partials.md b/content/en/templates/partials.md index 97cf93f71f3..d5fe5aadee0 100644 --- a/content/en/templates/partials.md +++ b/content/en/templates/partials.md @@ -128,7 +128,7 @@ Value: {{ partial "my-inline-partial.html" . }} ## Cached partials -The `partialCached` template function provides significant performance gains for complex templates that don't need to be re-rendered on every invocation. See [details][partialcached]. +The `partialCached` template function provides significant performance gains for complex templates that don't need to be re-rendered on every invocation. See [details][partialcached]. ## Examples diff --git a/content/en/templates/render-hooks.md b/content/en/templates/render-hooks.md index 13106b7f715..9ced3f62b35 100644 --- a/content/en/templates/render-hooks.md +++ b/content/en/templates/render-hooks.md @@ -97,7 +97,6 @@ IsBlock {{< new-in "0.108.0" >}} Ordinal {{< new-in "0.108.0" >}} : Zero-based ordinal for all the images in the current document. - ### Link with title markdown example ```md diff --git a/content/en/templates/robots.md b/content/en/templates/robots.md index b9cdb76dd27..eea6394261a 100644 --- a/content/en/templates/robots.md +++ b/content/en/templates/robots.md @@ -14,7 +14,7 @@ aliases: [/extras/robots-txt/] To generate a robots.txt file from a template, change the [site configuration]: -{{< code-toggle file="hugo" >}} +{{< code-toggle file=hugo >}} enableRobotsTXT = true {{< /code-toggle >}} diff --git a/content/en/templates/rss.md b/content/en/templates/rss.md index b72861117c7..df97c437c94 100644 --- a/content/en/templates/rss.md +++ b/content/en/templates/rss.md @@ -15,7 +15,7 @@ toc: true By default, when you build your site, Hugo generates RSS feeds for home, section, taxonomy, and term pages. Control feed generation in your site configuration. For example, to generate feeds for home and section pages, but not for taxonomy and term pages: -{{< code-toggle file=hugo copy=false >}} +{{< code-toggle file=hugo >}} [outputs] home = ['html', 'rss'] section = ['html', 'rss'] @@ -25,13 +25,13 @@ term = ['html'] To disable feed generation for all [page kinds]: -{{< code-toggle file=hugo copy=false >}} +{{< code-toggle file=hugo >}} disableKinds = ['rss'] {{< /code-toggle >}} By default, the number of items in each feed is unlimited. Change this as needed in your site configuration: -{{< code-toggle file=hugo copy=false >}} +{{< code-toggle file=hugo >}} [services.rss] limit = 42 {{< /code-toggle >}} @@ -40,7 +40,7 @@ Set `limit` to `-1` to generate an unlimited number of items per feed. The built-in RSS template will render the following values, if present, from your site configuration: -{{< code-toggle file=hugo copy=false >}} +{{< code-toggle file=hugo >}} copyright = '© 2023 ABC Widgets, Inc.' [params.author] name = 'John Doe' diff --git a/content/en/templates/section-templates.md b/content/en/templates/section-templates.md index b35009a2fea..ed12b30a3c6 100644 --- a/content/en/templates/section-templates.md +++ b/content/en/templates/section-templates.md @@ -25,7 +25,7 @@ See [Template Lookup](/templates/lookup-order/). Every `Page` in Hugo has a `.Kind` attribute. -{{% page-kinds %}} +{{% include "content-management/_common/page-kinds.md" %}} ## `.Site.GetPage` with sections @@ -69,10 +69,10 @@ The `.Site.GetPage` example that follows assumes the following project directory . └── content ├── blog - │   ├── _index.md # "title: My Hugo Blog" in the front matter - │   ├── post-1.md - │   ├── post-2.md - │   └── post-3.md + │ ├── _index.md # "title: My Hugo Blog" in the front matter + │ ├── post-1.md + │ ├── post-2.md + │ └── post-3.md └── events #Note there is no _index.md file in "events" ├── event-1.md └── event-2.md @@ -103,7 +103,7 @@ Which then returns the following: ``` [contentorg]: /content-management/organization/ -[getpage]: /functions/getpage/ +[getpage]: /methods/page/getpage [lists]: /templates/lists/ [lookup]: /templates/lookup-order/ [`where`]: /functions/collections/where diff --git a/content/en/templates/shortcode-templates.md b/content/en/templates/shortcode-templates.md index 41b52641cc9..e90ab685ae6 100644 --- a/content/en/templates/shortcode-templates.md +++ b/content/en/templates/shortcode-templates.md @@ -10,6 +10,7 @@ menu: weight: 130 weight: 130 toc: true +aliases: [/functions/get] --- Shortcodes are a means to consolidate templating into small, reusable snippets that you can embed directly inside your content. @@ -28,7 +29,7 @@ Hugo's built-in shortcodes cover many common, but not all, use cases. Luckily, H To create a shortcode, place an HTML template in the `layouts/shortcodes` directory of your [source organization]. Consider the file name carefully since the shortcode name will mirror that of the file but without the `.html` extension. For example, `layouts/shortcodes/myshortcode.html` will be called with either `{{}}` or `{{%/* myshortcode /*/%}}`. -You can organize your shortcodes in subfolders, e.g. in `layouts/shortcodes/boxes`. These shortcodes would then be accessible with their relative path, e.g: +You can organize your shortcodes in subdirectories, e.g. in `layouts/shortcodes/boxes`. These shortcodes would then be accessible with their relative path, e.g: ```go-html-template {{}} @@ -193,7 +194,7 @@ Would load the template at `/layouts/shortcodes/youtube.html`: {{< /code >}} -{{< code file="youtube-embed.html" copy=false >}} +{{< code file="youtube-embed.html" >}}
    @@ -297,7 +298,7 @@ The template for the `highlight` shortcode uses the following code, which is alr The rendered output of the HTML example code block will be as follows: -{{< code file="syntax-highlighted.html" copy=false >}} +{{< code file="syntax-highlighted.html" >}}
    <html>
         <body> This HTML </body>
     </html>
    @@ -349,9 +350,9 @@ This will output the following HTML. Note how the first two `img` shortcodes inh
     
     ## Error handling in shortcodes
     
    -Use the [errorf](/functions/fmt/errorf) template func and [.Position](/variables/shortcodes/) variable to get useful error messages in shortcodes:
    +Use the [errorf](/functions/fmt/errorf) template func and [.Position](/variables/shortcode/) variable to get useful error messages in shortcodes:
     
    -```bash
    +```sh
     {{ with .Get "name" }}
     {{ else }}
     {{ errorf "missing value for parameter 'name': %s" .Position }}
    @@ -360,7 +361,7 @@ Use the [errorf](/functions/fmt/errorf) template func and [.Position](/variables
     
     When the above fails, you will see an `ERROR` log similar to the below:
     
    -```bash
    +```sh
     ERROR 2018/11/07 10:05:55 missing value for parameter name: "/Users/bep/dev/go/gohugoio/hugo/docs/content/en/variables/shortcodes.md:32:1"
     ```
     
    @@ -374,7 +375,7 @@ You can also implement your shortcodes inline -- e.g. where you use them in the
     
     This feature is disabled by default, but can be enabled in your site configuration:
     
    -{{< code-toggle file="hugo" >}}
    +{{< code-toggle file=hugo >}}
     enableInlineShortcodes = true
     {{< /code-toggle >}}
     
    @@ -382,7 +383,7 @@ It is disabled by default for security reasons. The security model used by Hugo'
     
     And once enabled, you can do this in your content files:
     
    - ```go-text-template
    + ```go-html-template
      {{}}{{ now }}{{}}
      ```
     
    @@ -394,7 +395,7 @@ This means that the current page can be accessed via `.Page.Title` etc. This als
     
     The same inline shortcode can be reused later in the same content file, with different parameters if needed, using the self-closing syntax:
     
    - ```go-text-template
    + ```go-html-template
     {{}}
     ```
     
    @@ -408,8 +409,8 @@ The same inline shortcode can be reused later in the same content file, with dif
     [hugosc]: /content-management/shortcodes/#using-hugo-s-built-in-shortcodes
     [lookup order]: /templates/lookup-order/
     [pagevars]: /variables/page/
    -[parent]: /variables/shortcodes/
    -[shortcodesvars]: /variables/shortcodes/
    +[parent]: /variables/shortcode/
    +[shortcodesvars]: /variables/shortcode/
     [spfscs]: https://github.com/spf13/spf13.com/tree/master/layouts/shortcodes
     [vimeoexample]: #single-flexible-example-vimeo
     [youtubeshortcode]: /content-management/shortcodes/#youtube
    diff --git a/content/en/templates/single-page-templates.md b/content/en/templates/single-page-templates.md
    index a32e8231c81..54efddc6e3c 100644
    --- a/content/en/templates/single-page-templates.md
    +++ b/content/en/templates/single-page-templates.md
    @@ -43,14 +43,14 @@ This single page template makes use of Hugo [base templates], the [`.Format` fun
           {{ with .GetTerms "topics" }}
             
           {{ end }}
           {{ with .GetTerms "tags" }}
             
           {{ end }}
    @@ -74,7 +74,7 @@ To easily generate new instances of a content type (e.g., new `.md` files in a s
     [content type]: /content-management/types/
     [directory structure]: /getting-started/directory-structure/
     [dry]: https://en.wikipedia.org/wiki/Don%27t_repeat_yourself
    -[`.format` function]: /functions/format/
    +[`.format` function]: /methods/time/format/
     [front matter]: /content-management/front-matter/
     [pagetaxonomy]: /templates/taxonomy-templates/#list-terms-assigned-to-a-page
     [pagevars]: /variables/page/
    diff --git a/content/en/templates/sitemap-template.md b/content/en/templates/sitemap-template.md
    index 4f86380f160..bccd3781225 100644
    --- a/content/en/templates/sitemap-template.md
    +++ b/content/en/templates/sitemap-template.md
    @@ -67,7 +67,7 @@ To override the built-in sitemapindex.xml template, create a new file in either
     
     You may disable sitemap generation in your site configuration:
     
    -{{< code-toggle file="hugo" >}}
    +{{< code-toggle file=hugo >}}
     disableKinds = ['sitemap']
     {{}}
     
    diff --git a/content/en/templates/taxonomy-templates.md b/content/en/templates/taxonomy-templates.md
    index 1b15bd4efa1..ca96c618584 100644
    --- a/content/en/templates/taxonomy-templates.md
    +++ b/content/en/templates/taxonomy-templates.md
    @@ -152,7 +152,7 @@ Weights of zero are thus treated specially: if two pages have unequal weights, a
     
     Content can be assigned weight for each taxonomy that it's assigned to.
     
    -{{< code-toggle file="content/example.md" fm=true copy=false >}}
    +{{< code-toggle file="content/example.md" fm=true >}}
     tags = [ "a", "b", "c" ]
     tags_weight = 22
     categories = ["d"]
    @@ -205,7 +205,7 @@ To render an unordered list:
       

    {{ (site.GetPage $taxonomy).LinkTitle }}:

    {{ end }} @@ -220,7 +220,7 @@ To render a comma-delimited list: {{ (site.GetPage $taxonomy).LinkTitle }}: {{ range $k, $_ := . -}} {{ if $k }}, {{ end }} - {{ .LinkTitle }} + {{ .Title }} {{- end }}

    {{ end }} @@ -253,7 +253,7 @@ This would be very useful in a sidebar as “featured content”. You could even
  • {{ $key }}
  • {{ end }} @@ -288,7 +288,7 @@ This example will list all taxonomies and their terms, as well as all the conten {{ range $taxonomy, $terms := site.Taxonomies }}
  • {{ with site.GetPage $taxonomy }} - {{ .LinkTitle }} + {{ .Title }} {{ end }}
      {{ range $term, $weightedPages := $terms }} @@ -296,7 +296,7 @@ This example will list all taxonomies and their terms, as well as all the conten {{ .Page.LinkTitle }} @@ -322,8 +322,7 @@ Because taxonomies are lists, the [`.GetPage` function][getpage] can be used to
    {{< /code >}} -[delimit]: /functions/collections/delimit/ -[getpage]: /functions/getpage/ +[getpage]: /methods/page/getpage [lists]: /templates/lists/ [renderlists]: /templates/lists/ [single page template]: /templates/single-page-templates/ diff --git a/content/en/templates/views.md b/content/en/templates/views.md index 3d00a4ed612..595e3071b1b 100644 --- a/content/en/templates/views.md +++ b/content/en/templates/views.md @@ -36,7 +36,6 @@ To create a new view, create a template in each of your different content type d Hugo also has support for a default content template to be used in the event that a specific content view template has not been provided for that type. Content views can also be defined in the `_default` directory and will work the same as list and single templates who eventually trickle down to the `_default` directory as a matter of the lookup order. - ```txt ▾ layouts/ ▾ _default/ @@ -104,7 +103,7 @@ Continuing on the previous example, we can change our render function to use a s [lists]: /templates/lists/ [lookup]: /templates/lookup-order/ [pagevars]: /variables/page/ -[render]: /functions/render/ +[render]: /methods/page/render/ [single]: /templates/single-page-templates/ [spf]: https://spf13.com [spfsourceli]: https://github.com/spf13/spf13.com/blob/master/layouts/_default/li.html diff --git a/content/en/tools/editors.md b/content/en/tools/editors.md index ea83f37c685..8ff632e0cdd 100644 --- a/content/en/tools/editors.md +++ b/content/en/tools/editors.md @@ -1,46 +1,66 @@ --- -title: Editor plugins for Hugo +title: Editor plugins linkTitle: Editor plugins -description: The Hugo community uses a wide range of preferred tools and has developed plug-ins for some of the most popular text editors to help automate parts of your workflow. +description: The Hugo community uses a wide range of tools and has developed plugins for some of the most popular text editors to help automate parts of your workflow. categories: [developer tools] -keywords: [editor, plug-ins] +keywords: [editor, plugin] menu: docs: parent: developer-tools weight: 20 weight: 20 +toc: true --- -## Sublime Text +## Visual Studio Code -* [Hugofy](https://github.com/akmittal/Hugofy). Hugofy is a plugin for Sublime Text 3 to make life easier to use Hugo static site generator. -* [Hugo Snippets](https://packagecontrol.io/packages/Hugo%20Snippets). Hugo Snippets is a useful plugin for adding automatic snippets to Sublime Text 3. +[Front Matter](https://marketplace.visualstudio.com/items?itemName=eliostruyf.vscode-front-matter) +: Once you go for a static site, you need to think about how you are going to manage your articles. Front matter is a tool that helps you maintain the metadata/front matter of your articles like: creation date, modified date, slug, tile, SEO check, and more. -## Visual Studio Code +[Hugo Helper](https://marketplace.visualstudio.com/items?itemName=rusnasonov.vscode-hugo) +: Hugo Helper is a plugin for Visual Studio Code that has some useful commands for Hugo. The source code can be found [here](https://github.com/rusnasonov/vscode-hugo). + +[Hugo Language and Syntax Support](https://marketplace.visualstudio.com/items?itemName=budparr.language-hugo-vscode) +: Hugo Language and Syntax Support is a Visual Studio Code plugin for Hugo syntax highlighting and snippets. The source code can be found [here](https://github.com/budparr/language-hugo-vscode). -* [Front Matter](https://marketplace.visualstudio.com/items?itemName=eliostruyf.vscode-front-matter). Once you go for a static site, you need to think about how you are going to manage your articles. Front matter is a tool that helps you maintain the metadata/front matter of your articles like: creation date, modified date, slug, tile, SEO check, and many more... -* [Hugo Helper](https://marketplace.visualstudio.com/items?itemName=rusnasonov.vscode-hugo). Hugo Helper is a plugin for Visual Studio Code that has some useful commands for Hugo. The source code can be found [here](https://github.com/rusnasonov/vscode-hugo). -* [Hugo Language and Syntax Support](https://marketplace.visualstudio.com/items?itemName=budparr.language-hugo-vscode). Hugo Language and Syntax Support is a Visual Studio Code plugin for Hugo syntax highlighting and snippets. The source code can be found [here](https://github.com/budparr/language-hugo-vscode). -* [Hugo Themer](https://marketplace.visualstudio.com/items?itemName=eliostruyf.vscode-hugo-themer). Hugo Themer is an extension to help you while developing themes. It allows you to easily navigate through your theme files. -* [Hugofy](https://marketplace.visualstudio.com/items?itemName=akmittal.hugofy). Hugofy is a plugin for Visual Studio Code to "make life easier" when developing with Hugo. The source code can be found [here](https://github.com/akmittal/hugofy-vscode). -* [Prettier Plugin for Go Templates](https://github.com/NiklasPor/prettier-plugin-go-template). Format Hugo templates using this [Prettier](https://prettier.io/) plugin. See [installation instructions](https://discourse.gohugo.io/t/38403). -* [Syntax Highlighting for Hugo Shortcodes](https://marketplace.visualstudio.com/items?itemName=kaellarkin.hugo-shortcode-syntax). This extension adds some syntax highlighting for Shortcodes, making visual identification of individual pieces easier. - -Front Matter -Hugo Helper -Hugo Language and Syntax Support -Hugo Themer -Hugofy -Syntax Highlighting for Hugo Shortcodes +[Hugo Themer](https://marketplace.visualstudio.com/items?itemName=eliostruyf.vscode-hugo-themer) +: Hugo Themer is an extension to help you while developing themes. It allows you to easily navigate through your theme files. + +[Hugofy](https://marketplace.visualstudio.com/items?itemName=akmittal.hugofy) +: Hugofy is a plugin for Visual Studio Code to "make life easier" when developing with Hugo. The source code can be found [here](https://github.com/akmittal/hugofy-vscode). + +[Prettier Plugin for Go Templates](https://github.com/NiklasPor/prettier-plugin-go-template) +: Format Hugo templates using this [Prettier](https://prettier.io/) plugin. See [installation instructions](https://discourse.gohugo.io/t/38403). + +[Syntax Highlighting for Hugo Shortcodes](https://marketplace.visualstudio.com/items?itemName=kaellarkin.hugo-shortcode-syntax) +: This extension adds some syntax highlighting for Shortcodes, making visual identification of individual pieces easier. ## Emacs -* [emacs-easy-hugo](https://github.com/masasam/emacs-easy-hugo). Emacs major mode for managing hugo blogs. Note that Hugo also supports [Org-mode][formats]. -* [ox-hugo.el](https://ox-hugo.scripter.co). Native Org-mode exporter that exports to Blackfriday Markdown with Hugo front-matter. `ox-hugo` supports two common Org blogging flows --- exporting multiple Org subtrees in a single file to multiple Hugo posts, and exporting a single Org file to a single Hugo post. It also leverages the Org tag and property inheritance features. See [*Why ox-hugo?*](https://ox-hugo.scripter.co/doc/why-ox-hugo/) for more. +[emacs-easy-hugo](https://github.com/masasam/emacs-easy-hugo) +: Emacs major mode for managing hugo blogs. Note that Hugo also supports [Org-mode][formats]. + +[ox-hugo.el](https://ox-hugo.scripter.co) +: Native Org-mode exporter that exports to Blackfriday Markdown with Hugo front-matter. `ox-hugo` supports two common Org blogging flows --- exporting multiple Org subtrees in a single file to multiple Hugo posts, and exporting a single Org file to a single Hugo post. It also leverages the Org tag and property inheritance features. See [*Why ox-hugo?*](https://ox-hugo.scripter.co/doc/why-ox-hugo/) for more. + +## Sublime Text + +[Hugofy](https://github.com/akmittal/Hugofy) +: Hugofy is a plugin for Sublime Text 3 to make life easier to use Hugo static site generator. + +[Hugo Snippets](https://packagecontrol.io/packages/Hugo%20Snippets) +: Hugo Snippets is a useful plugin for adding automatic snippets to Sublime Text 3. ## Vim -* [Vim Hugo Helper](https://github.com/robertbasic/vim-hugo-helper). A small Vim plugin that facilitates authoring pages and blog posts with Hugo. -* [vim-hugo](https://github.com/phelipetls/vim-hugo). A Vim plugin with syntax highlighting for templates and a few other features. +[Vim Hugo Helper]: https://github.com/robertbasic/vim-hugo-helper + +[Vim Hugo Helper] +: A small Vim plugin that facilitates authoring pages and blog posts with Hugo. + +[xxx]: xxx + +[vim-hugo](https://github.com/phelipetls/vim-hugo) +: A Vim plugin with syntax highlighting for templates and a few other features. [formats]: /content-management/formats/ diff --git a/content/en/tools/frontends.md b/content/en/tools/frontends.md index bc4df22b594..2ecac6f05e3 100644 --- a/content/en/tools/frontends.md +++ b/content/en/tools/frontends.md @@ -1,5 +1,5 @@ --- -title: Frontend interfaces with Hugo +title: Frontend interfaces linkTitle: Frontends description: Do you prefer a graphical user interface over a text editor? Give these frontends a try. categories: [developer tools] @@ -9,14 +9,18 @@ menu: parent: developer-tools weight: 30 weight: 30 +toc: true --- -- [enwrite](https://github.com/zzamboni/enwrite). Enwrite enables evernote-powered, statically generated blogs and websites. Now posting to your blog or updating your website is as easy as writing a new note in Evernote! -- [Lipi](https://github.com/SohanChy/Lipi). Lipi is a native GUI frontend written in Java to manage your Hugo websites. -- [Decap CMS (formerly Netlify CMS)](https://decapcms.org/). Decap CMS is an open source, serverless solution for managing Git based content in static sites, and it works on any platform that can host static sites. A [Hugo/Decap CMS starter](https://github.com/decaporg/one-click-hugo-cms) is available to get new projects running quickly. -- [Hokus CMS](https://github.com/julianoappelklein/hokus). Hokus CMS is an open source, multi-platform, easy to use, desktop application for Hugo. Build from simple to complex user interfaces for Hugo websites by choosing from a dozen ready-to-use components — all for free, with no vendor lock-in. +## Commercial -## Commercial services +[CloudCannon](https://cloudcannon.com/hugo-cms/) +: The intuitive Git-based CMS for your Hugo website. CloudCannon syncs changes from your Git repository and pushes content changes back, so your development and content teams are always in sync. Edit all of your content on the page with visual editing, build entire pages with reusable custom components and then publish confidently. -- [DATOCMS](https://www.datocms.com) DatoCMS is a fully customizable administrative area for your static websites. Use your favorite website generator, let your clients publish new content independently, and the host the site anywhere you like. -- [CloudCannon](https://cloudcannon.com/hugo-cms/). The intuitive Git-based CMS for your Hugo website. CloudCannon syncs changes from your Git repository and pushes content changes back, so your development and content teams are always in sync. Edit all of your content on the page with visual editing, build entire pages with reusable custom components and then publish confidently. +[DATOCMS](https://www.datocms.com) +: DatoCMS is a fully customizable administrative area for your static websites. Use your favorite website generator, let your clients publish new content independently, and the host the site anywhere you like. + +## Open source + +[Decap CMS](https://decapcms.org/) +: Decap CMS is an open source, serverless solution for managing Git based content in static sites, and it works on any platform that can host static sites. A [Hugo/Decap CMS starter](https://github.com/decaporg/one-click-hugo-cms) is available to get new projects running quickly. diff --git a/content/en/tools/migrations.md b/content/en/tools/migrations.md index 8ba01f927d0..a23c19319ad 100644 --- a/content/en/tools/migrations.md +++ b/content/en/tools/migrations.md @@ -14,63 +14,88 @@ toc: true This section highlights some projects around Hugo that are independently developed. These tools try to extend the functionality of our static site generator or help you to get started. -{{% note %}} -Do you know or maintain a similar project around Hugo? Feel free to open a [pull request](https://github.com/gohugoio/hugoDocs/pulls) on GitHub if you think it should be added. -{{% /note %}} - Take a look at this list of migration tools if you currently use other blogging tools like Jekyll or WordPress but intend to switch to Hugo instead. They'll take care to export your content into Hugo-friendly formats. ## Jekyll -Alternatively, you can use the new [Jekyll import command](/commands/hugo_import_jekyll/). +Alternatively, you can use the [Jekyll import command](/commands/hugo_import_jekyll/). + +[JekyllToHugo](https://github.com/fredrikloch/JekyllToHugo) +: A Small script for converting Jekyll blog posts to a Hugo site. -- [JekyllToHugo](https://github.com/fredrikloch/JekyllToHugo) - A Small script for converting Jekyll blog posts to a Hugo site. -- [ConvertToHugo](https://github.com/coderzh/ConvertToHugo) - Convert your blog from Jekyll to Hugo. +[ConvertToHugo](https://github.com/coderzh/ConvertToHugo) +: Convert your blog from Jekyll to Hugo. ## Octopress -- [octohug](https://github.com/codebrane/octohug) - Octopress to Hugo migrator. +[octohug](https://github.com/codebrane/octohug) +: Octopress to Hugo migrator. ## DokuWiki -- [dokuwiki-to-hugo](https://github.com/wgroeneveld/dokuwiki-to-hugo) - Migrates your DokuWiki source pages from [DokuWiki syntax](https://www.dokuwiki.org/wiki:syntax) to Hugo Markdown syntax. Includes extra's like the TODO plugin. Written with extensibility in mind using python 3. Also generates a TOML header for each page. Designed to copypaste the wiki directory into your /content directory. +[dokuwiki-to-hugo](https://github.com/wgroeneveld/dokuwiki-to-hugo) +: Migrates your DokuWiki source pages from [DokuWiki syntax](https://www.dokuwiki.org/wiki:syntax) to Hugo Markdown syntax. Includes extra's like the TODO plugin. Written with extensibility in mind using python 3. Also generates a TOML header for each page. Designed to copypaste the wiki directory into your /content directory. ## WordPress -- [wordpress-to-hugo-exporter](https://github.com/SchumacherFM/wordpress-to-hugo-exporter) - A one-click WordPress plugin that converts all posts, pages, taxonomies, metadata, and settings to Markdown and YAML which can be dropped into Hugo. (Note: If you have trouble using this plugin, you can [export your site for Jekyll](https://wordpress.org/plugins/jekyll-exporter/) and use Hugo's built in Jekyll converter listed above.) -- [blog2md](https://github.com/palaniraja/blog2md) - Works with [exported xml](https://en.support.wordpress.com/export/) file of your free YOUR-TLD.wordpress.com website. It also saves approved comments to `YOUR-POST-NAME-comments.md` file along with posts. -- [wordhugopress](https://github.com/nantipov/wordhugopress) - A small utility written in Java, exports the entire WordPress site from the database and resource (e.g. images) files stored locally or remotely. Therefore, migration from the backup files is possible. Supports merging of the multiple WordPress sites into a single Hugo one. +[wordpress-to-hugo-exporter](https://github.com/SchumacherFM/wordpress-to-hugo-exporter) +: A one-click WordPress plugin that converts all posts, pages, taxonomies, metadata, and settings to Markdown and YAML which can be dropped into Hugo. (Note: If you have trouble using this plugin, you - \s-\scan [export your site for Jekyll](https://wordpress.org/plugins/jekyll-exporter/) and use Hugo's built in Jekyll converter listed above.) + +[blog2md](https://github.com/palaniraja/blog2md) +: Works with [exported xml](https://en.support.wordpress.com/export/) file of your free YOUR-TLD.wordpress.com website. It also saves approved comments to `YOUR-POST-NAME-comments.md` file along with posts. + +[wordhugopress](https://github.com/nantipov/wordhugopress) +: A small utility written in Java, exports the entire WordPress site from the database and resource (e.g. images) files stored locally or remotely. Therefore, migration from the backup files is possible. Supports merging of the multiple WordPress sites into a single Hugo one. ## Medium -- [medium2md](https://github.com/gautamdhameja/medium-2-md) - A simple Medium to Hugo exporter able to import stories in one command, including front matter. -- [medium-to-hugo](https://github.com/bgadrian/medium-to-hugo) - CLI tool written in Go to export medium posts into a Hugo compatible Markdown format. Tags and images are included. All images will be downloaded locally and linked appropriately. +[medium2md](https://github.com/gautamdhameja/medium-2-md) +: A simple Medium to Hugo exporter able to import stories in one command, including front matter. + +[medium-to-hugo](https://github.com/bgadrian/medium-to-hugo) +: CLI tool written in Go to export medium posts into a Hugo compatible Markdown format. Tags and images are included. All images will be downloaded locally and linked appropriately. ## Tumblr -- [tumblr-importr](https://github.com/carlmjohnson/tumblr-importr) - An importer that uses the Tumblr API to create a Hugo static site. -- [tumblr2hugomarkdown](https://github.com/Wysie/tumblr2hugomarkdown) - Export all your Tumblr content to Hugo Markdown files with preserved original formatting. -- [Tumblr to Hugo](https://github.com/jipiboily/tumblr-to-hugo) - A migration tool that converts each of your Tumblr posts to a content file with a proper title and path. Furthermore, "Tumblr to Hugo" creates a CSV file with the original URL and the new path on Hugo, to help you setup the redirections. +[tumblr-importr](https://github.com/carlmjohnson/tumblr-importr) +: An importer that uses the Tumblr API to create a Hugo static site. + +[tumblr2hugomarkdown](https://github.com/Wysie/tumblr2hugomarkdown) +: Export all your Tumblr content to Hugo Markdown files with preserved original formatting. + +[Tumblr to Hugo](https://github.com/jipiboily/tumblr-to-hugo) +: A migration tool that converts each of your Tumblr posts to a content file with a proper title and path. Furthermore, "Tumblr to Hugo" creates a CSV file with the original URL and the new path on Hugo, to help you setup the redirections. ## Drupal -- [drupal2hugo](https://github.com/danapsimer/drupal2hugo) - Convert a Drupal site to Hugo. +[drupal2hugo](https://github.com/danapsimer/drupal2hugo) +: Convert a Drupal site to Hugo. ## Joomla -- [hugojoomla](https://github.com/davetcc/hugojoomla) - This utility written in Java takes a Joomla database and converts all the content into Markdown files. It changes any URLs that are in Joomla's internal format and converts them to a suitable form. +[hugojoomla](https://github.com/davetcc/hugojoomla) +: This utility written in Java takes a Joomla database and converts all the content into Markdown files. It changes any URLs that are in Joomla's internal format and converts them to a suitable form. ## Blogger -- [blogimport](https://github.com/natefinch/blogimport) - A tool to import from Blogger posts to Hugo. -- [blogger-to-hugo](https://pypi.org/project/blogger-to-hugo/) - Another tool to import Blogger posts to Hugo. It also downloads embedded images so they will be stored locally. -- [blog2md](https://github.com/palaniraja/blog2md) - Works with [exported xml](https://support.google.com/blogger/answer/41387?hl=en) file of your YOUR-TLD.blogspot.com website. It also saves comments to `YOUR-POST-NAME-comments.md` file along with posts. -- [BloggerToHugo](https://github.com/huanlin/blogger-to-hugo) - Yet another tool to import Blogger posts to Hugo. For Windows platform only, and .NET Framework 4.5 is required. See README.md before using this tool. +[blogimport](https://github.com/natefinch/blogimport) +: A tool to import from Blogger posts to Hugo. + +[blogger-to-hugo](https://pypi.org/project/blogger-to-hugo/) +: Another tool to import Blogger posts to Hugo. It also downloads embedded images so they will be stored locally. + +[blog2md](https://github.com/palaniraja/blog2md) +: Works with [exported xml](https://support.google.com/blogger/answer/41387?hl=en) file of your YOUR-TLD.blogspot.com website. It also saves comments to `YOUR-POST-NAME-comments.md` file along with posts. + +[BloggerToHugo](https://github.com/huanlin/blogger-to-hugo) +: Yet another tool to import Blogger posts to Hugo. For Windows platform only, and .NET Framework 4.5 is required. See README.md before using this tool. ## Contentful -- [contentful-hugo](https://github.com/ModiiMedia/contentful-hugo) - A tool to create content-files for Hugo from content on [Contentful](https://www.contentful.com/). +[contentful-hugo](https://github.com/ModiiMedia/contentful-hugo) +: A tool to create content-files for Hugo from content on [Contentful](https://www.contentful.com/). ## BlogML -- [BlogML2Hugo](https://github.com/jijiechen/BlogML2Hugo) - A tool that helps you convert BlogML xml file to Hugo Markdown files. Users need to take care of links to attachments and images by themselves. This helps the blogs that export BlogML files (e.g. BlogEngine.NET) transform to hugo sites easily. +[BlogML2Hugo](https://github.com/jijiechen/BlogML2Hugo) +: A tool that helps you convert BlogML xml file to Hugo Markdown files. Users need to take care of links to attachments and images by themselves. This helps the blogs that export BlogML files (e.g. BlogEngine.NET) transform to hugo sites easily. diff --git a/content/en/tools/other.md b/content/en/tools/other.md index 27282791147..7b1489862be 100644 --- a/content/en/tools/other.md +++ b/content/en/tools/other.md @@ -17,7 +17,7 @@ And for all the other small things around Hugo: - [flickr-hugo-embed](https://github.com/nikhilm/flickr-hugo-embed) prints shortcodes to embed a set of images from an album on Flickr into Hugo. - [hugo-openapispec-shortcode](https://github.com/tenfourty/hugo-openapispec-shortcode) A shortcode that allows you to include [Open API Spec](https://openapis.org) (formerly known as Swagger Spec) in a page. - [HugoPhotoSwipe](https://github.com/GjjvdBurg/HugoPhotoSwipe) makes it easy to create image galleries using PhotoSwipe. -- [Hugo SFTP Upload](https://github.com/thomasmey/HugoSftpUpload) Syncs the local build of your Hugo website with your remote webserver via SFTP. +- [Hugo SFTP Upload](https://github.com/thomasmey/HugoSftpUpload) Syncs the local build of your Hugo website with your remote web server via SFTP. - [Emacs Easy Hugo](https://github.com/masasam/emacs-easy-hugo) Emacs package for writing blog posts in markdown or org-mode and building your site with Hugo. - [JAMStack Themes](https://jamstackthemes.dev/ssg/hugo/). JAMStack themes is a collection of site themes filterable by static site generator and supported CMS to help build CMS-connected sites using Hugo (linking to Hugo-specific themes). - [plausible-hugo](https://github.com/divinerites/plausible-hugo). Easy Hugo integration for Plausible Analytics, a simple, open-source, lightweight and privacy-friendly web analytics alternative to Google Analytics. diff --git a/content/en/tools/search.md b/content/en/tools/search.md index 620ba4862a6..b7facc186da 100644 --- a/content/en/tools/search.md +++ b/content/en/tools/search.md @@ -1,5 +1,5 @@ --- -title: Search for your Hugo website +title: Search tools linkTitle: Search description: See some of the open-source and commercial search options for your newly created Hugo website. menu: @@ -12,19 +12,42 @@ toc: true A static website with a dynamic search function? Yes, Hugo provides an alternative to embeddable scripts from Google or other search engines for static websites. Hugo allows you to provide your visitors with a custom search function by indexing your content files directly. -* [GitHub Gist for Hugo Workflow](https://gist.github.com/sebz/efddfc8fdcb6b480f567). This gist contains a simple workflow to create a search index for your static website. It uses a simple Grunt script to index all your content files and [lunr.js](https://lunrjs.com/) to serve the search results. +## Open source -* [hugo-lunr](https://www.npmjs.com/package/hugo-lunr). A simple way to add site search to your static Hugo site using [lunr.js](https://lunrjs.com/). Hugo-lunr will create an index file of any HTML and Markdown documents in your Hugo project. -* [hugo-lunr-zh](https://www.npmjs.com/package/hugo-lunr-zh). A bit like Hugo-lunr, but Hugo-lunr-zh can help you separate the Chinese keywords. -* [GitHub Gist for Fuse.js integration](https://gist.github.com/eddiewebb/735feb48f50f0ddd65ae5606a1cb41ae). This gist demonstrates how to leverage Hugo's existing build time processing to generate a searchable JSON index used by [Fuse.js](https://fusejs.io/) on the client-side. Although this gist uses Fuse.js for fuzzy matching, any client-side search tool capable of reading JSON indexes will work. Does not require npm, grunt or other build-time tools except Hugo! -* [hugo-search-index](https://www.npmjs.com/package/hugo-search-index). A library containing Gulp tasks and a prebuilt browser script that implements search. Gulp generates a search index from project markdown files. -* [hugofastsearch](https://gist.github.com/cmod/5410eae147e4318164258742dd053993). A usability and speed update to "GitHub Gist for Fuse.js integration" — global, keyboard-optimized search. -* [JS & Fuse.js tutorial](https://makewithhugo.com/add-search-to-a-hugo-site/) A simple client-side search solution, using FuseJS (does not require jQuery). -* [Pagefind](https://github.com/cloudcannon/pagefind). A fully static search library that aims to perform well on large sites, while using as little of your users' bandwidth as possible. -* [Hugo Lyra](https://github.com/paolomainardi/hugo-lyra). Hugo-Lyra is a JavaScript module to integrate [Lyra](https://github.com/LyraSearch/lyra) into a Hugo website. It contains the server-side part to generate the index and the client-side library (optional) to bootstrap the search engine easily. +[Pagefind](https://github.com/cloudcannon/pagefind) +: A fully static search library that aims to perform well on large sites, while using as little of your users' bandwidth as possible. -## Commercial search services +[GitHub Gist for Hugo Workflow](https://gist.github.com/sebz/efddfc8fdcb6b480f567) +: This gist contains a simple workflow to create a search index for your static website. It uses a simple Grunt script to index all your content files and [lunr.js](https://lunrjs.com/) to serve the search results. -* [Algolia](https://www.algolia.com/)'s Search API makes it easy to deliver a great search experience in your apps and websites. Algolia Search provides hosted full-text, numerical, faceted, and geolocalized search. -* [Bonsai](https://www.bonsai.io) is a fully-managed hosted Elasticsearch service that is fast, reliable, and simple to set up. Easily ingest your docs from Hugo into Elasticsearch following [this guide from the docs](https://docs.bonsai.io/hc/en-us/articles/13929190788756-Hugo). -* [ExpertRec](https://www.expertrec.com/) is a hosted search-as-a-service solution that is fast and scalable. Set-up and integration is extremely easy and takes only a few minutes. The search settings can be modified without coding using a dashboard. +[hugo-lunr](https://www.npmjs.com/package/hugo-lunr) +: A simple way to add site search to your static Hugo site using [lunr.js](https://lunrjs.com/). Hugo-lunr will create an index file of any HTML and Markdown documents in your Hugo project. + +[hugo-lunr-zh](https://www.npmjs.com/package/hugo-lunr-zh) +: A bit like Hugo-lunr, but Hugo-lunr-zh can help you separate the Chinese keywords. + +[GitHub Gist for Fuse.js integration](https://gist.github.com/eddiewebb/735feb48f50f0ddd65ae5606a1cb41ae) +: This gist demonstrates how to leverage Hugo's existing build time processing to generate a searchable JSON index used by [Fuse.js](https://fusejs.io/) on the client-side. Although this gist uses Fuse.js for fuzzy matching, any client-side search tool capable of reading JSON indexes will work. Does not require npm, grunt or other build-time tools except Hugo! + +[hugo-search-index](https://www.npmjs.com/package/hugo-search-index) +: A library containing Gulp tasks and a prebuilt browser script that implements search. Gulp generates a search index from project markdown files. + +[hugofastsearch](https://gist.github.com/cmod/5410eae147e4318164258742dd053993) +: A usability and speed update to "GitHub Gist for Fuse.js integration" — global, keyboard-optimized search. + +[JS & Fuse.js tutorial](https://makewithhugo.com/add-search-to-a-hugo-site/) +: A simple client-side search solution, using FuseJS (does not require jQuery). + +[Hugo Lyra](https://github.com/paolomainardi/hugo-lyra) +: Hugo-Lyra is a JavaScript module to integrate [Lyra](https://github.com/LyraSearch/lyra) into a Hugo website. It contains the server-side part to generate the index and the client-side library (optional) to bootstrap the search engine easily. + +## Commercial + +[Algolia](https://www.algolia.com/) +: Algolia's Search API makes it easy to deliver a great search experience in your apps and websites. Algolia Search provides hosted full-text, numerical, faceted, and geolocalized search. + +[Bonsai](https://www.bonsai.io) +: Bonsai is a fully-managed hosted Elasticsearch service that is fast, reliable, and simple to set up. Easily ingest your docs from Hugo into Elasticsearch following [this guide from the docs](https://docs.bonsai.io/hc/en-us/articles/13929190788756-Hugo). + +[ExpertRec](https://www.expertrec.com/) +: ExpertRec is a hosted search-as-a-service solution that is fast and scalable. Set-up and integration is extremely easy and takes only a few minutes. The search settings can be modified without coding using a dashboard. diff --git a/content/en/troubleshooting/faq.md b/content/en/troubleshooting/faq.md index bc29bfba1f5..15ca8d36eed 100644 --- a/content/en/troubleshooting/faq.md +++ b/content/en/troubleshooting/faq.md @@ -53,7 +53,7 @@ Also see this Twitter thread: If you process `SCSS` or `Sass` to `CSS` in your Hugo project with `libsass` as the transpiler or if you convert images to the `webp` format, you need the Hugo `extended` version, or else you may see an error message similar to the below: -```bash +```sh error: failed to transform resource: TOCSS: failed to transform "scss/main.scss" (text/x-scss): this feature is not available in your current Hugo version ``` diff --git a/content/en/variables/_common/_index.md b/content/en/variables/_common/_index.md new file mode 100644 index 00000000000..47d5812fba5 --- /dev/null +++ b/content/en/variables/_common/_index.md @@ -0,0 +1,13 @@ +--- +cascade: + _build: + list: never + publishResources: false + render: never +--- + + diff --git a/content/en/variables/_common/consistent-terminology.md b/content/en/variables/_common/consistent-terminology.md new file mode 100644 index 00000000000..e4dc1cb36cd --- /dev/null +++ b/content/en/variables/_common/consistent-terminology.md @@ -0,0 +1,16 @@ +--- +# Do not remove front matter. +--- + +{{% note %}} +Historically and inconsistently referred to as variables, parameters, or methods, we are making an effort to unify our terminology and use the terms consistently throughout the documentation. + +If it begins with a dot, it's a [method], not a [variable]. + +This page will remain in place under the "Variables" section as readers become familiar with the unified terminology. See the [methods] section for a complete list of methods by [object]. + +[method]: /getting-started/glossary/#method +[object]: /getting-started/glossary/#object +[methods]: /methods +[variable]: /getting-started/glossary/#variable +{{% /note %}} diff --git a/content/en/variables/_index.md b/content/en/variables/_index.md index c1f80a62c22..09d43b3eefe 100644 --- a/content/en/variables/_index.md +++ b/content/en/variables/_index.md @@ -2,14 +2,14 @@ title: Variables and parameters linkTitle: Overview description: Page-, file-, taxonomy-, and site-level variables and parameters available in templates. -categories: [variables and parameters] +categories: [variables] keywords: [variables,params,values,globals] menu: docs: identifier: variables-overview parent: variables - weight: 1 -weight: 1 + weight: 10 +weight: 10 aliases: [/templates/variables/] --- diff --git a/content/en/variables/file.md b/content/en/variables/file.md new file mode 100644 index 00000000000..33ee2736864 --- /dev/null +++ b/content/en/variables/file.md @@ -0,0 +1,18 @@ +--- +title: File variables +description: Retrieve file information about any page that is backed by a file. +categories: [variables] +keywords: [] +menu: + docs: + parent: variables + weight: 20 +weight: 20 +aliases: [/variables/file-variables/,/variables/files/] +--- + +{{% include "variables/_common/consistent-terminology.md" %}} + +See the documentation for the [`File`] method on a `Page` object. + +[`File`]: /methods/page/file diff --git a/content/en/variables/files.md b/content/en/variables/files.md deleted file mode 100644 index fa2a63a9ce3..00000000000 --- a/content/en/variables/files.md +++ /dev/null @@ -1,102 +0,0 @@ ---- -title: File variables -description: "Use File variables to access file-related values for each page that is backed by a file." -categories: [variables and parameters] -keywords: [files] -menu: - docs: - parent: variables - weight: 40 -toc: true -weight: 40 -aliases: [/variables/file-variables/] ---- -## Variables - -{{% note %}} -The path separators (slash or backslash) in `.File.Path`, `.File.Dir`, and `.File.Filename` depend on the operating system. -{{% /note %}} - -.File.Path -: (`string`) The file path, relative to the `content` directory. - -.File.Dir -: (`string`) The file path, excluding the file name, relative to the `content` directory. - -.File.LogicalName -: (`string`) The file name. - -.File.BaseFileName -: (`string`) The file name, excluding the extension. - -.File.TranslationBaseName -: (`string`) The file name, excluding the extension and language identifier. - -.File.Ext -: (`string`) The file extension. - -.File.Lang -: (`string`) The language associated with the given file. - - -.File.ContentBaseName -: (`string`) If the page is a branch or leaf bundle, the name of the containing directory, else the `.TranslationBaseName`. - -.File.Filename -: (`string`) The absolute file path. - -.File.UniqueID -: (`string`) The MD5 hash of `.File.Path`. - -## Examples - -```text -content/ -├── news/ -│   ├── b/ -│   │   ├── index.de.md <-- leaf bundle -│   │   └── index.en.md <-- leaf bundle -│   ├── a.de.md <-- regular content -│   ├── a.en.md <-- regular content -│   ├── _index.de.md <-- branch bundle -│   └── _index.en.md <-- branch bundle -├── _index.de.md -└── _index.en.md -``` - -With the content structure above, the `.File` objects for the English pages contain the following properties: - - |regular content|leaf bundle|branch bundle -:--|:--|:--|:-- -Path|news/a.en.md|news/b/index.en.md|news/_index.en.md -Dir|news/|news/b/|news/ -LogicalName|a.en.md|index.en.md|_index.en.md -BaseFileName|a.en|index.en|_index.en -TranslationBaseName|a|index|_index -Ext|md|md|md -Lang|en|en|en -ContentBaseName|a|b|news -Filename|/home/user/...|/home/user/...|/home/user/... -UniqueID|15be14b...|186868f...|7d9159d... - -## Defensive coding - -Some of the pages on a site may not be backed by a file. For example: - -- Top level section pages -- Taxonomy pages -- Term pages - -Without a backing file, Hugo will throw a warning if you attempt to access a `.File` property. For example: - -```text -WARN .File.ContentBaseName on zero object. Wrap it in if or with... -``` - -To code defensively: - -```go-html-template -{{ with .File }} - {{ .ContentBaseName }} -{{ end }} -``` diff --git a/content/en/variables/git.md b/content/en/variables/git.md index 8928ca6f0ac..6e430b8d628 100644 --- a/content/en/variables/git.md +++ b/content/en/variables/git.md @@ -1,71 +1,18 @@ --- title: Git variables -linkTitle: Git variables -description: Get the last Git revision information for every content file. -categories: [variables and parameters] -keywords: [git] +description: Retrieve Git information related to the last commit of any page. +categories: [variables] +keywords: [] menu: docs: parent: variables - weight: 70 -weight: 70 + weight: 30 +weight: 30 aliases: [/extras/gitinfo/] --- -{{% note %}} -Hugo's Git integrations should be fairly performant but *can* increase your build time. This will depend on the size of your Git history. -{{% /note %}} +{{% include "variables/_common/consistent-terminology.md" %}} -## `.GitInfo` prerequisites +See the documentation for the [`GitInfo`] method on a `Page` object. -1. The Hugo site must be in a Git-enabled directory. -2. The Git executable must be installed and in your system `PATH`. -3. The `.GitInfo` feature must be enabled in your Hugo project by passing `--enableGitInfo` flag on the command line or by setting `enableGitInfo` to `true` in your [site's configuration file][configuration]. - -## The `.GitInfo` object - -The `GitInfo` object contains the following fields: - -.AbbreviatedHash -: the abbreviated commit hash (e.g., `866cbcc`) - -.AuthorName -: the author's name, respecting [`.mailmap`](https://git-scm.com/docs/gitmailmap) - -.AuthorEmail -: the author's email address, respecting [`.mailmap`](https://git-scm.com/docs/gitmailmap) - -.AuthorDate -: the author date - -.Hash -: the commit hash (e.g., `866cbccdab588b9908887ffd3b4f2667e94090c3`) - -.Subject -: commit message subject (e.g., `tpl: Add custom index function`) - -## `.Lastmod` - -If the `.GitInfo` feature is enabled, `.Lastmod` (on `Page`) is fetched from Git i.e. `.GitInfo.AuthorDate`. This behavior can be changed by adding your own [front matter configuration for dates](/getting-started/configuration/#configure-front-matter). - -[configuration]: /getting-started/configuration/ - -## Hosting considerations - -On the site host, your repository must be "deep-cloned," so the returned `.GitInfo` data will be accurate. Otherwise, your site may display only data from your latest commit. Where it's not possible to configure a host's cloning depth, you must handle this through CI/CD (*e.g.*, a GitHub Action or GitLab CI/CD). See the following table: - -| Hosting service | Clone depth | Configurable? | -| :-------------- | :---------- | :-----------: | -| Cloudflare Pages | Shallow | ✔️ [^CFP] | -| DigitalOcean App Platform | Deep | ❌ | -| GitHub Pages | Shallow | ✔️ [^GHP] | -| GitLab Pages | Shallow | ✔️ [^GLP] | -| Netlify | Deep | ❌ | -| Render | Shallow | ❌ | -| Vercel | Shallow | ❌ | - -[^CFP]: To configure a Cloudflare Pages site for deep cloning, preface the site's normal Hugo build command with `git fetch --unshallow &&` (*e.g.*, `git fetch --unshallow && hugo`). - -[^GHP]: You can configure the GitHub Action to do a deep clone by specifying `fetch-depth: 0` in the applicable "checkout" step of your workflow file, as shown in the Hugo documentation's [example workflow file](/hosting-and-deployment/hosting-on-github/#procedure). - -[^GLP]: You can configure the GitLab Runner's clone depth [as explained in the GitLab documentation](https://docs.gitlab.com/ee/ci/large_repositories/#shallow-cloning); see also the Hugo documentation's [example workflow file](https://gohugo.io/hosting-and-deployment/hosting-on-gitlab/#configure-gitlab-cicd). +[`GitInfo`]: /methods/page/gitinfo diff --git a/content/en/variables/menus.md b/content/en/variables/menu-entry.md similarity index 81% rename from content/en/variables/menus.md rename to content/en/variables/menu-entry.md index 0fe727cd88d..27435ca2e02 100644 --- a/content/en/variables/menus.md +++ b/content/en/variables/menu-entry.md @@ -1,26 +1,38 @@ --- -title: Menu variables +title: Menu entry variables description: Use these variables and methods in your menu templates. -categories: [variables and parameters] +categories: [variables] keywords: [menus] menu: docs: parent: variables - weight: 50 -weight: 50 -aliases: [/variables/menu/] + weight: 40 +weight: 40 +toc: true +aliases: [/variables/menus/] --- -## Variables +{{% include "variables/_common/consistent-terminology.md" %}} -After [defining menu entries], access their properties in [menu templates] with these variables. +After [defining menu entries], access their properties in [menu templates] with these methods. + +## Methods .Children : (`menu`) A collection of child menu entries, if any, under the current menu entry. +.HasChildren +: (`bool`) Returns `true` if `.Children` is non-nil. + .Identifier : (`string`) The `identifier` property of the menu entry. If you define the menu entry [automatically], the page's `.Section`. +.IsEqual +: (`bool`) Returns `true` if the compared menu entries represent the same menu entry. + +.IsSameResource +: (`bool`) Returns `true` if the compared menu entries point to the same resource. + .KeyName : (`string`) The `identifier` property of the menu entry, else the `name` property. @@ -36,11 +48,6 @@ After [defining menu entries], access their properties in [menu templates] with .Page : (`page`) A reference to the page associated with the menu entry. - - .Params : (`map`) The `params` property of the menu entry. @@ -68,22 +75,9 @@ After [defining menu entries], access their properties in [menu templates] with - If you define the menu entry [automatically], the page's `.Weight`. - If you define the menu [in front matter] or [in site configuration], falls back to the page's `.Weight`. -## Methods - -.HasChildren -: (`bool`) Returns `true` if `.Children` is non-nil. - -.IsEqual -: (`bool`) Returns `true` if the compared menu entries represent the same menu entry. - -.IsSameResource -: (`bool`) Returns `true` if the compared menu entries point to the same resource. - -.Page.HasMenuCurrent -: (`bool`) Use this method to determine ancestors of the active menu entry. See [details](/functions/hasmenucurrent/). +## Related methods -.Page.IsMenuCurrent -: (`bool`) Use this method to determine the active menu entry. See [details](/functions/ismenucurrent/). +{{< list-pages-in-section path=/methods/page filter=methods_page_menu filterType=include titlePrefix=.Page. >}} [automatically]: /content-management/menus/#define-automatically [defining menu entries]: /content-management/menus/#overview diff --git a/content/en/variables/page.md b/content/en/variables/page.md index 16e5f31ffa1..8ee911007ec 100644 --- a/content/en/variables/page.md +++ b/content/en/variables/page.md @@ -1,357 +1,63 @@ --- title: Page variables -description: Page-level variables are defined in a content file's front matter, derived from the content's file location, or extracted from the content body itself. -categories: [variables and parameters] -keywords: [pages] +description: Use these methods with a Page object. +categories: [variables] +keywords: [page] menu: docs: parent: variables - weight: 20 -weight: 20 + weight: 50 +weight: 50 toc: true --- -The following is a list of page-level variables. Many of these will be defined in the front matter, derived from file location, or extracted from the content itself. +{{% include "variables/_common/consistent-terminology.md" %}} -## Page variables +## All methods -.AlternativeOutputFormats -: Contains all alternative formats for a given page; this variable is especially useful `link rel` list in your site's ``. (See [Output Formats](/templates/output-formats/).) +Use any of these methods in your templates. -.Aliases -: Aliases of this page +{{< list-pages-in-section path=/methods/page titlePrefix=. >}} -.Ancestors -: Ancestors of this page. +## Dates -.BundleType -: The [bundle] type: `leaf`, `branch`, or an empty string if the page is not a bundle. +Use these methods to access content dates. -.Content -: The content itself, defined below the front matter. +{{< list-pages-in-section path=/methods/page filter=methods_page_dates filterType=include titlePrefix=. omitElementIDs=true >}} -.Data -: The data specific to this type of page. +## Multilingual -.Date -: The date associated with the page. By default, this is the front matter `date` value. See [configuring dates] for a description of fallback values and precedence. See also `.ExpiryDate`, `.Lastmod`, and `.PublishDate`. +Use these methods with your multilingual projects. -.Description -: The description for the page. +{{< list-pages-in-section path=/methods/page filter=methods_page_multilinugual filterType=include titlePrefix=. omitElementIDs=true >}} -.Draft -: A boolean, `true` if the content is marked as a draft in the front matter. +## Navigation -.ExpiryDate -: The date on which the content is scheduled to expire. By default, this is the front matter `expiryDate` value. See [configuring dates] for a description of fallback values and precedence. See also `.Date`, `.Lastmod`, and `.PublishDate`. +Use these methods to create navigation links between pages. -.File -: Filesystem-related data for this content file. See also [File Variables]. - -.Fragments -: Fragments returns the fragments for this page. See [Page Fragments](#page-fragments). - -.FuzzyWordCount -: The approximate number of words in the content. - -.IsHome -: `true` in the context of the [homepage](/templates/homepage/). - -.IsNode -: Always `false` for regular content pages. - -.IsPage -: Always `true` for regular content pages. - -.IsSection -: `true` if [`.Kind`](/templates/section-templates/#page-kinds) is `section`. - -.IsTranslated -: `true` if there are translations to display. - -.Keywords -: The meta keywords for the content. - -.Kind -: The page's *kind*. Possible return values are `page`, `home`, `section`, `taxonomy`, or `term`. Note that there are also `RSS`, `sitemap`, `robotsTXT`, and `404` kinds, but these are only available during the rendering of each of these respective page's kind and therefore *not* available in any of the `Pages` collections. - -.Language -: A language object that points to the language's definition in the site configuration. `.Language.Lang` gives you the language code. - -.Lastmod -: The date on which the content was last modified. By default, if `enableGitInfo` is `true` in your site configuration, this is the Git author date, otherwise the front matter `lastmod` value. See [configuring dates] for a description of fallback values and precedence. See also `.Date`,`ExpiryDate`, `.PublishDate`, and [`.GitInfo`][gitinfo]. - -.LinkTitle -: Access when creating links to the content. If set, Hugo will use the `linktitle` from the front matter before `title`. - -.Next -: Points up to the next regular page (sorted by Hugo's [default sort](/templates/lists#default-weight--date--linktitle--filepath)). Example: `{{ with .Next }}{{ .Permalink }}{{ end }}`. Calling `.Next` from the first page returns `nil`. - -.NextInSection -: Points up to the next regular page below the same top level section (e.g. in `/blog`)). Pages are sorted by Hugo's [default sort](/templates/lists#default-weight--date--linktitle--filepath). Example: `{{ with .NextInSection }}{{ .Permalink }}{{ end }}`. Calling `.NextInSection` from the first page returns `nil`. - -.OutputFormats -: Contains all formats, including the current format, for a given page. Can be combined the with [`.Get` function](/functions/get/) to grab a specific format. (See [Output Formats](/templates/output-formats/).) - -.Permalink -: The Permanent link for this page; see [Permalinks](/content-management/urls/) - -.Plain -: The Page content stripped of HTML tags and presented as a string. You may need to pipe the result through the [`htmlUnescape`](/functions/transform/htmlunescape) function when rendering this value with the HTML [output format](/templates/output-formats#output-format-definitions). - -.PlainWords -: The slice of strings that results from splitting .Plain into words, as defined in Go's [strings.Fields](https://pkg.go.dev/strings#Fields). - -.Prev -: Points down to the previous regular page(sorted by Hugo's [default sort](/templates/lists#default-weight--date--linktitle--filepath)). Example: `{{ if .Prev }}{{ .Prev.Permalink }}{{ end }}`. Calling `.Prev` from the last page returns `nil`. - -.PrevInSection -: Points down to the previous regular page below the same top level section (e.g. `/blog`). Pages are sorted by Hugo's [default sort](/templates/lists#default-weight--date--linktitle--filepath). Example: `{{ if .PrevInSection }}{{ .PrevInSection.Permalink }}{{ end }}`. Calling `.PrevInSection` from the last page returns `nil`. - -.PublishDate -: The date on which the content was or will be published. By default, this is the front matter `publishDate` value. See [configuring dates] for a description of fallback values and precedence. See also `.Date`, `.ExpiryDate`, and `.Lastmod`. - -.RawContent -: Raw markdown content without the front matter. Useful with [remarkjs.com]( -https://remarkjs.com) - -.RenderShortcodes -: See [Render Shortcodes](#rendershortcodes). - -.ReadingTime -: The estimated time, in minutes, it takes to read the content. - -.Resources -: Resources such as images and CSS that are associated with this page - -.Ref -: Returns the permalink for a given reference (e.g., `.Ref "sample.md"`). `.Ref` does *not* handle in-page fragments correctly. See [Cross References](/content-management/cross-references/). - -.RelPermalink -: The relative permanent link for this page. - -.RelRef -: Returns the relative permalink for a given reference (e.g., `RelRef -"sample.md"`). `.RelRef` does *not* handle in-page fragments correctly. See [Cross References](/content-management/cross-references/). - -.Site -: See [Site Variables](/variables/site/). - -.Sites -: Returns all sites (languages). A typical use case would be to link back to the main language: `...`. - -.Sites.First -: Returns the site for the first language. If this is not a multilingual setup, it will return itself. - -.Summary -: A generated summary of the content for easily showing a snippet in a summary view. The breakpoint can be set manually by inserting <!--more--> at the appropriate place in the content page, or the summary can be written independent of the page text. See [Content Summaries](/content-management/summaries/) for more details. - -.TableOfContents -: The rendered [table of contents](/content-management/toc/) for the page. - -.Title -: The title for this page. - -.Translations -: A list of translated versions of the current page. See [Multilingual Mode](/content-management/multilingual/) for more information. - -.TranslationKey -: The key used to map language translations of the current page. See [Multilingual Mode](/content-management/multilingual/) for more information. - -.Truncated -: A boolean, `true` if the `.Summary` is truncated. Useful for showing a "Read more..." link only when necessary. See [Summaries](/content-management/summaries/) for more information. - -.Type -: The [content type](/content-management/types/) of the content (e.g., `posts`). - -.Weight -: Assigned weight (in the front matter) to this content, used in sorting. - -.WordCount -: The number of words in the content. +{{< list-pages-in-section path=/methods/page filter=methods_page_navigation filterType=include titlePrefix=. omitElementIDs=true >}} ## Page collections -List pages receive the following page collections in [context](/getting-started/glossary/#context): - -.Pages -: Regular pages within the current section (not recursive), and section pages for immediate descendant sections (not recursive). - -.RegularPages -: Regular pages within the current section (not recursive). - -.RegularPagesRecursive -: Regular pages within the current section, and regular pages within all descendant sections. - -## Writable page-scoped variables - -[.Scratch][scratch] -: Returns a Scratch to store and manipulate data. In contrast to the [`.Store`][store] method, this scratch is reset on server rebuilds. - -[.Store][store] -: Returns a Scratch to store and manipulate data. In contrast to the [`.Scratch`][scratch] method, this scratch is not reset on server rebuilds. - -## Section variables and methods - -Also see [Sections](/content-management/sections/). - -.CurrentSection -: The page's current section. The value can be the page itself if it is a section or the homepage. - -.FirstSection -: The page's first section below root, e.g. `/docs`, `/blog` etc. - -.InSection $anotherPage -: Whether the given page is in the current section. - -.IsAncestor $anotherPage -: Whether the current page is an ancestor of the given page. - -.IsDescendant $anotherPage -: Whether the current page is a descendant of the given page. - -.Parent -: A section's parent section or a page's section. - -.Section -: The [section](/content-management/sections/) this content belongs to. **Note:** For nested sections, this is the first path element in the directory, for example, `/blog/funny/mypost/ => blog`. - -.Sections -: The [sections](/content-management/sections/) below this content. - -## Page fragments - -{{< new-in "0.111.0" >}} - -The `.Fragments` method returns a list of fragments for the current page. - -.Headings -: A recursive list of headings for the current page. Can be used to generate a table of contents. - -{{< todo >}}add .Headings toc example{{< /todo >}} - -.Identifiers -: A sorted list of identifiers for the current page. Can be used to check if a page contains a specific identifier or if a page contains duplicate identifiers: - -```go-html-template -{{ if .Fragments.Identifiers.Contains "my-identifier" }} -

    Page contains identifier "my-identifier"

    -{{ end }} - -{{ if gt (.Fragments.Identifiers.Count "my-identifier") 1 }} -

    Page contains duplicate "my-identifier" fragments

    -{{ end }} -``` - -.HeadingsMap -: Holds a map of headings for the current page. Can be used to start the table of contents from a specific heading. - -Also see the [Go Doc](https://pkg.go.dev/github.com/gohugoio/hugo@v0.111.0/markup/tableofcontents#Fragments) for the return type. - -### Fragments in hooks and shortcodes - -`.Fragments` are safe to call from render hooks, even on the page you're on (`.Page.Fragments`). For shortcodes we recommend that all `.Fragments` usage is nested inside the `{{}}` shortcode delimiter (`{{%/**/%}}` takes part in the ToC creation so it's easy to end up in a situation where you bite yourself in the tail). - -## The `.RenderShortcodes` method {#rendershortcodes} - -{{< new-in "0.117.0" >}} This renders all the shortcodes in the content, preserving the surrounding markup (e.g. Markdown) as is. - -The common use case this is to composing a page from multiple content files while preserving a global context for table of contents and foot notes. - -This method is most often used in shortcode templates. A simple example of shortcode template including content from another page would look like: - -```go-html-template -{{ $p := site.GetPage (.Get 0) }} -{{ $p.RenderShortcodes }} -``` - -In the above it's important to understand and the difference between the two delimiters used when including a shortcode: - -* `{{}}` tells Hugo that the rendered shortcode does not need further processing (e.g. it's HTML). -* `{{%/* myshortcode */%}}` tells Hugo that the rendered shortcode needs further processing (e.g. it's Markdown). - -The latter is what you want to use for the include shortcode outlined above: - -```md -## Mypage -{{%/* include "mypage" */%}} -`````` - - -Also see [Use Shortcodes](/content-management/shortcodes/#use-shortcodes). - -## Page-level params - -Any other value defined in the front matter in a content file, including taxonomies, will be made available as part of the `.Params` variable. - -{{< code-toggle file="content/example.md" fm=true copy=false >}} -title: Example -categories: [one] -tags: [two,three,four] -{{< /code-toggle >}} - -With the above front matter, the `tags` and `categories` taxonomies are accessible via the following: - -* `.Params.tags` -* `.Params.categories` - -The `.Params` variable is particularly useful for the introduction of user-defined front matter fields in content files. For example, a Hugo website on book reviews could have the following front matter: - -{{< code-toggle file="content/example.md" fm=true copy=false >}} -title: Example -affiliatelink: "http://www.my-book-link.here" -recommendedby: "My Mother" -{{< /code-toggle >}} - -These fields would then be accessible to via `.Params.affiliatelink` and `.Params.recommendedby`. - -```go-html-template -

    Buy this book

    -

    It was recommended by {{ .Params.recommendedby }}.

    -``` - -This template would render as follows: - -```html -

    Buy this book

    -

    It was recommended by my Mother.

    -``` - -{{% note %}} -See [Archetypes](/content-management/archetypes/) for consistency of `Params` across pieces of content. -{{% /note %}} - -### The `.Param` method +Range through these collections when rendering lists on [section] pages, [taxonomy] pages, [term] pages, and the home page. -In Hugo, you can declare parameters in individual pages and globally for your entire website. A common use case is to have a general value for the site parameter and a more specific value for some of the pages (i.e., a header image): +[section]: /getting-started/glossary/#section +[taxonomy]: /getting-started/glossary/#taxonomy +[term]: /getting-started/glossary/#term +[context]: /getting-started/glossary/#context -```go-html-template -{{ $.Param "header_image" }} -``` +{{< list-pages-in-section path=/methods/page filter=methods_page_page_collections filterType=include titlePrefix=. omitElementIDs=true >}} -The `.Param` method provides a way to resolve a single value according to it's definition in a page parameter (i.e. in the content's front matter) or a site parameter (i.e., in your site configuration). +## Parameters -### Access nested fields in front matter +Use these methods to access page parameters. -When front matter contains nested fields like the following: +{{< list-pages-in-section path=/methods/page filter=methods_page_parameters filterType=include titlePrefix=. omitElementIDs=true >}} -{{< code-toggle file="content/example.md" fm=true copy=false >}} -title: Example -author: - given_name: John - family_name: Feminella - display_name: John Feminella -{{< /code-toggle >}} +## Sections -`.Param` can access these fields by concatenating the field names together with a dot: +Use these methods to access section pages, and their ancestors and descendants. See [details]. -```go-html-template -{{ $.Param "author.display_name" }} -``` +[details]: /content-management/sections/ -[configuring dates]: /getting-started/configuration/#configure-dates -[gitinfo]: /variables/git/ -[File Variables]: /variables/files/ -[bundle]: /content-management/page-bundles -[scratch]: /functions/scratch -[store]: /functions/store +{{< list-pages-in-section path=/methods/page filter=methods_page_sections filterType=include titlePrefix=. omitElementIDs=true >}} diff --git a/content/en/variables/pages.md b/content/en/variables/pages.md index 3917a6558c9..2f8e79d72fd 100644 --- a/content/en/variables/pages.md +++ b/content/en/variables/pages.md @@ -1,25 +1,39 @@ --- -title: Pages methods -description: Pages is the core page collection in Hugo and has many useful methods. -categories: [variables and parameters] +title: Pages variables +description: Use these methods with a collection of Page objects. +categories: [variables] keywords: [pages] menu: docs: parent: variables - weight: 21 -weight: 21 -aliases: [/pages] + weight: 60 +weight: 60 +aliases: [/variables/site-variables/] toc: true --- -Also see [List templates](/templates/lists) for an overview of sort methods. +{{% include "variables/_common/consistent-terminology.md" %}} -## .Next PAGE +## All methods -`.Next` and `.Prev` on `Pages` work similar to the methods with the same names on `.Page`, but are more flexible (and slightly slower) as they can be used on any page collection. +Use any of these methods with page collections in your templates. -`.Next` points **up** to the next page relative to the page sent in as the argument. Example: `{{ with .Site.RegularPages.Next . }}{{ .RelPermalink }}{{ end }}`. Calling `.Next` with the first page in the collection returns `nil`. +{{< list-pages-in-section path=/methods/pages titlePrefix=. >}} -## .Prev PAGE +## Sort by -`.Prev` points **down** to the previous page relative to the page sent in as the argument. Example: `{{ with .Site.RegularPages.Prev . }}{{ .RelPermalink }}{{ end }}`. Calling `.Prev` with the last page in the collection returns `nil`. +Use these methods to sort page collections. + +{{< list-pages-in-section path=/methods/pages filter=methods_pages_sort filterType=include titlePrefix=. omitElementIDs=true >}} + +## Group by + +Use these methods to group page collections. + +{{< list-pages-in-section path=/methods/pages filter=methods_pages_group filterType=include titlePrefix=. omitElementIDs=true >}} + +## Navigation + +Use these methods to create navigation links between pages. + +{{< list-pages-in-section path=/methods/pages filter=methods_pages_navigation filterType=include titlePrefix=. omitElementIDs=true >}} diff --git a/content/en/variables/shortcode.md b/content/en/variables/shortcode.md new file mode 100644 index 00000000000..51bf0c00cf3 --- /dev/null +++ b/content/en/variables/shortcode.md @@ -0,0 +1,16 @@ +--- +title: Shortcode variables +description: Use these methods in your shortcode templates. +categories: [variables] +keywords: [shortcodes] +menu: + docs: + parent: variables + weight: 70 +weight: 70 +aliases: [/variables/shortcodes] +--- + +{{% include "variables/_common/consistent-terminology.md" %}} + +{{< list-pages-in-section path=/methods/shortcode titlePrefix=. >}} diff --git a/content/en/variables/shortcodes.md b/content/en/variables/shortcodes.md deleted file mode 100644 index a03485d6fb4..00000000000 --- a/content/en/variables/shortcodes.md +++ /dev/null @@ -1,45 +0,0 @@ ---- -title: Shortcode variables -description: Shortcodes can access page variables and also have their own specific built-in variables. -categories: [variables and parameters] -keywords: [shortcodes] -menu: - docs: - parent: variables - weight: 20 -weight: 20 ---- - -[Shortcodes][shortcodes] have access to parameters delimited in the shortcode declaration via [`.Get`][getfunction], page- and site-level variables, and also the following shortcode-specific fields: - -.Name -: Shortcode name. - -.Ordinal -: Zero-based ordinal in relation to its parent. If the parent is the page itself, this ordinal will represent the position of this shortcode in the page content. - -.Page -: The owning ´Page`. - -.Parent -: provides access to the parent shortcode context in nested shortcodes. This can be very useful for inheritance of common shortcode parameters from the root. - -.Position -: Contains [file name and position](https://godoc.org/github.com/gohugoio/hugo/common/text#Position) for the shortcode in a page. Note that this can be relatively expensive to calculate, and is meant for error reporting. See [Error Handling in Shortcodes](/templates/shortcode-templates/#error-handling-in-shortcodes). - -.IsNamedParams -: boolean that returns `true` when the shortcode in question uses [named rather than positional parameters][shortcodes] - -.Inner -: represents the content between the opening and closing shortcode tags when a [closing shortcode][markdownshortcode] is used - -.Scratch -: returns a writable [`Scratch`][scratch] to store and manipulate data which will be attached to the shortcode context. This scratch is reset on server rebuilds. - -.InnerDeindent {{< new-in "0.100.0" >}} -: Gets the `.Inner` with any indentation removed. This is what's used in the built-in `{{}}` shortcode. - -[getfunction]: /functions/get/ -[markdownshortcode]: /content-management/shortcodes/#shortcodes-with-markdown -[shortcodes]: /templates/shortcode-templates/ -[scratch]: /functions/scratch diff --git a/content/en/variables/site.md b/content/en/variables/site.md index 2b9402c5fa2..3e0e964cbce 100644 --- a/content/en/variables/site.md +++ b/content/en/variables/site.md @@ -1,108 +1,55 @@ --- title: Site variables -description: Many, but not all, site-wide variables are defined in your site's configuration. However, Hugo provides a number of built-in variables for convenient access to global values in your templates. -categories: [variables and parameters] -keywords: [global,site] +description: Use these methods with Site objects. A multilingual project will have two or more sites, one for each language. +categories: [variables] +keywords: [site] menu: docs: parent: variables - weight: 10 -weight: 10 + weight: 80 +weight: 80 aliases: [/variables/site-variables/] toc: true --- -The following is a list of site-level (aka "global") variables. Many of these variables are defined in your site's [configuration file][config], whereas others are built into Hugo's core for convenient usage in your templates. +{{% include "variables/_common/consistent-terminology.md" %}} -## Get the site object from a partial +## All methods -All the methods below, e.g. `.Site.RegularPages` can also be reached via the global [`site`](/functions/site/) function, e.g. `site.RegularPages`, which can be handy in partials where the `Page` object isn't easily available. +Use any of these methods in your templates. -## Site variables +{{< list-pages-in-section path=/methods/site titlePrefix=.Site. >}} -.Site.AllPages -: array of all pages, regardless of their translation. +## Multilingual -.Site.BaseURL -: the base URL for the site as defined in the site configuration. +Use these methods with your multilingual projects. -.Site.BuildDrafts -: a boolean (default: `false`) to indicate whether to build drafts as defined in the site configuration. +{{< list-pages-in-section path=/methods/site filter=methods_site_multilingual filterType=include titlePrefix=.Site. omitElementIDs=true >}} -.Site.Copyright -: a string representing the copyright of your website as defined in the site configuration. +[`site`]: /functions/global/site +[context]: /getting-started/glossary/#context +[configuration file]: /getting-started/configuration -.Site.Data -: custom data, see [Data Templates](/templates/data-templates/). +## Page collections -.Site.Home -: reference to the homepage's [page object](/variables/page/) +Range through these collections when rendering lists on any page. -.Site.IsMultiLingual -: whether there are more than one language in this site. See [Multilingual](/content-management/multilingual/) for more information. +{{< list-pages-in-section path=/methods/site filter=methods_site_page_collections filterType=include titlePrefix=.Site. omitElementIDs=true >}} -.Site.Language.Lang -: the language code of the current locale (e.g., `en`). +## Global site function -.Site.Language.LanguageName -: the full language name (e.g. `English`). +Within a partial template, if you did not pass a `Page` or `Site` object in [context], you cannot use this syntax: -.Site.Language.Weight -: the weight that defines the order in the `.Site.Languages` list. +```go-html-template +{{ .Site.SomeMethod }} +``` -.Site.Language -: indicates the language currently being used to render the website. This object's attributes are set in site configurations' language definition. +Instead, use the global [`site`] function: -.Site.LanguageCode -: a string representing the language tag as defined in the site configuration. +```go-html-template +{{ site.SomeMethod }} +``` -.Site.LanguagePrefix -: this can be used to prefix URLs to point to the correct language. It will even work when there is only one defined language. See also the functions [absLangURL](/functions/urls/abslangurl) and [relLangURL](/functions/urls/rellangurl). - -.Site.Languages -: an ordered list (ordered by defined weight) of languages. - -.Site.LastChange -: a [time.Time](https://godoc.org/time#Time) value representing the date/time of the most recent change to your site. - -.Site.Menus -: all the menus in the site. - -.Site.Pages -: array of all content ordered by Date with the newest first. This array contains only the pages in the current language. - -.Site.RegularPages -: a shortcut to the *regular* page collection. `.Site.RegularPages` is equivalent to `where .Site.Pages "Kind" "page"`. - -.Site.Sections -: top-level directories of the site. - -.Site.Taxonomies -: the [taxonomies](/content-management/taxonomies/) for the entire site. Also see section [Access taxonomy data from any template](/variables/taxonomy/#access-taxonomy-data-from-any-template). - -.Site.Title -: a string representing the title of the site. - -## Site parameters - -`.Site.Params` is a container holding the values from the `params` section of your site configuration. - -### Example: `.Site.Params` - -The following `config.[yaml|toml|json]` defines a site-wide parameter for `description`: - -{{< code-toggle file="hugo" >}} -baseURL = "https://yoursite.example.com/" - -[params] - description = "Tesla's Awesome Hugo Site" - author = "Nikola Tesla" -{{}} - -You can use `.Site.Params` in a [partial template](/templates/partials/) to call the default site description: - -{{< code file="layouts/partials/head.html" >}} - -{{< /code >}} - -[config]: /getting-started/configuration/ +{{% note %}} +You can use the global site function in all templates to avoid context problems. Its usage is not limited to partial templates. +{{% /note %}} diff --git a/content/en/variables/sitemap.md b/content/en/variables/sitemap.md deleted file mode 100644 index 80fb1107681..00000000000 --- a/content/en/variables/sitemap.md +++ /dev/null @@ -1,24 +0,0 @@ ---- -title: Sitemap variables -description: -categories: [variables and parameters] -keywords: [sitemap] -menu: - docs: - parent: variables - weight: 80 -weight: 80 ---- - -A sitemap is a `Page` and therefore has all the [page variables][pagevars] available to use sitemap templates. They also have the following sitemap-specific variables available to them: - -.Sitemap.ChangeFreq -: the page change frequency - -.Sitemap.Priority -: the priority of the page - -.Sitemap.Filename -: the sitemap file name - -[pagevars]: /variables/page/ diff --git a/content/en/variables/taxonomy.md b/content/en/variables/taxonomy.md index 2c1e051a912..1637337e323 100644 --- a/content/en/variables/taxonomy.md +++ b/content/en/variables/taxonomy.md @@ -1,138 +1,21 @@ --- title: Taxonomy variables -description: Hugo's taxonomy system exposes variables to taxonomy and term templates. -categories: [variables and parameters] +description: Use these methods with Taxonomy objects. +categories: [variables] keywords: [taxonomy,term] menu: docs: parent: variables - weight: 30 -toc: true -weight: 30 + weight: 90 +weight: 90 --- -## Taxonomy templates +{{% include "variables/_common/consistent-terminology.md" %}} -Pages rendered by taxonomy templates have `.Kind` set to `taxonomy` and `.Type` set to the taxonomy name. +{{< list-pages-in-section path=/methods/taxonomy titlePrefix=. >}} -In taxonomy templates you may access `.Site`, `.Page`. `.Section`, and `.File` variables, as well as the following _taxonomy_ variables: +{{% note %}} +Within a taxonomy or term template use the [`Data`] method to retrieve information specific to the taxonomy or term. -.Data.Singular -: The singular name of the taxonomy (e.g., `tags => tag`). - -.Data.Plural -: The plural name of the taxonomy (e.g., `tags => tags`). - -.Data.Pages -: The collection of term pages related to this taxonomy. Aliased by `.Pages`. - -.Data.Terms -: A map of terms and weighted pages related to this taxonomy. - -.Data.Terms.Alphabetical -: A map of terms and weighted pages related to this taxonomy, sorted alphabetically in ascending order. Reverse the sort order with`.Data.Terms.Alphabetical.Reverse`. - -.Data.Terms.ByCount -: A map of terms and weighted pages related to this taxonomy, sorted by count in ascending order. Reverse the sort order with`.Data.Terms.ByCount.Reverse`. - -## Term templates - -Pages rendered by term templates have `.Kind` set to `term` and `.Type` set to the taxonomy name. - -In term templates you may access `.Site`, `.Page`. `.Section`, and `.File` variables, as well as the following _term_ variables: - -.Data.Singular -: The singular name of the taxonomy (e.g., `tags => tag`). - -.Data.Plural -: The plural name of the taxonomy (e.g., `tags => tags`). - -.Data.Pages -: The collection of content pages related to this taxonomy. Aliased by `.Pages`. - -.Data.Term -: The term itself (e.g., `tag-one`). - -## Access taxonomy data from any template - -Access the entire taxonomy data structure from any template with `site.Taxonomies`. This returns a map of taxonomies, terms, and a collection of weighted content pages related to each term. For example: - -```json -{ - "categories": { - "news": [ - { - "Weight": 0, - "Page": { - "Title": "Post 1", - "Date": "2022-12-18T15:13:35-08:00" - ... - } - }, - { - "Weight": 0, - "Page": { - "Title": "Post 2", - "Date": "2022-12-18T15:13:46-08:00", - ... - } - } - ] - }, - "tags": { - "international": [ - { - "Weight": 0, - "Page": { - "Title": "Post 1", - "Date": "2021-01-01T00:00:00Z" - ... - } - } - ] - } -} -``` - -Access a subset of the taxonomy data structure by chaining one or more identifiers, or by using the [`index`] function with one or more keys. For example, to access the collection of weighted content pages related to the news category, use either of the following: - -[`index`]: /functions/collections/indexfunction/ - -```go-html-template -{{ $pages := site.Taxonomies.categories.news }} -{{ $pages := index site.Taxonomies "categories" "news" }} -``` - -For example, to render the entire taxonomy data structure as a nested unordered list: - -```go-html-template -
      - {{ range $taxonomy, $terms := site.Taxonomies }} -
    • - {{ with site.GetPage $taxonomy }} - {{ .LinkTitle }} - {{ end }} -
        - {{ range $term, $weightedPages := $terms }} -
      • - {{ with site.GetPage (path.Join $taxonomy $term) }} - {{ .LinkTitle }} - {{ end }} -
      • - - {{ end }} -
      -
    • - {{ end }} -
    -``` - -See [Taxonomy Templates] for more examples. - -[Taxonomy Templates]: /templates/taxonomy-templates/ +[`Data`]: /methods/page/data +{{% /note %}} diff --git a/data/page_filters.yaml b/data/page_filters.yaml new file mode 100644 index 00000000000..b2810d7b48c --- /dev/null +++ b/data/page_filters.yaml @@ -0,0 +1,89 @@ +# Do not delete. Required for layouts/shortcodes/list-pages-in-section.html. +# +# When calling the list-pages-in-section shortcode, you can specify a page +# filter, and whether the pages in the filter should be included or excluded +# from the list. +# +# For example: +# +# {{< list-pages-in-section path=/functions/images filter=functions_images_no_filters filterType=exclude >}} + +functions_images_no_filters: + - /functions/images/filter + - /functions/images/config +methods_site_multilingual: + - /methods/site/ismultilingual + - /methods/site/language + - /methods/site/languageprefix + - /methods/site/languages +methods_site_page_collections: + - /methods/site/pages + - /methods/site/regularpages + - /methods/site/sections +methods_page_dates: + - /methods/page/date + - /methods/page/expirydate + - /methods/page/lastmod + - /methods/page/publishdate +methods_page_menu: + - /methods/page/hasmenucurrent + - /methods/page/ismenucurrent +methods_page_multilinugual: + - /methods/page/alltranslations + - /methods/page/istranslated + - /methods/page/language + - /methods/page/translationkey + - /methods/page/translations +methods_page_page_collections: + - /methods/page/pages + - /methods/page/regularpages + - /methods/page/regularpagesrecursive + - /methods/page/sections +methods_page_parameters: + - /methods/page/param + - /methods/page/params +methods_page_sections: + - /methods/page/ancestors + - /methods/page/currentsection + - /methods/page/firstsection + - /methods/page/insection + - /methods/page/isancestor + - /methods/page/isdescendant + - /methods/page/parent + - /methods/page/sections + - /methods/page/section +methods_pages_sort: + - /methods/pages/bydate + - /methods/pages/byexpirydate + - /methods/pages/bylanguage + - /methods/pages/bylastmod + - /methods/pages/bylength + - /methods/pages/bylinktitle + - /methods/pages/byparam + - /methods/pages/bypublishdate + - /methods/pages/bytitle + - /methods/pages/byweight + - /methods/pages/reverse +methods_pages_group: + - /methods/pages/groupby + - /methods/pages/groupbydate + - /methods/pages/groupbyexpirydate + - /methods/pages/groupbylastmod + - /methods/pages/groupbyparam + - /methods/pages/groupbyparamdate + - /methods/pages/groupbypublishdate + - /methods/pages/groupbydate + - /methods/pages/groupbydate + - /methods/pages/groupbydate + - /methods/pages/groupbydate + - /methods/pages/groupbydate + - /methods/pages/groupbydate + - /methods/pages/reverse +methods_pages_navigation: + - /methods/pages/next + - /methods/pages/prev +methods_page_navigation: + - /methods/page/next + - /methods/page/nextinsection + - /methods/page/prev + - /methods/page/previnsection diff --git a/data/titles.toml b/data/titles.toml deleted file mode 100644 index 2348c85611d..00000000000 --- a/data/titles.toml +++ /dev/null @@ -1,2 +0,0 @@ -[Showcase] -title = "Site Showcase" diff --git a/hugo.toml b/hugo.toml index 3126027a941..31e3cce2127 100644 --- a/hugo.toml +++ b/hugo.toml @@ -68,10 +68,6 @@ ID = 'G-MBZGKNMDWC' threshold = 80 includeNewer = true toLower = false - [[related.indices]] - toLower = true - name = "relatedFunctions" - weight = 60 [[related.indices]] name = "keywords" weight = 60 diff --git a/layouts/_default/_markup/render-link.html b/layouts/_default/_markup/render-link.html new file mode 100644 index 00000000000..aced7c8ec0d --- /dev/null +++ b/layouts/_default/_markup/render-link.html @@ -0,0 +1,250 @@ +{{- /* Last modified: 2023-09-04T09:23:04-07:00 */}} + +{{- /* +Copyright 2023 Veriphor LLC + +Licensed under the Apache License, Version 2.0 (the "License"); you may not +use this file except in compliance with the License. You may obtain a copy of +the License at + +https://www.apache.org/licenses/LICENSE-2.0 + +Unless required by applicable law or agreed to in writing, software +distributed under the License is distributed on an "AS IS" BASIS, WITHOUT +WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the +License for the specific language governing permissions and limitations under +the License. +*/}} + +{{- /* +This render hook resolves internal destinations by looking for a matching: + + 1. Content page + 2. Page resource (a file in the current page bundle) + 3. Section resource (a file in the current section) + 4. Global resource (a file in the assets directory) + +It skips the section resource lookup if the current page is a leaf bundle. + +External destinations are not modified. + +You must place global resources in the assets directory. If you have placed +your resources in the static directory, and you are unable or unwilling to move +them, you must mount the static directory to the assets directory by including +both of these entries in your site configuration: + + [[module.mounts]] + source = 'assets' + target = 'assets' + + [[module.mounts]] + source = 'static' + target = 'assets' + +By default, if this render hook is unable to resolve a destination, including a +fragment if present, it passes the destination through without modification. To +emit a warning or error, set the error level in your site configuration: + + [params.render_hooks.link] + errorLevel = 'warning' # ignore (default), warning, or error (fails the build) + +When you set the error level to warning, and you are in a development +environment, you can visually highlight broken internal links: + + [params.render_hooks.link] + errorLevel = 'warning' # ignore (default), warning, or error (fails the build) + highlightBroken = true # true or false (default) + +This will add a "broken" class to anchor elements with invalid src attributes. +Add a rule to your CSS targeting the broken links: + + a.broken { + background: #ff0; + border: 2px solid #f00; + padding: 0.1em 0.2em; + } + +This render hook may be unable to resolve destinations created with the ref and +relref shortcodes. Unless you set the error level to ignore you should not use +either of these shortcodes in conjunction with this render hook. + +@context {string} Destination The link destination. +@context {page} Page A reference to the page containing the link. +@context {string} PlainText The link description as plain text. +@context {string} Text The link description. +@context {string} Title The link title. + +@returns {template.html} +*/}} + +{{- /* Initialize. */}} +{{- $renderHookName := "link" }} + +{{- /* Verify minimum required version. */}} +{{- $minHugoVersion := "0.114.0" }} +{{- if lt hugo.Version $minHugoVersion }} + {{- errorf "The %q render hook requires Hugo v%s or later." $renderHookName $minHugoVersion }} +{{- end }} + +{{- /* Error level when unable to resolve destination: ignore, warning, or error. */}} +{{- $errorLevel := or site.Params.render_hooks.link.errorLevel "ignore" | lower }} + +{{- /* If true, adds "broken" class to broken links. Applicable in development environment when errorLevel is warning. */}} +{{- $highlightBrokenLinks := or site.Params.render_hooks.link.highlightBroken false }} + +{{- /* Validate error level. */}} +{{- if not (in (slice "ignore" "warning" "error") $errorLevel) }} + {{- errorf "The %q render hook is misconfigured. The errorLevel %q is invalid. Please check your site configuration." $renderHookName $errorLevel }} +{{- end }} + +{{- /* Determine content path for warning and error messages. */}} +{{- $contentPath := "" }} +{{- with .Page.File }} + {{- $contentPath = .Path }} +{{- else }} + {{- $contentPath = .Path }} +{{- end }} + +{{- /* Parse destination. */}} +{{- $u := urls.Parse .Destination }} + +{{- /* Set common message. */}} +{{- $msg := printf "The %q render hook was unable to resolve the destination %q in %s" $renderHookName $u.String $contentPath }} + +{{- /* Set attributes for anchor element. */}} +{{- $attrs := dict "href" $u.String }} +{{- if $u.IsAbs }} + {{- /* Destination is a remote resource. */}} + {{- $attrs = merge $attrs (dict "rel" "external") }} +{{- else }} + {{- with $u.Path }} + {{- with $p := or ($.Page.GetPage .) ($.Page.GetPage (strings.TrimRight "/" .)) }} + {{- /* Destination is a page. */}} + {{- $href := .RelPermalink }} + {{- with $u.RawQuery }} + {{- $href = printf "%s?%s" $href . }} + {{- end }} + {{- with $u.Fragment }} + {{- $ctx := dict + "contentPath" $contentPath + "errorLevel" $errorLevel + "page" $p + "parsedURL" $u + "renderHookName" $renderHookName + }} + {{- partial "inline/h-rh-l/validate-fragment.html" $ctx }} + {{- $href = printf "%s#%s" $href . }} + {{- end }} + {{- $attrs = dict "href" $href }} + {{- else }} + {{- with $.Page.Resources.Get $u.Path }} + {{- /* Destination is a page resource; drop query and fragment. */}} + {{- $attrs = dict "href" .RelPermalink }} + {{- else }} + {{- with (and (ne $.Page.BundleType "leaf") ($.Page.CurrentSection.Resources.Get $u.Path)) }} + {{- /* Destination is a section resource, and current page is not a leaf bundle. */}} + {{- $attrs = dict "href" .RelPermalink }} + {{- else }} + {{- with resources.Get $u.Path }} + {{- /* Destination is a global resource; drop query and fragment. */}} + {{- $attrs = dict "href" .RelPermalink }} + {{- else }} + {{- if eq $errorLevel "warning" }} + {{- warnf $msg }} + {{- if and $highlightBrokenLinks (eq hugo.Environment "development") }} + {{- $attrs = merge $attrs (dict "class" "broken") }} + {{- end }} + {{- else if eq $errorLevel "error" }} + {{- errorf $msg }} + {{- end }} + {{- end }} + {{- end }} + {{- end }} + {{- end }} + {{- else }} + {{- with $u.Fragment }} + {{- /* Destination is on the same page; prepend relative permalink. */}} + {{- $ctx := dict + "contentPath" $contentPath + "errorLevel" $errorLevel + "page" $.Page + "parsedURL" $u + "renderHookName" $renderHookName + }} + {{- partial "inline/h-rh-l/validate-fragment.html" $ctx }} + {{- $attrs = dict "href" (printf "%s#%s" $.Page.RelPermalink .) }} + {{- else }} + {{- if eq $errorLevel "warning" }} + {{- warnf $msg }} + {{- if and $highlightBrokenLinks (eq hugo.Environment "development") }} + {{- $attrs = merge $attrs (dict "class" "broken") }} + {{- end }} + {{- else if eq $errorLevel "error" }} + {{- errorf $msg }} + {{- end }} + {{- end }} + {{- end }} +{{- end }} +{{- with .Title }} + {{- $attrs = merge $attrs (dict "title" .) }} +{{- end -}} + +{{- /* Render anchor element. */ -}} +{{ .Text | safeHTML }} + +{{- define "partials/inline/h-rh-l/validate-fragment.html" }} + {{- /* + Validates the fragment portion of a link destination. + + @context {string} contentPath The page containing the link. + @context {srting} errorLevel The error level when unable to resolve destination; ignore (default), warning, or error. + @context {page} page The page corresponding to the link destination + @context {struct} parsedURL The link destination parsed by urls.Parse. + @context {string} renderHookName The name of the render hook. + */}} + + {{- /* Initialize. */}} + {{- $contentPath := .contentPath }} + {{- $errorLevel := .errorLevel }} + {{- $p := .page }} + {{- $u := .parsedURL }} + {{- $renderHookName := .renderHookName }} + + {{- /* Validate. */}} + {{- with $u.Fragment }} + {{- if $p.Fragments.Identifiers.Contains . }} + {{- if gt ($p.Fragments.Identifiers.Count .) 1 }} + {{- $msg := printf "The %q render hook detected duplicate heading IDs %q in %s" $renderHookName . $contentPath }} + {{- if eq $errorLevel "warning" }} + {{- warnf $msg }} + {{- else if eq $errorLevel "error" }} + {{- errorf $msg }} + {{- end }} + {{- end }} + {{- else }} + {{- /* Determine target path for warning and error message. */}} + {{- $targetPath := "" }} + {{- with $p.File }} + {{- $targetPath = .Path }} + {{- else }} + {{- $targetPath = .Path }} + {{- end }} + {{- /* Set common message. */}} + {{- $msg := printf "The %q render hook was unable to find heading ID %q in %s. See %s" $renderHookName . $targetPath $contentPath }} + {{- if eq $targetPath $contentPath }} + {{- $msg = printf "The %q render hook was unable to find heading ID %q in %s" $renderHookName . $targetPath }} + {{- end }} + {{- /* Throw warning or error. */}} + {{- if eq $errorLevel "warning" }} + {{- warnf $msg }} + {{- else if eq $errorLevel "error" }} + {{- errorf $msg }} + {{- end }} + {{- end }} + {{- end }} + +{{- end -}} diff --git a/layouts/_default/page.html b/layouts/_default/page.html index 2435e3ed7f8..1e5dcde76ee 100644 --- a/layouts/_default/page.html +++ b/layouts/_default/page.html @@ -1,11 +1,22 @@
    - {{ if .Params.categories }} + {{ if in (slice "functions" "methods") .Type }} + {{ with .FirstSection }} + + {{ humanize .Title | upper }} + + {{ end }} + {{ with .CurrentSection }} + + {{ humanize .Title | upper }} + + {{ end }} + {{ else }} {{ range .Params.categories }} - {{ humanize . | upper }} + {{ humanize . | upper }} {{ end }} - {{end}} + {{ end }}

    {{ .Title }}

    @@ -29,6 +40,7 @@

    {{- partial "docs/functions-signatures.html" . -}} + {{- partial "docs/functions-return-type.html" . -}} {{- partial "docs/functions-aliases.html" . -}} {{ .Content }}
    diff --git a/layouts/_default/single.html b/layouts/_default/single.html new file mode 100644 index 00000000000..4127c6dd06a --- /dev/null +++ b/layouts/_default/single.html @@ -0,0 +1,29 @@ +{{ define "main" }} +
    +
    + +
    + {{ partial "nav-links-docs.html" . }} +
    + +
    + {{ partial "right-sidebar" . }} +
    + +
    +
    + +
    +
    + {{ partial "docs/page-meta-data.html" . }} + {{ partial "page-edit.html" . }} + {{ partial "tags.html" . }} +
    +
    +{{ end }} diff --git a/layouts/partials/boxes-section-summaries.html b/layouts/partials/boxes-section-summaries.html index 268336cf56c..b293dc17da7 100644 --- a/layouts/partials/boxes-section-summaries.html +++ b/layouts/partials/boxes-section-summaries.html @@ -11,23 +11,31 @@

    - {{- if eq .context.Section "functions" -}} - {{ .context.LinkTitle }} - {{- else -}} {{ .context.Title }} - {{- end -}}

    -