]> git.earlybird.gay Git - today/commitdiff
Improve parent scoping; decouple compute id from html id
authorearly <me@earlybird.gay>
Sat, 21 Dec 2024 02:14:57 +0000 (19:14 -0700)
committerearly <me@earlybird.gay>
Sat, 21 Dec 2024 02:14:57 +0000 (19:14 -0700)
web/internal/compile/compile.go
web/internal/compile/compute.go
web/internal/compile/conversions.go
web/internal/compile/template.go
web/page/serve.go
web/render/data.go

index 0081d2bd2fd4cb731b617ed6c2eca85a25fa8801..ea89f754a966a8c4c89656aa0cd3d1d1b1a38b7b 100644 (file)
@@ -71,7 +71,6 @@ func Compile(root TemplateSource, transform ...func(root *html.Node)) ([]Result,
                }, document.FirstChild)
 
                // Replace template functions in the root before we add any subsources
-               // fmt.Println(root.TemplateFuncs())
                htmltree.Walk(document, func(n *html.Node) (bool, error) {
                        replaceTemplateFuncs(root, n)
                        return false, nil
@@ -111,11 +110,15 @@ func Compile(root TemplateSource, transform ...func(root *html.Node)) ([]Result,
                                                for fname, f := range templateSource.TemplateFuncs() {
                                                        templateFuncs[snakeCase(n.Data)+"_"+fname] = f
                                                }
-                                               // Add $today_parent before template sources
+                                               // Add parent scoping
                                                n.Parent.InsertBefore(&html.Node{
                                                        Type: html.TextNode,
-                                                       Data: "{{ $today_parent := . }}",
+                                                       Data: "{{ call $today_upParent . }}",
                                                }, n)
+                                               n.Parent.InsertBefore(&html.Node{
+                                                       Type: html.TextNode,
+                                                       Data: "{{ call $today_downParent }}",
+                                               }, n.NextSibling)
                                                // Parse HTML fragment and replace the content of the node with it
                                                computeSubSource, err := insertTemplateSource(lang, templateSource, n)
                                                if err != nil {
@@ -143,10 +146,13 @@ func Compile(root TemplateSource, transform ...func(root *html.Node)) ([]Result,
 
                // POSTPROCESSING
 
-               // Assign .SetDot to $setDot
+               // Assign functions
+               funcAssigns := `{{ $today_setDot := .SetDot }}
+               {{ $today_upParent := .UpParent }}
+               {{ $today_downParent := .DownParent }}`
                document.InsertBefore(&html.Node{
                        Type: html.TextNode,
-                       Data: `{{ $setDot := .SetDot }}{{ with .Data }}`,
+                       Data: funcAssigns + `{{ with .Data }}`,
                }, document.FirstChild)
                // Add end to end
                document.InsertBefore(&html.Node{
@@ -157,7 +163,7 @@ func Compile(root TemplateSource, transform ...func(root *html.Node)) ([]Result,
                // Split up templates in text nodes
                splitTemplateNodes(document)
 
-               // Traverse the document and add $setDot on entry and exit from pipelines
+               // Traverse the document and add $today_setDot on entry and exit from pipelines
                addSetDots(document)
 
                for _, tf := range transform {
index ddc3256a8acce46aa3c37c47d7dfcd7b5315dd08..1b296371afdeea79cc6cdf5c21b67aad135e00b3 100644 (file)
@@ -2,6 +2,7 @@
 package compile
 
 import (
+       "errors"
        "fmt"
        "net/http"
        "reflect"
@@ -16,14 +17,35 @@ type computeNode struct {
        compute          render.OnLoadFunc
        asDataFromParent map[string]string
        children         []*computeNode
+
+       setDot     func(any) error
+       upParent   func(any) error
+       downParent func() error
 }
 
-func (root *computeNode) Compute(r *http.Request) (render.Data, func(any) error, error) {
-       var dot any
-       setDot := func(val any) error {
+func (root *computeNode) Compute(r *http.Request) (render.Data, error) {
+       var dot, parent any
+       var parentStack []any
+       root.setDot = func(val any) error {
                dot = val
                return nil
        }
+       root.upParent = func(val any) error {
+               parent = val
+               parentStack = append(parentStack, val)
+               return nil
+       }
+       root.downParent = func() error {
+               if len(parentStack) == 0 {
+                       return errors.New("unbalanced parent stack control")
+               } else if len(parentStack) == 1 {
+                       parent = nil
+               } else {
+                       parent = parentStack[len(parentStack)-1]
+               }
+               parentStack = parentStack[:len(parentStack)-1]
+               return nil
+       }
        ctx := r.Context()
 
        var f func(n *computeNode, data render.Data) error
@@ -59,6 +81,7 @@ func (root *computeNode) Compute(r *http.Request) (render.Data, func(any) error,
                                        }
                                        childData.Set(dataKey, val)
                                }
+                               childData.Set("today_parent", parent)
                                f(child, childData)
                                return childData, child.compute(ctx, childData)
                        })
@@ -69,7 +92,19 @@ func (root *computeNode) Compute(r *http.Request) (render.Data, func(any) error,
        }
 
        data := render.NewData(r)
-       return data, setDot, f(root, data)
+       return data, f(root, data)
+}
+
+func (root *computeNode) SetDot(val any) error {
+       return root.setDot(val)
+}
+
+func (root *computeNode) UpParent(val any) error {
+       return root.upParent(val)
+}
+
+func (root *computeNode) DownParent() error {
+       return root.downParent()
 }
 
 // Get a variable from data.
index 303aff125673960cac4a75357cee438b7f3c5723..cc4ea8307bae0a892ce1f77060ece55fee20838c 100644 (file)
@@ -5,5 +5,5 @@ import "strings"
 
 // EXTREMELY lazy snake caser
 func snakeCase(s string) string {
-       return strings.ToLower(strings.Replace(s, "-", "_", -1))
+       return strings.ToLower(strings.ReplaceAll(s, "-", "_"))
 }
index bf90afbc69d413e9aa1bae19db4de0f970b52c50..0225dc4c1093e46638eb0cfd771eb79ebc3d4990 100644 (file)
@@ -4,6 +4,7 @@ package compile
 import (
        "errors"
        "fmt"
+       "math/rand/v2"
        "regexp"
        "slices"
        "strings"
@@ -291,9 +292,9 @@ func replaceTemplateFields(n *html.Node) {
                                inAttrs && isDataAttr[i] {
                                // special case; . is parentable but shouldn't be appended
                                if token == "." {
-                                       token = "$today_parent"
+                                       token = ".today_parent"
                                } else {
-                                       token = "$today_parent" + token
+                                       token = ".today_parent" + token
                                }
                        }
                        output = append(output, token)
@@ -328,7 +329,7 @@ func removeDataAttrs(n *html.Node) map[string]string {
 func insertTemplateSource(lang language.Tag, subSource TemplateSource, context *html.Node) (*computeNode, error) {
        computedName := snakeCase(subSource.Name())
        // ===== CONTEXT PREPROCESSING =====
-       // Raise any fields pre-existing in the context to $today_parent.field.
+       // Raise any fields pre-existing in the context to .today_parent.field.
        raiseDepth := 0
        raiseStack := []string{}
        htmltree.Walk(context.FirstChild, func(n *html.Node) (bool, error) {
@@ -503,15 +504,11 @@ func insertTemplateSource(lang language.Tag, subSource TemplateSource, context *
        }
 
        // Generate a computeNode for this part.
-       htmlId := getAttr(context, "id")
-       scopeName := computedName + "_" + snakeCase(htmlId)
-       if htmlId == "" || containsTemplate(htmlId) {
-               htmlId = ""
-               scopeName = computedName
-       }
+       computeId := fmt.Sprintf("%x", rand.Int32())
+       scopeName := computedName + "_" + computeId
        compute := &computeNode{
                name:             computedName,
-               htmlId:           htmlId,
+               htmlId:           computeId,
                compute:          subSource.OnLoad(),
                asDataFromParent: asData,
        }
@@ -568,7 +565,7 @@ func addSetDots(document *html.Node) {
                        if setsDot(n.Data) {
                                n.Parent.InsertBefore(&html.Node{
                                        Type: html.TextNode,
-                                       Data: `{{ call $setDot . }}`,
+                                       Data: `{{ call $today_setDot . }}`,
                                }, n.NextSibling)
                                endAcc = append(endAcc, true)
                        } else {
@@ -580,7 +577,7 @@ func addSetDots(document *html.Node) {
                        if shouldSet {
                                n.Parent.InsertBefore(&html.Node{
                                        Type: html.TextNode,
-                                       Data: "{{ call $setDot . }}",
+                                       Data: "{{ call $today_setDot . }}",
                                }, n.NextSibling)
                        }
                        if strings.Contains(n.Data, "else") {
index 427983f01b204ffaba2e09c36cd20ae85e816e6b..a4ea10ee38b911dc6adab11e5973ea15f5da6ba6 100644 (file)
@@ -9,7 +9,9 @@ import (
 )
 
 type RootData struct {
-       SetDot func(value any) error
+       SetDot     func(value any) error
+       UpParent   func(value any) error
+       DownParent func() error
 
        Data render.Data
 }
@@ -21,12 +23,15 @@ type renderable struct {
 
 func (p *Page) ServeHTTP(w http.ResponseWriter, r *http.Request) {
        toRender := p.chooseRenderable(r)
-       data, setDot, err := toRender.templateLoad.Compute(r)
+       loader := toRender.templateLoad
+       data, err := loader.Compute(r)
        if err != nil {
                panic(err)
        }
        root := RootData{
-               SetDot: setDot,
+               SetDot:     loader.SetDot,
+               UpParent:   loader.UpParent,
+               DownParent: loader.DownParent,
 
                Data: data,
        }
index 3f9fe9feb2269d3962413bcb7e7c04e678b01a81..d344a25aa5e4645e6225c2efc9003936a18b6f3f 100644 (file)
@@ -49,5 +49,8 @@ func (d Data) Set(key string, value any) {
 
 type OnLoadFunc func(ctx context.Context, data Data) error
 type Loader interface {
-       Compute(r *http.Request) (Data, func(any) error, error)
+       Compute(r *http.Request) (Data, error)
+       SetDot(val any) error
+       UpParent(val any) error
+       DownParent() error
 }