Add reference to parent shortcode
authorBjørn Erik Pedersen <bjorn.erik.pedersen@gmail.com>
Tue, 8 Mar 2016 19:56:24 +0000 (20:56 +0100)
committerBjørn Erik Pedersen <bjorn.erik.pedersen@gmail.com>
Wed, 9 Mar 2016 10:06:40 +0000 (11:06 +0100)
Fixes #1936

docs/content/extras/shortcodes.md
hugolib/shortcode.go
hugolib/shortcode_test.go

index ef78b30c99117e40cba57b13031a261d343d6dd0..34c6f348685ff0960d139a13f8db7190a228fb31 100644 (file)
@@ -273,6 +273,8 @@ such as `{{ if .IsNamedParams }}...{{ else }}...{{ end }}`.  See the
 
 You can also use the variable `.Page` to access all the normal [Page Variables](/templates/variables/).
 
+A shortcodes can be nested. In a nested shortcode you can access the parent shortcode context with `.Parent`. This can be very useful for inheritance of common shortcode parameters from the root.
+
 ## Single Positional Example: youtube
 
     {{</* youtube 09jf3ow9jfw */>}}
index 4cd1db87b7711d9328eeb64f5633cc21639f42de..578093210e1116856835c840f21ee9ead84471a0 100644 (file)
@@ -34,6 +34,7 @@ type ShortcodeWithPage struct {
        Params        interface{}
        Inner         template.HTML
        Page          *Page
+       Parent        *ShortcodeWithPage
        IsNamedParams bool
 }
 
@@ -189,7 +190,7 @@ const innerNewlineRegexp = "\n"
 const innerCleanupRegexp = `\A<p>(.*)</p>\n\z`
 const innerCleanupExpand = "$1"
 
-func renderShortcode(sc shortcode, p *Page, t tpl.Template) string {
+func renderShortcode(sc shortcode, parent *ShortcodeWithPage, p *Page, t tpl.Template) string {
        tmpl := getShortcodeTemplate(sc.name, t)
 
        if tmpl == nil {
@@ -197,7 +198,7 @@ func renderShortcode(sc shortcode, p *Page, t tpl.Template) string {
                return ""
        }
 
-       data := &ShortcodeWithPage{Params: sc.params, Page: p}
+       data := &ShortcodeWithPage{Params: sc.params, Page: p, Parent: parent}
        if sc.params != nil {
                data.IsNamedParams = reflect.TypeOf(sc.params).Kind() == reflect.Map
        }
@@ -209,7 +210,7 @@ func renderShortcode(sc shortcode, p *Page, t tpl.Template) string {
                        case string:
                                inner += innerData.(string)
                        case shortcode:
-                               inner += renderShortcode(innerData.(shortcode), p, t)
+                               inner += renderShortcode(innerData.(shortcode), data, p, t)
                        default:
                                jww.ERROR.Printf("Illegal state on shortcode rendering of '%s' in page %s. Illegal type in inner data: %s ",
                                        sc.name, p.BaseFileName(), reflect.TypeOf(innerData))
@@ -271,7 +272,7 @@ func extractAndRenderShortcodes(stringToParse string, p *Page, t tpl.Template) (
                        // need to have something to replace with
                        renderedShortcodes[key] = ""
                } else {
-                       renderedShortcodes[key] = renderShortcode(sc, p, t)
+                       renderedShortcodes[key] = renderShortcode(sc, nil, p, t)
                }
        }
 
index 26854562e620301b9262677c850f1212c9bba867..306b53e8430ca062553665095d949c93ddb0e5ac 100644 (file)
@@ -232,6 +232,17 @@ func TestNestedComplexSC(t *testing.T) {
                "-row-1-s-col-2-<strong>s</strong>-aside-3-<strong>s</strong>-asideStop-4-s-colStop-5-s-rowStop-6-s", tem)
 }
 
+func TestParentShortcode(t *testing.T) {
+       tem := tpl.New()
+       tem.AddInternalShortcode("r1.html", `1: {{ .Get "pr1" }} {{ .Inner }}`)
+       tem.AddInternalShortcode("r2.html", `2: {{ .Parent.Get "pr1" }}{{ .Get "pr2" }} {{ .Inner }}`)
+       tem.AddInternalShortcode("r3.html", `3: {{ .Parent.Parent.Get "pr1" }}{{ .Parent.Get "pr2" }}{{ .Get "pr3" }} {{ .Inner }}`)
+
+       CheckShortCodeMatch(t, `{{< r1 pr1="p1" >}}1: {{< r2 pr2="p2" >}}2: {{< r3 pr3="p3" >}}{{< /r3 >}}{{< /r2 >}}{{< /r1 >}}`,
+               "1: p1 1: 2: p1p2 2: 3: p1p2p3 ", tem)
+
+}
+
 func TestFigureImgWidth(t *testing.T) {
        tem := tpl.New()
        CheckShortCodeMatch(t, `{{% figure src="/found/here" class="bananas orange" alt="apple" width="100px" %}}`, "\n<figure class=\"bananas orange\">\n    \n        <img src=\"/found/here\" alt=\"apple\" width=\"100px\" />\n    \n    \n</figure>\n", tem)