From 4427206988bb2b1ee57621e4e8d5a89f5dd3d74c Mon Sep 17 00:00:00 2001
From: faiface <faiface@ksp.sk>
Date: Wed, 14 Dec 2016 17:53:29 +0100
Subject: [PATCH] add general Shape and embed it in specific shapes

---
 graphics.go | 406 +++++++++++++++++-----------------------------------
 1 file changed, 133 insertions(+), 273 deletions(-)

diff --git a/graphics.go b/graphics.go
index 886a507..b0485bb 100644
--- a/graphics.go
+++ b/graphics.go
@@ -85,111 +85,73 @@ func (g *Group) Do(sub func(pixelgl.Context)) {
 	sub(g.context)
 }
 
-// Sprite is a picture that can be drawn on the screen. Optionally it can be color masked or tranformed.
+// Shape is a general drawable shape constructed from vertices.
 //
-// Usually, you only transform objects when you're drawing them (by passing transforms to the Draw method).
-// With sprites however, it can be useful to also transform them "statically". For example, sprites are
-// anchor by their bottom-left corner by default. Setting a transform can change this anchor to the center,
-// or wherever you want.
-type Sprite struct {
+// Vertices are specified in the vertex array of a shape. A shape can have a picture, a color (mask) and a static
+// transform.
+//
+// Usually you use this type only indirectly throught other specific shapes (sprites, polygons, ...) embedding it.
+type Shape struct {
 	parent    pixelgl.Doer
-	color     color.Color
 	picture   Picture
+	color     color.Color
 	transform Transform
 	va        *pixelgl.VertexArray
 }
 
-// NewSprite creates a new sprite with the supplied picture. The sprite's size is the size of the supplied picture.
-// If you want to change the sprite's size, change it's transform.
-func NewSprite(parent pixelgl.Doer, picture Picture) *Sprite {
-	s := &Sprite{
+// NewShape creates a new shape with specified parent, picture, color, transform and vertex array.
+func NewShape(parent pixelgl.Doer, picture Picture, c color.Color, transform Transform, va *pixelgl.VertexArray) *Shape {
+	return &Shape{
 		parent:    parent,
-		color:     color.White,
 		picture:   picture,
-		transform: Position(0),
-	}
-
-	parent.Do(func(ctx pixelgl.Context) {
-		var err error
-		s.va, err = pixelgl.NewVertexArray(
-			picture.Texture(),
-			ctx.Shader().VertexFormat(),
-			pixelgl.DynamicUsage,
-			4,
-			[]int{0, 1, 2, 0, 2, 3},
-		)
-		if err != nil {
-			panic(errors.Wrap(err, "failed to create sprite"))
-		}
-	})
-
-	w, h := picture.Bounds().Size.XY()
-	for i, p := range []Vec{V(0, 0), V(w, 0), V(w, h), V(0, h)} {
-		texCoord := V(
-			(picture.Bounds().X()+p.X())/float64(picture.Texture().Width()),
-			(picture.Bounds().Y()+p.Y())/float64(picture.Texture().Height()),
-		)
-
-		s.va.SetVertexAttributeVec2(
-			i,
-			pixelgl.Position,
-			mgl32.Vec2{
-				float32(p.X()),
-				float32(p.Y()),
-			},
-		)
-		s.va.SetVertexAttributeVec4(i, pixelgl.Color, mgl32.Vec4{1, 1, 1, 1})
-		s.va.SetVertexAttributeVec2(
-			i,
-			pixelgl.TexCoord,
-			mgl32.Vec2{
-				float32(texCoord.X()),
-				float32(texCoord.Y()),
-			},
-		)
+		color:     c,
+		transform: transform,
+		va:        va,
 	}
-
-	return s
 }
 
-// Delete deletes a sprite. Note, that this does not delete it's picture.
-func (s *Sprite) Delete() {
+// Delete deletes the underlying
+func (s *Shape) Delete() {
 	s.va.Delete()
 }
 
-// Picture returns the sprite's picture.
-func (s *Sprite) Picture() Picture {
+// SetPicture changes the picture of a shape.
+func (s *Shape) SetPicture(picture Picture) {
+	s.picture = picture
+}
+
+// Picture returns the current picture of a shape.
+func (s *Shape) Picture() Picture {
 	return s.picture
 }
 
-// SetColor sets a mask color of a sprite.
-func (s *Sprite) SetColor(c color.Color) {
+// SetColor changes the color (mask) of a shape.
+func (s *Shape) SetColor(c color.Color) {
 	s.color = c
 }
 
-// Color returns the mask color of a sprite. Default is white.
-func (s *Sprite) Color() color.Color {
+// Color returns the current color (mask) of a shape.
+func (s *Shape) Color() color.Color {
 	return s.color
 }
 
-// SetTransform sets a "static" transform of a sprite. Setting a transform is equivalent to passing
-// the transform as the last parameter to Draw.
-//
-//   sprite.SetTransform(transform)
-//   sprite.Draw(camera)
-//   // same as below
-//   sprite.Draw(camera, tranform)
-func (s *Sprite) SetTransform(t Transform) {
-	s.transform = t
+// SetTransform changes the ("static") transform of a shape.
+func (s *Shape) SetTransform(transform Transform) {
+	s.transform = transform
 }
 
-// Transform returns the static transform of a sprite.
-func (s *Sprite) Transform() Transform {
+// Transform returns the current ("static") transform of a shape.
+func (s *Shape) Transform() Transform {
 	return s.transform
 }
 
+// VertexArray changes the underlying vertex array of a shape.
+func (s *Shape) VertexArray() *pixelgl.VertexArray {
+	return s.va
+}
+
 // Draw draws a sprite transformed by the supplied transforms applied in the reverse order.
-func (s *Sprite) Draw(t ...Transform) {
+func (s *Shape) Draw(t ...Transform) {
 	mat := mgl32.Ident3()
 	for i := range t {
 		mat = mat.Mul3(t[i].Mat3())
@@ -198,36 +160,78 @@ func (s *Sprite) Draw(t ...Transform) {
 
 	s.parent.Do(func(ctx pixelgl.Context) {
 		r, g, b, a := colorToRGBA(s.color)
-		ctx.Shader().SetUniformVec4(pixelgl.MaskColor, mgl32.Vec4{r, g, b, a})
-		ctx.Shader().SetUniformMat3(pixelgl.Transform, mat)
+		ctx.Shader().SetUniformAttr(maskColorVec4, mgl32.Vec4{r, g, b, a})
+		ctx.Shader().SetUniformAttr(transformMat3, mat)
+
+		if s.picture.Texture() != nil {
+			s.picture.Texture().Do(func(pixelgl.Context) {
+				s.va.Draw()
+			})
+		} else {
+			s.va.Draw()
+		}
+	})
+}
+
+// Sprite is a picture that can be drawn on the screen. Optionally it can be color masked or tranformed.
+//
+// Usually, you only transform objects when you're drawing them (by passing transforms to the Draw method).
+// With sprites however, it can be useful to also transform them "statically". For example, sprites are
+// anchor by their bottom-left corner by default. Setting a transform can change this anchor to the center,
+// or wherever you want.
+type Sprite struct {
+	*Shape
+}
 
-		s.va.Draw()
+// NewSprite creates a new sprite with the supplied picture. The sprite's size is the size of the supplied picture.
+// If you want to change the sprite's size, change it's transform.
+func NewSprite(parent pixelgl.Doer, picture Picture) *Sprite {
+	var va *pixelgl.VertexArray
+
+	parent.Do(func(ctx pixelgl.Context) {
+		var err error
+		va, err = pixelgl.NewVertexArray(
+			picture.Texture(),
+			ctx.Shader().VertexFormat(),
+			pixelgl.DynamicUsage,
+			4,
+			[]int{0, 1, 2, 0, 2, 3},
+		)
+		if err != nil {
+			panic(errors.Wrap(err, "failed to create sprite"))
+		}
 	})
+
+	w, h := picture.Bounds().Size.XY()
+	for i, p := range []Vec{V(0, 0), V(w, 0), V(w, h), V(0, h)} {
+		texCoord := V(
+			(picture.Bounds().X()+p.X())/float64(picture.Texture().Width()),
+			(picture.Bounds().Y()+p.Y())/float64(picture.Texture().Height()),
+		)
+
+		va.SetVertexAttr(i, positionVec2, mgl32.Vec2{float32(p.X()), float32(p.Y())})
+		va.SetVertexAttr(i, colorVec4, mgl32.Vec4{1, 1, 1, 1})
+		va.SetVertexAttr(i, texCoordVec2, mgl32.Vec2{float32(texCoord.X()), float32(texCoord.Y())})
+	}
+
+	return &Sprite{NewShape(parent, picture, color.White, Position(0), va)}
 }
 
 // LineColor a line shape (with sharp ends) filled with a single color.
 type LineColor struct {
-	parent pixelgl.Doer
-	color  color.Color
-	a, b   Vec
-	width  float64
-	va     *pixelgl.VertexArray
+	*Shape
+	a, b  Vec
+	width float64
 }
 
 // NewLineColor creates a new line shape between points A and B filled with a single color. Parent is an object
 // that this shape belongs to, such as a window, or a graphics effect.
 func NewLineColor(parent pixelgl.Doer, c color.Color, a, b Vec, width float64) *LineColor {
-	lc := &LineColor{
-		parent: parent,
-		color:  c,
-		a:      a,
-		b:      b,
-		width:  width,
-	}
+	var va *pixelgl.VertexArray
 
 	parent.Do(func(ctx pixelgl.Context) {
 		var err error
-		lc.va, err = pixelgl.NewVertexArray(
+		va, err = pixelgl.NewVertexArray(
 			parent,
 			ctx.Shader().VertexFormat(),
 			pixelgl.DynamicUsage,
@@ -240,12 +244,12 @@ func NewLineColor(parent pixelgl.Doer, c color.Color, a, b Vec, width float64) *
 	})
 
 	for i := 0; i < 4; i++ {
-		lc.va.SetVertexAttributeVec4(i, pixelgl.Color, mgl32.Vec4{1, 1, 1, 1})
-		lc.va.SetVertexAttributeVec2(i, pixelgl.TexCoord, mgl32.Vec2{-1, -1})
+		va.SetVertexAttr(i, colorVec4, mgl32.Vec4{1, 1, 1, 1})
+		va.SetVertexAttr(i, texCoordVec2, mgl32.Vec2{-1, -1})
 	}
 
+	lc := &LineColor{NewShape(parent, Picture{}, c, Position(0), va), a, b, width}
 	lc.setPoints()
-
 	return lc
 }
 
@@ -253,20 +257,10 @@ func NewLineColor(parent pixelgl.Doer, c color.Color, a, b Vec, width float64) *
 func (lc *LineColor) setPoints() {
 	r := (lc.b - lc.a).Unit().Scaled(lc.width / 2).Rotated(math.Pi / 2)
 	for i, p := range []Vec{lc.a - r, lc.a + r, lc.b - r, lc.b + r} {
-		lc.va.SetVertexAttributeVec2(i, pixelgl.Position, mgl32.Vec2{float32(p.X()), float32(p.Y())})
+		lc.va.SetVertexAttr(i, positionVec2, mgl32.Vec2{float32(p.X()), float32(p.Y())})
 	}
 }
 
-// SetColor changes the color of a line.
-func (lc *LineColor) SetColor(c color.Color) {
-	lc.color = c
-}
-
-// Color returns the current color of a line.
-func (lc *LineColor) Color() color.Color {
-	return lc.color
-}
-
 // SetA changes the position of the first endpoint of a line.
 func (lc *LineColor) SetA(a Vec) {
 	lc.a = a
@@ -300,43 +294,16 @@ func (lc *LineColor) Width() float64 {
 	return lc.width
 }
 
-// Draw draws a line transformed by the supplied transforms applied in the reverse order.
-func (lc *LineColor) Draw(t ...Transform) {
-	mat := mgl32.Ident3()
-	for i := range t {
-		mat = mat.Mul3(t[i].Mat3())
-	}
-
-	lc.parent.Do(func(ctx pixelgl.Context) {
-		r, g, b, a := colorToRGBA(lc.color)
-		ctx.Shader().SetUniformVec4(pixelgl.MaskColor, mgl32.Vec4{r, g, b, a})
-		ctx.Shader().SetUniformMat3(pixelgl.Transform, mat)
-	})
-
-	lc.va.Draw()
-}
-
-// Delete destroys a line shape and releases it's video memory. Do not use this shape after calling Delete.
-func (lc *LineColor) Delete() {
-	lc.va.Delete()
-}
-
-// PolygonColor is a polygon shape filled with a single color.
+// PolygonColor is a convex polygon shape filled with a single color.
 type PolygonColor struct {
-	parent pixelgl.Doer
-	color  color.Color
+	*Shape
 	points []Vec
-	va     *pixelgl.VertexArray
 }
 
 // NewPolygonColor creates a new polygon shape filled with a single color. Parent is an object that this shape belongs to,
 // such as a window, or a graphics effect.
 func NewPolygonColor(parent pixelgl.Doer, c color.Color, points ...Vec) *PolygonColor {
-	pc := &PolygonColor{
-		parent: parent,
-		color:  c,
-		points: points,
-	}
+	var va *pixelgl.VertexArray
 
 	var indices []int
 	for i := 2; i < len(points); i++ {
@@ -345,7 +312,7 @@ func NewPolygonColor(parent pixelgl.Doer, c color.Color, points ...Vec) *Polygon
 
 	parent.Do(func(ctx pixelgl.Context) {
 		var err error
-		pc.va, err = pixelgl.NewVertexArray(
+		va, err = pixelgl.NewVertexArray(
 			parent,
 			ctx.Shader().VertexFormat(),
 			pixelgl.DynamicUsage,
@@ -358,50 +325,25 @@ func NewPolygonColor(parent pixelgl.Doer, c color.Color, points ...Vec) *Polygon
 	})
 
 	for i, p := range points {
-		pc.va.SetVertexAttributeVec2(
-			i,
-			pixelgl.Position,
-			mgl32.Vec2{
-				float32(p.X()),
-				float32(p.Y()),
-			},
-		)
-		pc.va.SetVertexAttributeVec4(
-			i,
-			pixelgl.Color,
-			mgl32.Vec4{1, 1, 1, 1},
-		)
-		pc.va.SetVertexAttributeVec2(
-			i,
-			pixelgl.TexCoord,
-			mgl32.Vec2{-1, -1},
-		)
+		va.SetVertexAttr(i, positionVec2, mgl32.Vec2{float32(p.X()), float32(p.Y())})
+		va.SetVertexAttr(i, colorVec4, mgl32.Vec4{1, 1, 1, 1})
+		va.SetVertexAttr(i, texCoordVec2, mgl32.Vec2{-1, -1})
 	}
 
-	return pc
+	return &PolygonColor{NewShape(parent, Picture{}, c, Position(0), va), points}
 }
 
-// Count returns the number of points in a polygon.
-func (pc *PolygonColor) Count() int {
+// PointNum returns the number of points in a polygon.
+func (pc *PolygonColor) PointNum() int {
 	return len(pc.points)
 }
 
-// SetColor changes the color of a polygon to c.
-func (pc *PolygonColor) SetColor(c color.Color) {
-	pc.color = c
-}
-
-// Color returns the current color of a polygon.
-func (pc *PolygonColor) Color() color.Color {
-	return pc.color
-}
-
 // SetPoint changes the position of a point in a polygon.
 //
 // If the index is out of range, this function panics.
 func (pc *PolygonColor) SetPoint(i int, point Vec) {
 	pc.points[i] = point
-	pc.va.SetVertexAttributeVec2(i, pixelgl.Position, mgl32.Vec2{float32(point.X()), float32(point.Y())})
+	pc.va.SetVertexAttr(i, positionVec2, mgl32.Vec2{float32(point.X()), float32(point.Y())})
 }
 
 // Point returns the position of a point in a polygon.
@@ -411,34 +353,11 @@ func (pc *PolygonColor) Point(i int) Vec {
 	return pc.points[i]
 }
 
-// Draw draws a polygon transformed by the supplied transforms applied in the reverse order.
-func (pc *PolygonColor) Draw(t ...Transform) {
-	mat := mgl32.Ident3()
-	for i := range t {
-		mat = mat.Mul3(t[i].Mat3())
-	}
-
-	pc.parent.Do(func(ctx pixelgl.Context) {
-		r, g, b, a := colorToRGBA(pc.color)
-		ctx.Shader().SetUniformVec4(pixelgl.MaskColor, mgl32.Vec4{r, g, b, a})
-		ctx.Shader().SetUniformMat3(pixelgl.Transform, mat)
-	})
-
-	pc.va.Draw()
-}
-
-// Delete destroys a polygon shape and releases it's video memory. Do not use this shape after calling Delete.
-func (pc *PolygonColor) Delete() {
-	pc.va.Delete()
-}
-
 // EllipseColor is an ellipse shape filled with a single color.
 type EllipseColor struct {
-	parent pixelgl.Doer
-	color  color.Color
+	*Shape
 	radius Vec
 	fill   float64
-	va     *pixelgl.VertexArray
 }
 
 // NewEllipseColor creates a new ellipse shape filled with a single color. Parent is an object that this shape belongs to,
@@ -446,14 +365,9 @@ type EllipseColor struct {
 // be filled (from the outside). The value of 1 means that the whole ellipse is filled. The value of 0 means that none of the
 // ellipse is filled (which makes the ellipse invisible).
 func NewEllipseColor(parent pixelgl.Doer, c color.Color, radius Vec, fill float64) *EllipseColor {
-	const n = 256
+	var va *pixelgl.VertexArray
 
-	ec := &EllipseColor{
-		parent: parent,
-		color:  c,
-		radius: radius,
-		fill:   fill,
-	}
+	const n = 256
 
 	var indices []int
 	for i := 2; i < (n+1)*2; i++ {
@@ -462,7 +376,7 @@ func NewEllipseColor(parent pixelgl.Doer, c color.Color, radius Vec, fill float6
 
 	parent.Do(func(ctx pixelgl.Context) {
 		var err error
-		ec.va, err = pixelgl.NewVertexArray(
+		va, err = pixelgl.NewVertexArray(
 			parent,
 			ctx.Shader().VertexFormat(),
 			pixelgl.DynamicUsage,
@@ -477,85 +391,31 @@ func NewEllipseColor(parent pixelgl.Doer, c color.Color, radius Vec, fill float6
 	for k := 0; k < n+1; k++ {
 		i, j := k*2, k*2+1
 		angle := math.Pi * 2 * float64(k%n) / n
-		ec.va.SetVertexAttributeVec2(
-			i,
-			pixelgl.Position,
-			mgl32.Vec2{
-				float32(math.Cos(angle)),
-				float32(math.Sin(angle)),
-			},
-		)
-		ec.va.SetVertexAttributeVec4(
-			i,
-			pixelgl.Color,
-			mgl32.Vec4{1, 1, 1, 1},
-		)
-		ec.va.SetVertexAttributeVec2(
-			i,
-			pixelgl.TexCoord,
-			mgl32.Vec2{-1, -1},
-		)
-		ec.va.SetVertexAttributeVec2(
-			j,
-			pixelgl.Position,
-			mgl32.Vec2{
-				float32(math.Cos(angle) * (1 - fill)),
-				float32(math.Sin(angle) * (1 - fill)),
-			},
-		)
-		ec.va.SetVertexAttributeVec4(
-			j,
-			pixelgl.Color,
-			mgl32.Vec4{1, 1, 1, 1},
-		)
-		ec.va.SetVertexAttributeVec2(
-			j,
-			pixelgl.TexCoord,
-			mgl32.Vec2{-1, -1},
-		)
-	}
 
-	return ec
-}
-
-// SetColor changes the color of an ellipse.
-func (ec *EllipseColor) SetColor(c color.Color) {
-	ec.color = c
-}
-
-// Color returns the current color of an ellipse.
-func (ec *EllipseColor) Color() color.Color {
-	return ec.color
-}
+		va.SetVertexAttr(i, positionVec2, mgl32.Vec2{
+			float32(math.Cos(angle) * radius.X()),
+			float32(math.Sin(angle) * radius.Y()),
+		})
+		va.SetVertexAttr(i, colorVec4, mgl32.Vec4{1, 1, 1, 1})
+		va.SetVertexAttr(i, texCoordVec2, mgl32.Vec2{-1, -1})
+
+		va.SetVertexAttr(j, positionVec2, mgl32.Vec2{
+			float32(math.Cos(angle) * radius.X() * (1 - fill)),
+			float32(math.Sin(angle) * radius.Y() * (1 - fill)),
+		})
+		va.SetVertexAttr(j, colorVec4, mgl32.Vec4{1, 1, 1, 1})
+		va.SetVertexAttr(j, texCoordVec2, mgl32.Vec2{-1, -1})
+	}
 
-// SetRadius sets the radius (which can be different in X and Y axis) of an ellipse.
-func (ec *EllipseColor) SetRadius(radius Vec) {
-	ec.radius = radius
+	return &EllipseColor{NewShape(parent, Picture{}, c, Position(0), va), radius, fill}
 }
 
-// Radius returns the current radius of an ellipse.
+// Radius returns the radius of an ellipse.
 func (ec *EllipseColor) Radius() Vec {
 	return ec.radius
 }
 
-// Draw dras an ellipse transformed by the supplied transforms applied in the reverse order.
-func (ec *EllipseColor) Draw(t ...Transform) {
-	mat := mgl32.Ident3()
-	for i := range t {
-		mat = mat.Mul3(t[i].Mat3())
-	}
-	mat = mat.Mul3(mgl32.Scale2D(float32(ec.radius.X()), float32(ec.radius.Y())))
-
-	ec.parent.Do(func(ctx pixelgl.Context) {
-		r, g, b, a := colorToRGBA(ec.color)
-		ctx.Shader().SetUniformVec4(pixelgl.MaskColor, mgl32.Vec4{r, g, b, a})
-		ctx.Shader().SetUniformMat3(pixelgl.Transform, mat)
-	})
-
-	ec.va.Draw()
-}
-
-// Delete destroys an ellipse shape and releases it's video memory. Do not use this shape after calling Delete.
-func (ec *EllipseColor) Delete() {
-	ec.va.Delete()
+// Fill returns the fill ratio of an ellipse.
+func (ec *EllipseColor) Fill() float64 {
+	return ec.fill
 }
-- 
GitLab