// decl keeps track of the variable mappings, i.e. $mysite => .Site etc.
 type decl map[string]string
 
-var paramsPaths = [][]string{
-       {"Params"},
-       {"Site", "Params"},
-
-       // Site and Pag referenced from shortcodes
-       {"Page", "Site", "Params"},
-       {"Page", "Params"},
+const (
+       paramsIdentifier = "Params"
+)
 
-       {"Site", "Language", "Params"},
+// Containers that may contain Params that we will not touch.
+var reservedContainers = map[string]bool{
+       // Aka .Site.Data.Params which must stay case sensitive.
+       "Data": true,
 }
 
 type templateContext struct {
        for i := index; i < len(idents); i++ {
                idents[i] = strings.ToLower(idents[i])
        }
+
 }
 
 // indexOfReplacementStart will return the index of where to start doing replacement,
                return -1
        }
 
-       first := idents[0]
-       firstIsVar := first[0] == '$'
+       if l == 1 {
+               first := idents[0]
+               if first == "" || first == paramsIdentifier || first[0] == '$' {
+                       // This can not be a Params.x
+                       return -1
+               }
+       }
 
-       if l == 1 && !firstIsVar {
-               // This can not be a Params.x
+       var lookFurther bool
+       var needsVarExpansion bool
+       for _, ident := range idents {
+               if ident[0] == '$' {
+                       lookFurther = true
+                       needsVarExpansion = true
+                       break
+               } else if ident == paramsIdentifier {
+                       lookFurther = true
+                       break
+               }
+       }
+
+       if !lookFurther {
                return -1
        }
 
-       if !firstIsVar {
-               found := false
-               for _, paramsPath := range paramsPaths {
-                       if first == paramsPath[0] {
-                               found = true
-                               break
+       var resolvedIdents []string
+
+       if !needsVarExpansion {
+               resolvedIdents = idents
+       } else {
+               var ok bool
+               resolvedIdents, ok = d.resolveVariables(idents)
+               if !ok {
+                       return -1
+               }
+       }
+
+       var paramFound bool
+       for i, ident := range resolvedIdents {
+               if ident == paramsIdentifier {
+                       if i > 0 {
+                               container := resolvedIdents[i-1]
+                               if reservedContainers[container] {
+                                       // .Data.Params.someKey
+                                       return -1
+                               }
+                               if !d.isKeyword(container) {
+                                       // where $pages ".Params.toc_hide" "!=" true
+                                       return -1
+                               }
                        }
+                       if i < len(resolvedIdents)-1 {
+                               next := resolvedIdents[i+1]
+                               if !d.isKeyword(next) {
+                                       return -1
+                               }
+                       }
+
+                       paramFound = true
+                       break
                }
-               if !found {
-                       return -1
+       }
+
+       if !paramFound {
+               return -1
+       }
+
+       var paramSeen bool
+       idx := -1
+       for i, ident := range idents {
+               if ident == "" || ident[0] == '$' {
+                       continue
+               }
+
+               if ident == paramsIdentifier {
+                       paramSeen = true
+                       idx = -1
+
+               } else {
+                       if paramSeen {
+                               return i
+                       }
+                       if idx == -1 {
+                               idx = i
+                       }
                }
        }
+       return idx
+
+}
 
+func (d decl) resolveVariables(idents []string) ([]string, bool) {
        var (
-               resolvedIdents []string
-               replacements   []string
-               replaced       []string
+               replacements []string
+               replaced     []string
        )
 
        // An Ident can start out as one of
 
                if i > 20 {
                        // bail out
-                       return -1
+                       return nil, false
                }
 
                potentialVar := replacements[i]
 
                if !ok {
                        // Temporary range vars. We do not care about those.
-                       return -1
+                       return nil, false
                }
 
                replacement = strings.TrimPrefix(replacement, ".")
                }
        }
 
-       resolvedIdents = append(replaced, idents[1:]...)
-
-       for _, paramPath := range paramsPaths {
-               if index := indexOfFirstRealIdentAfterWords(resolvedIdents, idents, paramPath...); index != -1 {
-                       return index
-               }
-       }
-
-       return -1
+       return append(replaced, idents[1:]...), true
 
 }
 
-func indexOfFirstRealIdentAfterWords(resolvedIdents, idents []string, words ...string) int {
-       if !sliceStartsWith(resolvedIdents, words...) {
-               return -1
-       }
-
-       for i, ident := range idents {
-               if ident == "" || ident[0] == '$' {
-                       continue
-               }
-               found := true
-               for _, word := range words {
-                       if ident == word {
-                               found = false
-                               break
-                       }
-               }
-               if found {
-                       return i
-               }
-       }
-
-       return -1
-}
-
-func sliceStartsWith(slice []string, words ...string) bool {
-
-       if len(slice) < len(words) {
-               return false
-       }
-
-       for i, word := range words {
-               if word != slice[i] {
-                       return false
-               }
-       }
-       return true
+func (d decl) isKeyword(s string) bool {
+       return !strings.ContainsAny(s, " -\"")
 }
 
 
 import (
        "bytes"
+       "fmt"
        "testing"
 
        "html/template"
 var (
        testFuncs = map[string]interface{}{
                "Echo": func(v interface{}) interface{} { return v },
+               "where": func(seq, key interface{}, args ...interface{}) (interface{}, error) {
+                       return map[string]interface{}{
+                               "ByWeight": fmt.Sprintf("%v:%v:%v", seq, key, args),
+                       }, nil
+               },
        }
 
        paramsData = map[string]interface{}{
                "Slice":    []int{1, 3},
                "Params": map[string]interface{}{
                        "lower": "P1L",
+                       "slice": []int{1, 3},
+               },
+               "Pages": map[string]interface{}{
+                       "ByWeight": []int{1, 3},
+               },
+               "CurrentSection": map[string]interface{}{
+                       "Params": map[string]interface{}{
+                               "lower": "pcurrentsection",
+                       },
                },
                "Site": map[string]interface{}{
                        "Params": map[string]interface{}{
 
        paramsTempl = `
 {{ $page := . }}
+{{ $pages := .Pages }}
 {{ $pageParams := .Params }}
 {{ $site := .Site }}
 {{ $siteParams := .Site.Params }}
 {{ $data := .Site.Data }}
 {{ $notparam := .NotParam }}
 
+PCurrentSection: {{ .CurrentSection.Params.LOWER }}
 P1: {{ .Params.LOWER }}
 P1_2: {{ $.Params.LOWER }}
 P1_3: {{ $page.Params.LOWER }}
 F1: {{ printf "themes/%s-theme" .Site.Params.LOWER }}
 F2: {{ Echo (printf "themes/%s-theme" $lower) }}
 F3: {{ Echo (printf "themes/%s-theme" .Site.Params.LOWER) }}
+
+PSLICE: {{ range .Params.SLICE }}PSLICE{{.}}|{{ end }}
+
+{{ $pages := "foo" }}
+{{ $pages := where $pages ".Params.toc_hide" "!=" true }}
+PARAMS STRING: {{ $pages.ByWeight }}
+PARAMS STRING2: {{ with $pages }}{{ .ByWeight }}{{ end }}
+{{ $pages3 := where ".Params.TOC_HIDE" "!=" .Params.LOWER }}
+PARAMS STRING3: {{ $pages3.ByWeight }}
 `
 )
 
        require.Contains(t, result, "F2: themes/P2L-theme")
        require.Contains(t, result, "F3: themes/P2L-theme")
 
+       require.Contains(t, result, "PSLICE: PSLICE1|PSLICE3|")
+       require.Contains(t, result, "PARAMS STRING: foo:.Params.toc_hide:[!= true]")
+       require.Contains(t, result, "PARAMS STRING2: foo:.Params.toc_hide:[!= true]")
+       require.Contains(t, result, "PARAMS STRING3: .Params.TOC_HIDE:!=:[P1L]")
+
+       // Issue #5068
+       require.Contains(t, result, "PCurrentSection: pcurrentsection")
+
 }
 
 func BenchmarkTemplateParamsKeysToLower(b *testing.B) {
                        "Params": map[string]interface{}{
                                "colors": map[string]interface{}{
                                        "blue": "Amber",
+                                       "pretty": map[string]interface{}{
+                                               "first": "Indigo",
+                                       },
                                },
                        },
                }
                paramsTempl = `
 {{$__amber_1 := .Params.Colors}}
 {{$__amber_2 := $__amber_1.Blue}}
+{{$__amber_3 := $__amber_1.Pretty}}
+{{$__amber_4 := .Params}}
+
 Color: {{$__amber_2}}
 Blue: {{ $__amber_1.Blue}}
+Pretty First1: {{ $__amber_3.First}}
+Pretty First2: {{ $__amber_1.Pretty.First}}
+Pretty First3: {{ $__amber_4.COLORS.PRETTY.FIRST}}
 `
        )
 
        result := b.String()
 
        require.Contains(t, result, "Color: Amber")
+       require.Contains(t, result, "Blue: Amber")
+       require.Contains(t, result, "Pretty First1: Indigo")
+       require.Contains(t, result, "Pretty First2: Indigo")
+       require.Contains(t, result, "Pretty First3: Indigo")
 
 }