From 2202c83299c324a2589566d616f5e41d8e28c343 Mon Sep 17 00:00:00 2001 From: Chase Fleming <1666730+chasefleming@users.noreply.github.com> Date: Sat, 21 Oct 2023 14:19:22 -0700 Subject: [PATCH] Introduce `Node` interface --- README.md | 2 +- elem.go | 28 +++++++++++------ elements.go | 54 ++++++++++++++++---------------- elements_test.go | 6 ++-- examples/htmx-fiber-form/main.go | 4 +-- utils.go | 8 ++--- utils_test.go | 2 +- 7 files changed, 57 insertions(+), 47 deletions(-) diff --git a/README.md b/README.md index a0d94fe..569305f 100644 --- a/README.md +++ b/README.md @@ -61,7 +61,7 @@ With `elem`, you can easily generate lists of elements from slices of data using ```go items := []string{"Item 1", "Item 2", "Item 3"} -liElements := elem.TransformEach(items, func(item string) *Element { +liElements := elem.TransformEach(items, func(item string) Node { return elem.Li(nil, elem.Text(item)) }) diff --git a/elem.go b/elem.go index b42249b..8ad2471 100644 --- a/elem.go +++ b/elem.go @@ -29,10 +29,25 @@ var voidElements = map[string]struct{}{ type Attrs map[string]string +type Node interface { + RenderTo(builder *strings.Builder) + Render() string +} + +type TextNode string + +func (t TextNode) RenderTo(builder *strings.Builder) { + builder.WriteString(string(t)) +} + +func (t TextNode) Render() string { + return string(t) +} + type Element struct { Tag string Attrs Attrs - Children []interface{} // Can be either string (for text) or another Element + Children []Node } func (e *Element) RenderTo(builder *strings.Builder) { @@ -65,14 +80,9 @@ func (e *Element) RenderTo(builder *strings.Builder) { // Close opening tag builder.WriteString(`>`) - // Build the content (either child text or nested elements) + // Build the content for _, child := range e.Children { - switch c := child.(type) { - case string: - builder.WriteString(c) - case *Element: - c.RenderTo(builder) - } + child.RenderTo(builder) } // Append closing tag @@ -87,7 +97,7 @@ func (e *Element) Render() string { return builder.String() } -func NewElement(tag string, attrs Attrs, children ...interface{}) *Element { +func NewElement(tag string, attrs Attrs, children ...Node) *Element { return &Element{ Tag: tag, Attrs: attrs, diff --git a/elements.go b/elements.go index e36f220..9af90ad 100644 --- a/elements.go +++ b/elements.go @@ -2,25 +2,25 @@ package elem // ========== Document Structure ========== -func Body(props Attrs, children ...interface{}) *Element { +func Body(props Attrs, children ...Node) *Element { return NewElement("body", props, children...) } -func Head(props Attrs, children ...interface{}) *Element { +func Head(props Attrs, children ...Node) *Element { return NewElement("head", props, children...) } -func Html(props Attrs, children ...interface{}) *Element { +func Html(props Attrs, children ...Node) *Element { return NewElement("html", props, children...) } -func Title(props Attrs, children ...interface{}) *Element { +func Title(props Attrs, children ...Node) *Element { return NewElement("title", props, children...) } // ========== Text Formatting and Structure ========== -func A(props Attrs, children ...interface{}) *Element { +func A(props Attrs, children ...Node) *Element { return NewElement("a", props, children...) } @@ -28,31 +28,31 @@ func Br(props Attrs) *Element { return NewElement("br", props) } -func Blockquote(props Attrs, children ...interface{}) *Element { +func Blockquote(props Attrs, children ...Node) *Element { return NewElement("blockquote", props, children...) } -func Code(props Attrs, children ...interface{}) *Element { +func Code(props Attrs, children ...Node) *Element { return NewElement("code", props, children...) } -func Div(props Attrs, children ...interface{}) *Element { +func Div(props Attrs, children ...Node) *Element { return NewElement("div", props, children...) } -func Em(props Attrs, children ...interface{}) *Element { +func Em(props Attrs, children ...Node) *Element { return NewElement("em", props, children...) } -func H1(props Attrs, children ...interface{}) *Element { +func H1(props Attrs, children ...Node) *Element { return NewElement("h1", props, children...) } -func H2(props Attrs, children ...interface{}) *Element { +func H2(props Attrs, children ...Node) *Element { return NewElement("h2", props, children...) } -func H3(props Attrs, children ...interface{}) *Element { +func H3(props Attrs, children ...Node) *Element { return NewElement("h3", props, children...) } @@ -60,43 +60,43 @@ func Hr(props Attrs) *Element { return NewElement("hr", props) } -func P(props Attrs, children ...interface{}) *Element { +func P(props Attrs, children ...Node) *Element { return NewElement("p", props, children...) } -func Pre(props Attrs, children ...interface{}) *Element { +func Pre(props Attrs, children ...Node) *Element { return NewElement("pre", props, children...) } -func Span(props Attrs, children ...interface{}) *Element { +func Span(props Attrs, children ...Node) *Element { return NewElement("span", props, children...) } -func Strong(props Attrs, children ...interface{}) *Element { +func Strong(props Attrs, children ...Node) *Element { return NewElement("strong", props, children...) } -func Text(content string) string { - return content +func Text(content string) TextNode { + return TextNode(content) } // ========== Lists ========== -func Li(props Attrs, children ...interface{}) *Element { +func Li(props Attrs, children ...Node) *Element { return NewElement("li", props, children...) } -func Ul(props Attrs, children ...interface{}) *Element { +func Ul(props Attrs, children ...Node) *Element { return NewElement("ul", props, children...) } // ========== Forms ========== -func Button(props Attrs, children ...interface{}) *Element { +func Button(props Attrs, children ...Node) *Element { return NewElement("button", props, children...) } -func Form(attrs Attrs, children ...interface{}) *Element { +func Form(attrs Attrs, children ...Node) *Element { return NewElement("form", attrs, children...) } @@ -104,19 +104,19 @@ func Input(attrs Attrs) *Element { return NewElement("input", attrs) } -func Label(attrs Attrs, children ...interface{}) *Element { +func Label(attrs Attrs, children ...Node) *Element { return NewElement("label", attrs, children...) } -func Option(attrs Attrs, content string) *Element { +func Option(attrs Attrs, content TextNode) *Element { return NewElement("option", attrs, content) } -func Select(attrs Attrs, children ...interface{}) *Element { +func Select(attrs Attrs, children ...Node) *Element { return NewElement("select", attrs, children...) } -func Textarea(attrs Attrs, content string) *Element { +func Textarea(attrs Attrs, content TextNode) *Element { return NewElement("textarea", attrs, content) } @@ -132,6 +132,6 @@ func Meta(props Attrs) *Element { return NewElement("meta", props) } -func Script(props Attrs, children ...interface{}) *Element { +func Script(props Attrs, children ...Node) *Element { return NewElement("script", props, children...) } diff --git a/elements_test.go b/elements_test.go index 130243f..c9e39aa 100644 --- a/elements_test.go +++ b/elements_test.go @@ -152,19 +152,19 @@ func TestInput(t *testing.T) { func TestLabel(t *testing.T) { expected := `` - el := Label(Attrs{attrs.For: "username"}, "Username") + el := Label(Attrs{attrs.For: "username"}, Text("Username")) assert.Equal(t, expected, el.Render()) } func TestSelectAndOption(t *testing.T) { expected := `` - el := Select(Attrs{attrs.Name: "color"}, Option(Attrs{attrs.Value: "red"}, "Red"), Option(Attrs{attrs.Value: "blue"}, "Blue")) + el := Select(Attrs{attrs.Name: "color"}, Option(Attrs{attrs.Value: "red"}, Text("Red")), Option(Attrs{attrs.Value: "blue"}, Text("Blue"))) assert.Equal(t, expected, el.Render()) } func TestTextarea(t *testing.T) { expected := `` - el := Textarea(Attrs{attrs.Name: "comment", attrs.Rows: "5"}, "Leave a comment...") + el := Textarea(Attrs{attrs.Name: "comment", attrs.Rows: "5"}, Text("Leave a comment...")) assert.Equal(t, expected, el.Render()) } diff --git a/examples/htmx-fiber-form/main.go b/examples/htmx-fiber-form/main.go index d1a36ae..2a8da0c 100644 --- a/examples/htmx-fiber-form/main.go +++ b/examples/htmx-fiber-form/main.go @@ -42,14 +42,14 @@ func main() { htmx.HXPost: "/submit-form", htmx.HXSwap: "outerHTML", }, - elem.Label(elem.Attrs{attrs.For: "name"}, "Name: "), + elem.Label(elem.Attrs{attrs.For: "name"}, elem.Text("Name: ")), elem.Input(elem.Attrs{ attrs.Type: "text", attrs.Name: "name", attrs.ID: "name", }), elem.Br(nil), - elem.Label(elem.Attrs{attrs.For: "email"}, "Email: "), + elem.Label(elem.Attrs{attrs.For: "email"}, elem.Text("Email: ")), elem.Input(elem.Attrs{ attrs.Type: "email", attrs.Name: "email", diff --git a/utils.go b/utils.go index 36f84fd..d60cb80 100644 --- a/utils.go +++ b/utils.go @@ -39,10 +39,10 @@ func Show(condition bool, ifTrue, ifFalse *Element) *Element { } // TransformEach maps a slice of items to a slice of Elements using the provided function -func TransformEach[T any](items []T, fn func(T) *Element) []*Element { - var elements []*Element +func TransformEach[T any](items []T, fn func(T) Node) []Node { + var nodes []Node for _, item := range items { - elements = append(elements, fn(item)) + nodes = append(nodes, fn(item)) } - return elements + return nodes } diff --git a/utils_test.go b/utils_test.go index 6573af3..ff7a2f5 100644 --- a/utils_test.go +++ b/utils_test.go @@ -43,7 +43,7 @@ func TestShow(t *testing.T) { func TestTransformEach(t *testing.T) { items := []string{"Item 1", "Item 2", "Item 3"} - elements := TransformEach(items, func(item string) *Element { + elements := TransformEach(items, func(item string) Node { return Li(nil, Text(item)) })