Introduce Placer interface (#18) When implemented, the `Place` method of the `Placer` interface tells `Render` in `El` where to put a Node. This is relevant for helpers that want to be rendered like attributes, inside the parent element. Fixes the bug where `attr.Classes` was rendered outside the element.
Markus Wüstenberg markus@maragu.dk
Thu, 24 Sep 2020 13:19:52 +0200
4 files changed, 57 insertions(+), 8 deletions(-)
M attr/attributes.go → attr/attributes.go
@@ -35,6 +35,10 @@ sort.Strings(included) return g.Attr("class", strings.Join(included, " ")).Render() } +func (c Classes) Place() g.Placement { + return g.Inside +} + // String satisfies fmt.Stringer. func (c Classes) String() string { return c.Render()
M attr/attributes_test.go → attr/attributes_test.go
@@ -3,6 +3,7 @@ import ( "testing" + g "github.com/maragudk/gomponents" "github.com/maragudk/gomponents/assert" "github.com/maragudk/gomponents/attr" ) @@ -27,6 +28,11 @@ "hat": true, "partyhat": true, "turtlehat": false, }) + }) + + t.Run("renders as attribute in an element", func(t *testing.T) { + e := g.El("div", attr.Classes{"hat": true}) + assert.Equal(t, `<div class="hat"/>`, e) }) t.Run("also works with fmt", func(t *testing.T) {
M gomponents.go → gomponents.go
@@ -19,6 +19,20 @@ type Node interface { Render() string } +// Placer can be implemented to tell Render functions where to place the string representation of a Node +// in the parent element. +type Placer interface { + Place() Placement +} + +// Placement is used with the Placer interface. +type Placement int + +const ( + Outside = Placement(iota) + Inside +) + // NodeFunc is render function that is also a Node. type NodeFunc func() string @@ -26,6 +40,10 @@ func (n NodeFunc) Render() string { return n() } +func (n NodeFunc) Place() Placement { + return Outside +} + // String satisfies fmt.Stringer. func (n NodeFunc) String() string { return n.Render() @@ -35,7 +53,7 @@ // El creates an element DOM Node with a name and child Nodes. // Use this if no convenience creator exists. func El(name string, children ...Node) NodeFunc { return func() string { - var b, attrString, childrenString strings.Builder + var b, inside, outside strings.Builder b.WriteString("<") b.WriteString(name) @@ -46,22 +64,28 @@ return b.String() } for _, c := range children { - if _, ok := c.(attr); ok { - attrString.WriteString(c.Render()) - } else { - childrenString.WriteString(c.Render()) + if p, ok := c.(Placer); ok { + switch p.Place() { + case Inside: + inside.WriteString(c.Render()) + case Outside: + outside.WriteString(c.Render()) + } + continue } + // If c doesn't implement Placer, default to outside + outside.WriteString(c.Render()) } - b.WriteString(attrString.String()) + b.WriteString(inside.String()) - if childrenString.Len() == 0 { + if outside.Len() == 0 { b.WriteString("/>") return b.String() } b.WriteString(">") - b.WriteString(childrenString.String()) + b.WriteString(outside.String()) b.WriteString("</") b.WriteString(name) b.WriteString(">") @@ -95,6 +119,10 @@ if a.value == nil { return fmt.Sprintf(" %v", a.name) } return fmt.Sprintf(` %v="%v"`, a.name, *a.value) +} + +func (a attr) Place() Placement { + return Inside } // String satisfies fmt.Stringer.
M gomponents_test.go → gomponents_test.go
@@ -52,6 +52,12 @@ } }) } +type outsider struct{} + +func (o outsider) Render() string { + return "outsider" +} + func TestEl(t *testing.T) { t.Run("renders an empty element if no children given", func(t *testing.T) { e := g.El("div") @@ -71,6 +77,11 @@ t.Run("renders attributes at the correct place regardless of placement in parameter list", func(t *testing.T) { e := g.El("div", g.El("span"), g.Attr("class", "hat")) assert.Equal(t, `<div class="hat"><span/></div>`, e) + }) + + t.Run("renders outside if node does not implement placer", func(t *testing.T) { + e := g.El("div", outsider{}) + assert.Equal(t, `<div>outsider</div>`, e) }) }