go-opengl-pixel/sprite.go

120 lines
3.0 KiB
Go
Raw Normal View History

2017-03-07 15:47:55 -06:00
package pixel
import "image/color"
2017-03-31 08:00:59 -05:00
// Sprite is a drawable frame of a Picture. It's anchored by the center of it's Picture's frame.
//
// Frame specifies a rectangular portion of the Picture that will be drawn. For example, this
// creates a Sprite that draws the whole Picture:
//
// sprite := pixel.NewSprite(pic, pic.Bounds())
//
// To achieve different anchoring, transformations and color masking, use SetMatrix and SetColorMask
// methods.
//
// Note, that Sprite caches the results of MakePicture from Targets it's drawn to for each Picture
// it's set to. What it means is that using a Sprite with an unbounded number of Pictures leads to a
// memory leak, since Sprite caches them and never forgets. In such a situation, create a new Sprite
// for each Picture.
2017-03-07 15:47:55 -06:00
type Sprite struct {
2017-03-31 08:00:59 -05:00
tri *TrianglesData
frame Rect
d Drawer
matrix Matrix
mask RGBA
2017-03-07 15:47:55 -06:00
}
2017-03-31 08:00:59 -05:00
// NewSprite creates a Sprite from the supplied frame of a Picture.
func NewSprite(pic Picture, frame Rect) *Sprite {
2017-03-07 15:47:55 -06:00
tri := MakeTrianglesData(6)
s := &Sprite{
tri: tri,
d: Drawer{Triangles: tri},
}
s.matrix = IM
2017-04-10 10:25:56 -05:00
s.mask = Alpha(1)
2017-03-31 08:00:59 -05:00
s.Set(pic, frame)
2017-03-07 15:47:55 -06:00
return s
}
2017-03-31 08:00:59 -05:00
// Set sets a new frame of a Picture for this Sprite.
func (s *Sprite) Set(pic Picture, frame Rect) {
2017-03-07 17:43:01 -06:00
s.d.Picture = pic
2017-03-31 08:00:59 -05:00
if frame != s.frame {
s.frame = frame
s.calcData()
2017-03-07 17:43:01 -06:00
}
}
// Picture returns the current Sprite's Picture.
func (s *Sprite) Picture() Picture {
return s.d.Picture
}
2017-03-31 08:00:59 -05:00
// Frame returns the current Sprite's frame.
func (s *Sprite) Frame() Rect {
return s.frame
}
// Draw draws the Sprite onto the provided Target. The Sprite will be transformed by the given Matrix.
//
// This method is equivalent to calling DrawColorMask with nil color mask.
func (s *Sprite) Draw(t Target, matrix Matrix) {
s.DrawColorMask(t, matrix, nil)
}
// DrawColorMask draw the Sprite onto the provided Target. The Sprite will be transformed by the
// given Matrix and all of it's color will be multiplied by the given mask.
//
// If the mask is nil, a fully opaque white mask will be used, which causes no effect.
func (s *Sprite) DrawColorMask(t Target, matrix Matrix, mask color.Color) {
dirty := false
if matrix != s.matrix {
s.matrix = matrix
dirty = true
}
if mask == nil {
mask = Alpha(1)
}
if mask != s.mask {
2017-04-12 09:18:25 -05:00
s.mask = ToRGBA(mask)
dirty = true
2017-04-12 09:18:25 -05:00
}
if dirty {
s.calcData()
}
s.d.Draw(t)
}
func (s *Sprite) calcData() {
2017-03-07 15:47:55 -06:00
var (
2017-03-31 08:00:59 -05:00
center = s.frame.Center()
horizontal = X(s.frame.W() / 2)
vertical = Y(s.frame.H() / 2)
2017-03-07 15:47:55 -06:00
)
2017-03-07 15:50:49 -06:00
2017-03-07 15:47:55 -06:00
(*s.tri)[0].Position = -horizontal - vertical
(*s.tri)[1].Position = +horizontal - vertical
(*s.tri)[2].Position = +horizontal + vertical
(*s.tri)[3].Position = -horizontal - vertical
(*s.tri)[4].Position = +horizontal + vertical
(*s.tri)[5].Position = -horizontal + vertical
2017-03-07 15:50:49 -06:00
2017-03-07 15:47:55 -06:00
for i := range *s.tri {
(*s.tri)[i].Color = s.mask
2017-03-07 15:47:55 -06:00
(*s.tri)[i].Picture = center + (*s.tri)[i].Position
(*s.tri)[i].Intensity = 1
2017-03-23 14:27:51 -05:00
}
2017-03-23 14:27:51 -05:00
// matrix and mask
for i := range *s.tri {
(*s.tri)[i].Position = s.matrix.Project((*s.tri)[i].Position)
2017-03-23 14:27:51 -05:00
(*s.tri)[i].Color = s.mask
2017-03-07 15:47:55 -06:00
}
s.d.Dirty()
}