2016-12-03 09:25:41 -06:00
|
|
|
package pixel
|
|
|
|
|
2017-01-25 12:50:49 -06:00
|
|
|
import (
|
|
|
|
"fmt"
|
|
|
|
"image/color"
|
|
|
|
)
|
2016-12-03 09:25:41 -06:00
|
|
|
|
2017-01-04 17:19:45 -06:00
|
|
|
// TrianglesData specifies a list of Triangles vertices with three common properties: Position,
|
|
|
|
// Color and Texture.
|
|
|
|
type TrianglesData []struct {
|
|
|
|
Position Vec
|
2017-01-10 16:54:35 -06:00
|
|
|
Color NRGBA
|
2017-01-04 17:19:45 -06:00
|
|
|
Texture Vec
|
2016-12-06 10:53:10 -06:00
|
|
|
}
|
|
|
|
|
2017-01-28 13:01:59 -06:00
|
|
|
// MakeTrianglesData creates TrianglesData of length len initialized with default property values.
|
|
|
|
//
|
|
|
|
// Prefer this function to make(TrianglesData, len), because make zeros them, while this function
|
|
|
|
// does a correct intialization.
|
|
|
|
func MakeTrianglesData(len int) TrianglesData {
|
|
|
|
td := TrianglesData{}
|
|
|
|
td.SetLen(len)
|
|
|
|
return td
|
|
|
|
}
|
|
|
|
|
|
|
|
// Len returns the number of vertices in TrianglesData.
|
|
|
|
func (td *TrianglesData) Len() int {
|
|
|
|
return len(*td)
|
|
|
|
}
|
|
|
|
|
2017-01-26 16:38:37 -06:00
|
|
|
// SetLen resizes TrianglesData to len, while keeping the original content.
|
|
|
|
//
|
|
|
|
// If len is greater than TrianglesData's current length, the new data is filled with default
|
|
|
|
// values ((0, 0), white, (-1, -1)).
|
|
|
|
func (td *TrianglesData) SetLen(len int) {
|
2017-01-12 18:53:52 -06:00
|
|
|
if len > td.Len() {
|
2017-01-13 17:47:49 -06:00
|
|
|
needAppend := len - td.Len()
|
|
|
|
for i := 0; i < needAppend; i++ {
|
|
|
|
*td = append(*td, struct {
|
|
|
|
Position Vec
|
|
|
|
Color NRGBA
|
|
|
|
Texture Vec
|
|
|
|
}{V(0, 0), NRGBA{1, 1, 1, 1}, V(-1, -1)})
|
2017-01-12 18:53:52 -06:00
|
|
|
}
|
2016-12-06 10:53:10 -06:00
|
|
|
}
|
2017-01-12 18:53:52 -06:00
|
|
|
if len < td.Len() {
|
|
|
|
*td = (*td)[:len]
|
2017-01-01 15:12:12 -06:00
|
|
|
}
|
2017-01-12 18:53:52 -06:00
|
|
|
}
|
2017-01-01 15:12:12 -06:00
|
|
|
|
2017-01-28 13:01:59 -06:00
|
|
|
// Slice returns a sub-Triangles of this TrianglesData.
|
|
|
|
func (td *TrianglesData) Slice(i, j int) Triangles {
|
|
|
|
s := TrianglesData((*td)[i:j])
|
|
|
|
return &s
|
2017-01-26 16:40:29 -06:00
|
|
|
}
|
|
|
|
|
2017-01-28 13:01:59 -06:00
|
|
|
func (td *TrianglesData) updateData(t Triangles) {
|
2017-01-04 17:19:45 -06:00
|
|
|
// fast path optimization
|
|
|
|
if t, ok := t.(*TrianglesData); ok {
|
2017-01-28 13:01:59 -06:00
|
|
|
copy(*td, *t)
|
2017-01-04 17:19:45 -06:00
|
|
|
return
|
|
|
|
}
|
2016-12-14 10:53:29 -06:00
|
|
|
|
2017-01-04 17:19:45 -06:00
|
|
|
// slow path manual copy
|
|
|
|
if t, ok := t.(TrianglesPosition); ok {
|
2017-01-28 13:01:59 -06:00
|
|
|
for i := range *td {
|
2017-01-04 17:19:45 -06:00
|
|
|
(*td)[i].Position = t.Position(i)
|
2016-12-14 10:53:29 -06:00
|
|
|
}
|
2017-01-04 17:19:45 -06:00
|
|
|
}
|
|
|
|
if t, ok := t.(TrianglesColor); ok {
|
2017-01-28 13:01:59 -06:00
|
|
|
for i := range *td {
|
2017-01-12 18:53:52 -06:00
|
|
|
(*td)[i].Color = t.Color(i)
|
2016-12-17 19:06:50 -06:00
|
|
|
}
|
|
|
|
}
|
2017-01-04 17:19:45 -06:00
|
|
|
if t, ok := t.(TrianglesTexture); ok {
|
2017-01-28 13:01:59 -06:00
|
|
|
for i := range *td {
|
2017-01-04 17:19:45 -06:00
|
|
|
(*td)[i].Texture = t.Texture(i)
|
2016-12-20 07:56:45 -06:00
|
|
|
}
|
2016-12-20 07:23:50 -06:00
|
|
|
}
|
2016-12-17 19:06:50 -06:00
|
|
|
}
|
|
|
|
|
2017-01-12 18:53:52 -06:00
|
|
|
// Update copies vertex properties from the supplied Triangles into this TrianglesData.
|
|
|
|
//
|
|
|
|
// TrianglesPosition, TrianglesColor and TrianglesTexture are supported.
|
|
|
|
func (td *TrianglesData) Update(t Triangles) {
|
2017-01-28 13:01:59 -06:00
|
|
|
if td.Len() != t.Len() {
|
|
|
|
panic(fmt.Errorf("%T.Update: invalid triangles length", td))
|
|
|
|
}
|
|
|
|
td.updateData(t)
|
2017-01-11 06:16:27 -06:00
|
|
|
}
|
|
|
|
|
2017-01-12 04:44:54 -06:00
|
|
|
// Copy returns an exact independent copy of this TrianglesData.
|
|
|
|
func (td *TrianglesData) Copy() Triangles {
|
2017-01-28 13:01:59 -06:00
|
|
|
copyTd := TrianglesData{}
|
|
|
|
copyTd.SetLen(td.Len())
|
2017-01-12 04:44:54 -06:00
|
|
|
copyTd.Update(td)
|
|
|
|
return ©Td
|
|
|
|
}
|
|
|
|
|
2017-01-04 17:19:45 -06:00
|
|
|
// Position returns the position property of i-th vertex.
|
|
|
|
func (td *TrianglesData) Position(i int) Vec {
|
|
|
|
return (*td)[i].Position
|
2016-12-14 10:53:29 -06:00
|
|
|
}
|
2016-12-06 10:53:10 -06:00
|
|
|
|
2017-01-04 17:19:45 -06:00
|
|
|
// Color returns the color property of i-th vertex.
|
2017-01-12 18:07:13 -06:00
|
|
|
func (td *TrianglesData) Color(i int) NRGBA {
|
2017-01-04 17:19:45 -06:00
|
|
|
return (*td)[i].Color
|
2016-12-06 10:53:10 -06:00
|
|
|
}
|
|
|
|
|
2017-01-04 17:19:45 -06:00
|
|
|
// Texture returns the texture property of i-th vertex.
|
|
|
|
func (td *TrianglesData) Texture(i int) Vec {
|
|
|
|
return (*td)[i].Texture
|
2016-12-03 13:20:16 -06:00
|
|
|
}
|
|
|
|
|
2017-01-04 17:19:45 -06:00
|
|
|
// TrianglesDrawer is a helper type that wraps Triangles and turns them into a Drawer.
|
2016-12-03 09:25:41 -06:00
|
|
|
//
|
2017-01-04 17:19:45 -06:00
|
|
|
// It does so by creating a separate Triangles instance for each Target. The instances are
|
|
|
|
// correctly updated alongside the wrapped Triangles.
|
|
|
|
type TrianglesDrawer struct {
|
|
|
|
Triangles
|
2016-12-03 09:25:41 -06:00
|
|
|
|
2017-01-28 13:01:59 -06:00
|
|
|
tris map[Target]TargetTriangles
|
2017-01-04 17:19:45 -06:00
|
|
|
dirty bool
|
2016-12-03 12:51:43 -06:00
|
|
|
}
|
|
|
|
|
2017-01-04 17:19:45 -06:00
|
|
|
func (td *TrianglesDrawer) flush() {
|
|
|
|
if !td.dirty {
|
|
|
|
return
|
2016-12-08 08:25:00 -06:00
|
|
|
}
|
2017-01-04 17:19:45 -06:00
|
|
|
td.dirty = false
|
2016-12-08 08:25:00 -06:00
|
|
|
|
2017-01-04 17:19:45 -06:00
|
|
|
for _, t := range td.tris {
|
2017-02-02 19:09:06 -06:00
|
|
|
t.SetLen(td.Len())
|
2017-01-04 17:19:45 -06:00
|
|
|
t.Update(td.Triangles)
|
2016-12-14 10:53:29 -06:00
|
|
|
}
|
2017-01-04 17:19:45 -06:00
|
|
|
}
|
2016-12-03 12:51:43 -06:00
|
|
|
|
2017-01-04 17:19:45 -06:00
|
|
|
// Draw draws the wrapped Triangles onto the provided Target.
|
|
|
|
func (td *TrianglesDrawer) Draw(target Target) {
|
|
|
|
if td.tris == nil {
|
2017-01-28 13:01:59 -06:00
|
|
|
td.tris = make(map[Target]TargetTriangles)
|
2017-01-04 17:19:45 -06:00
|
|
|
}
|
2016-12-19 18:23:33 -06:00
|
|
|
|
2017-01-04 17:19:45 -06:00
|
|
|
td.flush()
|
2016-12-03 12:51:43 -06:00
|
|
|
|
2017-01-04 17:19:45 -06:00
|
|
|
tri := td.tris[target]
|
|
|
|
if tri == nil {
|
|
|
|
tri = target.MakeTriangles(td.Triangles)
|
|
|
|
td.tris[target] = tri
|
|
|
|
}
|
|
|
|
tri.Draw()
|
2016-12-03 12:51:43 -06:00
|
|
|
}
|
|
|
|
|
2017-01-28 13:01:59 -06:00
|
|
|
// Dirty marks the underlying container as changed (dirty). Always call this when you change the
|
|
|
|
// underlying Triangles (by Update, SetLen, etc.).
|
2017-01-25 15:02:53 -06:00
|
|
|
func (td *TrianglesDrawer) Dirty() {
|
|
|
|
td.dirty = true
|
|
|
|
}
|
|
|
|
|
2017-01-25 10:33:33 -06:00
|
|
|
// Sprite is a picture that can be drawn onto a Target. To change the position/rotation/scale of
|
|
|
|
// the Sprite, use Target's SetTransform method.
|
2017-01-10 16:54:35 -06:00
|
|
|
type Sprite struct {
|
2017-01-25 15:02:53 -06:00
|
|
|
data TrianglesData
|
2017-01-25 15:32:02 -06:00
|
|
|
td TrianglesDrawer
|
2017-01-25 12:04:37 -06:00
|
|
|
pic *Picture
|
2017-01-10 16:54:35 -06:00
|
|
|
}
|
|
|
|
|
|
|
|
// NewSprite creates a Sprite with the supplied Picture. The dimensions of the returned Sprite match
|
|
|
|
// the dimensions of the Picture.
|
|
|
|
func NewSprite(pic *Picture) *Sprite {
|
|
|
|
s := &Sprite{
|
2017-01-25 15:02:53 -06:00
|
|
|
data: TrianglesData{
|
|
|
|
{Position: V(0, 0), Color: NRGBA{1, 1, 1, 1}, Texture: V(0, 0)},
|
|
|
|
{Position: V(0, 0), Color: NRGBA{1, 1, 1, 1}, Texture: V(1, 0)},
|
|
|
|
{Position: V(0, 0), Color: NRGBA{1, 1, 1, 1}, Texture: V(1, 1)},
|
|
|
|
{Position: V(0, 0), Color: NRGBA{1, 1, 1, 1}, Texture: V(0, 0)},
|
|
|
|
{Position: V(0, 0), Color: NRGBA{1, 1, 1, 1}, Texture: V(1, 1)},
|
|
|
|
{Position: V(0, 0), Color: NRGBA{1, 1, 1, 1}, Texture: V(0, 1)},
|
|
|
|
},
|
2017-01-10 16:54:35 -06:00
|
|
|
}
|
2017-01-25 15:02:53 -06:00
|
|
|
s.td = TrianglesDrawer{Triangles: &s.data}
|
2017-01-10 16:54:35 -06:00
|
|
|
s.SetPicture(pic)
|
|
|
|
return s
|
|
|
|
}
|
|
|
|
|
|
|
|
// SetPicture changes the Picture of the Sprite and resizes it accordingly.
|
|
|
|
func (s *Sprite) SetPicture(pic *Picture) {
|
2017-01-25 12:55:41 -06:00
|
|
|
oldPic := s.pic
|
|
|
|
s.pic = pic
|
2017-01-25 12:57:42 -06:00
|
|
|
if oldPic != nil && oldPic.Bounds().Size == pic.Bounds().Size {
|
2017-01-25 12:55:41 -06:00
|
|
|
return
|
|
|
|
}
|
2017-01-10 16:54:35 -06:00
|
|
|
w, h := pic.Bounds().Size.XY()
|
2017-01-25 15:02:53 -06:00
|
|
|
s.data[0].Position = V(0, 0)
|
|
|
|
s.data[1].Position = V(w, 0)
|
2017-01-25 15:43:10 -06:00
|
|
|
s.data[2].Position = V(w, h)
|
2017-01-25 15:02:53 -06:00
|
|
|
s.data[3].Position = V(0, 0)
|
|
|
|
s.data[4].Position = V(w, h)
|
|
|
|
s.data[5].Position = V(0, h)
|
|
|
|
s.td.Dirty()
|
2017-01-10 16:54:35 -06:00
|
|
|
}
|
|
|
|
|
|
|
|
// Picture returns the current Picture of the Sprite.
|
|
|
|
func (s *Sprite) Picture() *Picture {
|
|
|
|
return s.pic
|
|
|
|
}
|
|
|
|
|
|
|
|
// Draw draws the Sprite onto the provided Target.
|
2017-01-25 12:50:49 -06:00
|
|
|
func (s *Sprite) Draw(t Target) {
|
|
|
|
t.SetPicture(s.pic)
|
|
|
|
s.td.Draw(t)
|
|
|
|
}
|
|
|
|
|
|
|
|
// Polygon is a convex polygon shape filled with a single color.
|
|
|
|
type Polygon struct {
|
2017-01-25 15:02:53 -06:00
|
|
|
data TrianglesData
|
2017-01-25 15:32:02 -06:00
|
|
|
td TrianglesDrawer
|
2017-01-25 12:50:49 -06:00
|
|
|
col NRGBA
|
|
|
|
}
|
|
|
|
|
|
|
|
// NewPolygon creates a Polygon with specified color and points. Points can be in clock-wise or
|
|
|
|
// counter-clock-wise order, it doesn't matter. They should however form a convex polygon.
|
|
|
|
func NewPolygon(c color.Color, points ...Vec) *Polygon {
|
|
|
|
p := &Polygon{
|
2017-01-28 13:01:59 -06:00
|
|
|
data: TrianglesData{},
|
2017-01-25 12:50:49 -06:00
|
|
|
}
|
2017-01-25 15:02:53 -06:00
|
|
|
p.td = TrianglesDrawer{Triangles: &p.data}
|
2017-01-25 12:50:49 -06:00
|
|
|
p.SetColor(c)
|
|
|
|
p.SetPoints(points...)
|
|
|
|
return p
|
|
|
|
}
|
|
|
|
|
|
|
|
// SetColor changes the color of the Polygon.
|
|
|
|
//
|
|
|
|
// If the Polygon is very large, this method might end up being too expensive. Consider using
|
|
|
|
// a color mask on a Target, in such a case.
|
|
|
|
func (p *Polygon) SetColor(c color.Color) {
|
|
|
|
p.col = NRGBAModel.Convert(c).(NRGBA)
|
2017-01-25 15:02:53 -06:00
|
|
|
for i := range p.data {
|
|
|
|
p.data[i].Color = p.col
|
2017-01-25 12:50:49 -06:00
|
|
|
}
|
2017-01-25 15:02:53 -06:00
|
|
|
p.td.Dirty()
|
2017-01-25 12:50:49 -06:00
|
|
|
}
|
|
|
|
|
|
|
|
// Color returns the current color of the Polygon.
|
|
|
|
func (p *Polygon) Color() NRGBA {
|
|
|
|
return p.col
|
|
|
|
}
|
|
|
|
|
|
|
|
// SetPoints sets the points of the Polygon. The number of points might differ from the original
|
|
|
|
// count.
|
|
|
|
//
|
|
|
|
// This method is more effective, than creating a new Polygon with the given points.
|
2017-01-25 12:53:15 -06:00
|
|
|
//
|
|
|
|
// However, it is less expensive than using a transform on a Target.
|
2017-01-25 12:50:49 -06:00
|
|
|
func (p *Polygon) SetPoints(points ...Vec) {
|
2017-01-26 16:38:37 -06:00
|
|
|
p.data.SetLen(len(points))
|
2017-01-25 12:50:49 -06:00
|
|
|
for i, pt := range points {
|
2017-01-25 15:02:53 -06:00
|
|
|
p.data[i].Position = pt
|
|
|
|
p.data[i].Color = p.col
|
2017-01-25 12:50:49 -06:00
|
|
|
}
|
2017-01-25 15:02:53 -06:00
|
|
|
p.td.Dirty()
|
2017-01-25 12:50:49 -06:00
|
|
|
}
|
|
|
|
|
|
|
|
// Points returns a slice of points of the Polygon in the order they where supplied.
|
|
|
|
func (p *Polygon) Points() []Vec {
|
|
|
|
points := make([]Vec, p.data.Len())
|
2017-01-25 15:02:53 -06:00
|
|
|
for i := range p.data {
|
|
|
|
points[i] = p.data[i].Position
|
2017-01-25 12:50:49 -06:00
|
|
|
}
|
|
|
|
return points
|
|
|
|
}
|
|
|
|
|
|
|
|
// Draw draws the Polygon onto the Target.
|
|
|
|
func (p *Polygon) Draw(t Target) {
|
|
|
|
t.SetPicture(nil)
|
|
|
|
p.td.Draw(t)
|
2017-01-10 16:54:35 -06:00
|
|
|
}
|