all repos — gomponents @ 267d40bbea6036f913c4047a6a2055b3e5d7bb96

HTML components in pure Go

Add Map function (#40)

`Map` makes it easier to build lists of elements without having to iterate.
Markus Wüstenberg markus@maragu.dk
Mon, 02 Nov 2020 10:59:16 +0100
commit

267d40bbea6036f913c4047a6a2055b3e5d7bb96

parent

6c8f0c235287edf7252fe239d4c9beb258c6ff01

3 files changed, 36 insertions(+), 6 deletions(-)

jump to
M examples/simple/simple.goexamples/simple/simple.go
@@ -61,11 +61,11 @@ {"/", "Home"}, 		{"/foo", "Foo"},
 		{"/bar", "Bar"},
 	}
-	var lis []g.Node
-	for _, item := range items {
-		lis = append(lis, el.Li(el.A(item.path,
-			attr.Classes(map[string]bool{"is-active": props.path == item.path}),
-			g.Text(item.text))))
-	}
+	lis := g.Map(len(items), func(i int) g.Node {
+		item := items[i]
+		return el.Li(
+			el.A(item.path, attr.Classes(map[string]bool{"is-active": props.path == item.path}), g.Text(item.text)),
+		)
+	})
 	return el.Ul(attr.Class("nav"), g.Group(lis))
 }
M gomponents.gogomponents.go
@@ -217,3 +217,20 @@ // Render must happen through a parent element created with El or a helper. func Group(children []Node) Node {
 	return group{children: children}
 }
+
+// Map something enumerable to a list of Nodes.
+// Example:
+// 	items := []string{"hat", "partyhat"}
+//
+// 	lis := g.Map(len(items), func(i int) g.Node {
+// 		return g.El("li", g.Text(items[i]))
+// 	})
+//
+// 	list := g.El("ul", lis...)
+func Map(length int, cb func(i int) Node) []Node {
+	var nodes []Node
+	for i := 0; i < length; i++ {
+		nodes = append(nodes, cb(i))
+	}
+	return nodes
+}
M gomponents_test.gogomponents_test.go
@@ -183,3 +183,16 @@ t.FailNow() 		}
 	})
 }
+
+func TestMap(t *testing.T) {
+	t.Run("maps slices to nodes", func(t *testing.T) {
+		items := []string{"hat", "partyhat", "turtlehat"}
+		lis := g.Map(len(items), func(i int) g.Node {
+			return g.El("li", g.Text(items[i]))
+		})
+
+		list := g.El("ul", lis...)
+
+		assert.Equal(t, `<ul><li>hat</li><li>partyhat</li><li>turtlehat</li></ul>`, list)
+	})
+}