go-opengl-pixel/pixelgl/canvas.go

486 lines
11 KiB
Go
Raw Normal View History

package pixelgl
import (
2017-03-05 17:28:52 -06:00
"fmt"
"image/color"
2017-03-06 06:05:45 -06:00
"math"
2017-02-11 07:09:47 -06:00
"github.com/faiface/glhf"
"github.com/faiface/mainthread"
"github.com/faiface/pixel"
"github.com/go-gl/mathgl/mgl32"
"github.com/pkg/errors"
)
2017-03-15 16:55:43 -05:00
// Canvas is an off-screen rectangular BasicTarget and Picture at the same time, that you can draw
// onto.
//
2017-03-05 17:28:52 -06:00
// It supports TrianglesPosition, TrianglesColor, TrianglesPicture and PictureColor.
type Canvas struct {
2017-03-08 12:19:20 -06:00
// these should **only** be accessed through orig
f *glhf.Frame
borders pixel.Rect
pixels []uint8
dirty bool
2017-03-08 12:19:20 -06:00
// these should **never** be accessed through orig
s *glhf.Shader
bounds pixel.Rect
mat mgl32.Mat3
col mgl32.Vec4
2017-01-21 20:05:46 -06:00
smooth bool
2017-03-08 12:19:20 -06:00
orig *Canvas
}
2017-03-15 16:55:43 -05:00
// NewCanvas creates a new empty, fully transparent Canvas with given bounds. If the smooth flag is
2017-03-05 17:28:52 -06:00
// set, then stretched Pictures will be smoothed and will not be drawn pixely onto this Canvas.
func NewCanvas(bounds pixel.Rect, smooth bool) *Canvas {
c := &Canvas{
smooth: smooth,
mat: mgl32.Ident3(),
col: mgl32.Vec4{1, 1, 1, 1},
}
2017-03-07 10:45:46 -06:00
c.orig = c
2017-03-05 17:28:52 -06:00
mainthread.Call(func() {
var err error
2017-02-11 07:09:47 -06:00
c.s, err = glhf.NewShader(
canvasVertexFormat,
canvasUniformFormat,
canvasVertexShader,
canvasFragmentShader,
)
if err != nil {
2017-03-05 17:28:52 -06:00
panic(errors.Wrap(err, "failed to create Canvas, there's a bug in the shader"))
}
})
2017-03-05 17:28:52 -06:00
c.SetBounds(bounds)
return c
}
2017-03-05 17:28:52 -06:00
// MakeTriangles creates a specialized copy of the supplied Triangles that draws onto this Canvas.
//
// TrianglesPosition, TrianglesColor and TrianglesPicture are supported.
func (c *Canvas) MakeTriangles(t pixel.Triangles) pixel.TargetTriangles {
return &canvasTriangles{
GLTriangles: NewGLTriangles(c.s, t),
dst: c,
2017-01-21 20:05:46 -06:00
}
2017-03-05 17:28:52 -06:00
}
2017-01-21 20:05:46 -06:00
2017-03-05 17:28:52 -06:00
// MakePicture create a specialized copy of the supplied Picture that draws onto this Canvas.
//
// PictureColor is supported.
func (c *Canvas) MakePicture(p pixel.Picture) pixel.TargetPicture {
// short paths
2017-03-06 12:04:57 -06:00
if cp, ok := p.(*canvasPicture); ok {
2017-03-06 17:32:58 -06:00
tp := new(canvasPicture)
*tp = *cp
tp.dst = c
2017-03-06 17:32:58 -06:00
return tp
2017-03-06 12:04:57 -06:00
}
if ccp, ok := p.(*canvasCanvasPicture); ok {
tp := new(canvasCanvasPicture)
*tp = *ccp
tp.dst = c
return tp
}
2017-03-07 15:44:25 -06:00
// Canvas special case
if canvas, ok := p.(*Canvas); ok {
return &canvasCanvasPicture{
src: canvas,
dst: c,
}
}
2017-03-06 12:04:57 -06:00
2017-03-06 06:05:45 -06:00
bounds := p.Bounds()
2017-03-07 10:45:46 -06:00
bx, by, bw, bh := intBounds(bounds)
2017-03-06 06:05:45 -06:00
pixels := make([]uint8, 4*bw*bh)
if pd, ok := p.(*pixel.PictureData); ok {
// PictureData short path
2017-03-20 17:11:50 -05:00
for y := 0; y < bh; y++ {
for x := 0; x < bw; x++ {
nrgba := pd.Pix[y*pd.Stride+x]
off := (y*bw + x) * 4
pixels[off+0] = nrgba.R
pixels[off+1] = nrgba.G
pixels[off+2] = nrgba.B
pixels[off+3] = nrgba.A
}
}
} else if p, ok := p.(pixel.PictureColor); ok {
2017-03-06 06:05:45 -06:00
for y := 0; y < bh; y++ {
for x := 0; x < bw; x++ {
at := pixel.V(
math.Max(float64(bx+x), bounds.Min.X()),
math.Max(float64(by+y), bounds.Min.Y()),
2017-03-06 06:05:45 -06:00
)
color := p.Color(at)
2017-03-20 17:11:50 -05:00
off := (y*bw + x) * 4
pixels[off+0] = uint8(color.R * 255)
pixels[off+1] = uint8(color.G * 255)
pixels[off+2] = uint8(color.B * 255)
pixels[off+3] = uint8(color.A * 255)
2017-03-06 06:05:45 -06:00
}
}
2017-03-05 17:28:52 -06:00
}
2017-01-21 20:05:46 -06:00
2017-03-05 17:28:52 -06:00
var tex *glhf.Texture
mainthread.Call(func() {
2017-03-06 06:05:45 -06:00
tex = glhf.NewTexture(bw, bh, c.smooth, pixels)
2017-01-21 20:05:46 -06:00
})
2017-03-06 12:04:57 -06:00
cp := &canvasPicture{
tex: tex,
pixels: pixels,
bounds: pixel.R(
2017-03-06 17:32:58 -06:00
float64(bx), float64(by),
float64(bw), float64(bh),
),
dst: c,
2017-03-05 17:28:52 -06:00
}
2017-03-06 17:32:58 -06:00
cp.orig = cp
2017-03-06 12:04:57 -06:00
return cp
}
// SetMatrix sets a Matrix that every point will be projected by.
func (c *Canvas) SetMatrix(m pixel.Matrix) {
for i := range m {
c.mat[i] = float32(m[i])
}
}
2017-03-05 17:28:52 -06:00
// SetColorMask sets a color that every color in triangles or a picture will be multiplied by.
func (c *Canvas) SetColorMask(col color.Color) {
nrgba := pixel.NRGBA{R: 1, G: 1, B: 1, A: 1}
if col != nil {
2017-03-23 17:07:49 -05:00
nrgba = pixel.ToNRGBA(col)
2017-03-05 17:28:52 -06:00
}
c.col = mgl32.Vec4{
float32(nrgba.R),
float32(nrgba.G),
float32(nrgba.B),
float32(nrgba.A),
}
}
2017-03-05 17:28:52 -06:00
// SetBounds resizes the Canvas to the new bounds. Old content will be preserved.
2017-03-07 10:45:46 -06:00
//
2017-03-14 13:27:28 -05:00
// If the new Bounds fit into the Original borders, no new Canvas will be allocated.
2017-03-05 17:28:52 -06:00
func (c *Canvas) SetBounds(bounds pixel.Rect) {
2017-03-08 09:56:02 -06:00
c.bounds = bounds
// if this bounds fit into the original bounds, no need to reallocate
2017-03-14 13:27:28 -05:00
if c.orig.borders.Contains(bounds.Min) && c.orig.borders.Contains(bounds.Max) {
2017-03-05 17:28:52 -06:00
return
}
mainthread.Call(func() {
2017-03-08 12:19:20 -06:00
oldF := c.orig.f
2017-03-05 17:28:52 -06:00
2017-03-07 10:45:46 -06:00
_, _, w, h := intBounds(bounds)
2017-03-05 17:28:52 -06:00
c.f = glhf.NewFrame(w, h, c.smooth)
// preserve old content
if oldF != nil {
2017-03-08 12:19:20 -06:00
relBounds := bounds
relBounds = relBounds.Moved(-c.orig.borders.Min)
2017-03-07 10:45:46 -06:00
ox, oy, ow, oh := intBounds(relBounds)
2017-03-05 17:28:52 -06:00
oldF.Blit(
c.f,
ox, oy, ox+ow, oy+oh,
ox, oy, ox+ow, oy+oh,
)
}
})
2017-03-07 10:45:46 -06:00
2017-03-08 12:19:20 -06:00
// detach from orig
2017-03-08 09:56:02 -06:00
c.borders = bounds
2017-03-08 12:19:20 -06:00
c.pixels = nil
2017-03-07 10:45:46 -06:00
c.dirty = true
2017-03-08 12:19:20 -06:00
c.orig = c
}
2017-03-05 17:28:52 -06:00
// Bounds returns the rectangular bounds of the Canvas.
func (c *Canvas) Bounds() pixel.Rect {
return c.bounds
}
2017-03-15 16:55:43 -05:00
// SetSmooth sets whether stretched Pictures drawn onto this Canvas should be drawn smooth or
2017-03-05 17:28:52 -06:00
// pixely.
func (c *Canvas) SetSmooth(smooth bool) {
c.smooth = smooth
}
2017-03-15 16:55:43 -05:00
// Smooth returns whether stretched Pictures drawn onto this Canvas are set to be drawn smooth or
// pixely.
2017-03-05 17:28:52 -06:00
func (c *Canvas) Smooth() bool {
return c.smooth
}
2017-03-07 10:45:46 -06:00
// must be manually called inside mainthread
func (c *Canvas) setGlhfBounds() {
bounds := c.bounds
bounds.Moved(c.orig.borders.Min)
2017-03-07 10:45:46 -06:00
bx, by, bw, bh := intBounds(bounds)
glhf.Bounds(bx, by, bw, bh)
}
2017-03-15 16:55:43 -05:00
// Clear fills the whole Canvas with a single color.
2017-03-05 17:28:52 -06:00
func (c *Canvas) Clear(color color.Color) {
2017-03-07 10:45:46 -06:00
c.orig.dirty = true
2017-03-23 17:07:49 -05:00
nrgba := pixel.ToNRGBA(color)
2017-03-07 10:45:46 -06:00
2017-03-08 08:02:40 -06:00
// color masking
nrgba = nrgba.Mul(pixel.NRGBA{
R: float64(c.col[0]),
G: float64(c.col[1]),
B: float64(c.col[2]),
A: float64(c.col[3]),
})
2017-03-05 17:28:52 -06:00
mainthread.CallNonBlock(func() {
2017-03-07 10:45:46 -06:00
c.setGlhfBounds()
2017-03-08 12:19:20 -06:00
c.orig.f.Begin()
2017-03-05 17:28:52 -06:00
glhf.Clear(
float32(nrgba.R),
float32(nrgba.G),
float32(nrgba.B),
float32(nrgba.A),
)
2017-03-08 12:19:20 -06:00
c.orig.f.End()
2017-03-05 17:28:52 -06:00
})
}
2017-03-07 10:45:46 -06:00
// Color returns the color of the pixel over the given position inside the Canvas.
func (c *Canvas) Color(at pixel.Vec) pixel.NRGBA {
if c.orig.dirty {
mainthread.Call(func() {
2017-03-08 12:19:20 -06:00
tex := c.orig.f.Texture()
tex.Begin()
c.orig.pixels = tex.Pixels(0, 0, tex.Width(), tex.Height())
tex.End()
2017-03-07 10:45:46 -06:00
})
c.orig.dirty = false
}
if !c.bounds.Contains(at) {
return pixel.NRGBA{}
}
2017-03-08 09:56:02 -06:00
bx, by, bw, _ := intBounds(c.orig.borders)
2017-03-07 10:45:46 -06:00
x, y := int(at.X())-bx, int(at.Y())-by
off := y*bw + x
return pixel.NRGBA{
R: float64(c.orig.pixels[off*4+0]) / 255,
G: float64(c.orig.pixels[off*4+1]) / 255,
B: float64(c.orig.pixels[off*4+2]) / 255,
A: float64(c.orig.pixels[off*4+3]) / 255,
}
}
type canvasTriangles struct {
2017-03-05 17:28:52 -06:00
*GLTriangles
dst *Canvas
}
func (ct *canvasTriangles) draw(tex *glhf.Texture, bounds pixel.Rect) {
ct.dst.orig.dirty = true
2017-03-07 10:45:46 -06:00
2017-03-05 17:28:52 -06:00
// save the current state vars to avoid race condition
mat := ct.dst.mat
col := ct.dst.col
mainthread.CallNonBlock(func() {
ct.dst.setGlhfBounds()
2017-03-08 12:19:20 -06:00
ct.dst.orig.f.Begin()
ct.dst.s.Begin()
ct.dst.s.SetUniformAttr(canvasBounds, mgl32.Vec4{
float32(ct.dst.bounds.Min.X()),
float32(ct.dst.bounds.Min.Y()),
float32(ct.dst.bounds.W()),
float32(ct.dst.bounds.H()),
2017-03-06 17:32:58 -06:00
})
ct.dst.s.SetUniformAttr(canvasTransform, mat)
ct.dst.s.SetUniformAttr(canvasColorMask, col)
if tex == nil {
2017-03-05 17:28:52 -06:00
ct.vs.Begin()
ct.vs.Draw()
ct.vs.End()
} else {
tex.Begin()
2017-03-05 17:28:52 -06:00
ct.dst.s.SetUniformAttr(canvasTexBounds, mgl32.Vec4{
float32(bounds.Min.X()),
float32(bounds.Min.Y()),
float32(bounds.W()),
float32(bounds.H()),
2017-03-05 17:28:52 -06:00
})
if tex.Smooth() != ct.dst.smooth {
tex.SetSmooth(ct.dst.smooth)
2017-03-05 17:28:52 -06:00
}
ct.vs.Begin()
ct.vs.Draw()
ct.vs.End()
tex.End()
}
ct.dst.s.End()
2017-03-08 12:19:20 -06:00
ct.dst.orig.f.End()
})
}
2017-03-05 17:28:52 -06:00
func (ct *canvasTriangles) Draw() {
ct.draw(nil, pixel.Rect{})
2017-03-05 17:28:52 -06:00
}
type canvasPicture struct {
tex *glhf.Texture
pixels []uint8
bounds pixel.Rect
2017-03-05 17:28:52 -06:00
2017-03-06 17:32:58 -06:00
orig *canvasPicture
dst *Canvas
2017-03-05 17:28:52 -06:00
}
func (cp *canvasPicture) Bounds() pixel.Rect {
return cp.bounds
}
func (cp *canvasPicture) Color(at pixel.Vec) pixel.NRGBA {
if !cp.bounds.Contains(at) {
return pixel.NRGBA{}
}
bx, by, bw, _ := intBounds(cp.bounds)
x, y := int(at.X())-bx, int(at.Y())-by
off := y*bw + x
return pixel.NRGBA{
R: float64(cp.pixels[off*4+0]) / 255,
G: float64(cp.pixels[off*4+1]) / 255,
B: float64(cp.pixels[off*4+2]) / 255,
A: float64(cp.pixels[off*4+3]) / 255,
}
}
2017-03-05 17:28:52 -06:00
func (cp *canvasPicture) Draw(t pixel.TargetTriangles) {
ct := t.(*canvasTriangles)
if cp.dst != ct.dst {
2017-03-08 16:51:53 -06:00
panic(fmt.Errorf("(%T).Draw: TargetTriangles generated by different Canvas", cp))
2017-03-05 17:28:52 -06:00
}
ct.draw(cp.tex, cp.bounds)
}
type canvasCanvasPicture struct {
src, dst *Canvas
}
func (ccp *canvasCanvasPicture) Bounds() pixel.Rect {
return ccp.src.Bounds()
}
func (ccp *canvasCanvasPicture) Color(at pixel.Vec) pixel.NRGBA {
if !ccp.Bounds().Contains(at) {
return pixel.NRGBA{}
}
return ccp.src.Color(at)
}
func (ccp *canvasCanvasPicture) Draw(t pixel.TargetTriangles) {
ct := t.(*canvasTriangles)
if ccp.dst != ct.dst {
2017-03-08 16:51:53 -06:00
panic(fmt.Errorf("(%T).Draw: TargetTriangles generated by different Canvas", ccp))
}
ct.draw(ccp.src.orig.f.Texture(), ccp.Bounds())
2017-03-05 17:28:52 -06:00
}
const (
2017-03-05 17:28:52 -06:00
canvasPosition int = iota
canvasColor
canvasTexture
canvasIntensity
)
2017-02-11 07:09:47 -06:00
var canvasVertexFormat = glhf.AttrFormat{
2017-03-05 17:28:52 -06:00
canvasPosition: {Name: "position", Type: glhf.Vec2},
canvasColor: {Name: "color", Type: glhf.Vec4},
canvasTexture: {Name: "texture", Type: glhf.Vec2},
canvasIntensity: {Name: "intensity", Type: glhf.Float},
}
const (
2017-03-05 17:28:52 -06:00
canvasTransform int = iota
canvasColorMask
2017-03-06 17:32:58 -06:00
canvasBounds
2017-03-06 09:09:09 -06:00
canvasTexBounds
)
2017-02-11 07:09:47 -06:00
var canvasUniformFormat = glhf.AttrFormat{
canvasTransform: {Name: "transform", Type: glhf.Mat3},
canvasColorMask: {Name: "colorMask", Type: glhf.Vec4},
canvasBounds: {Name: "bounds", Type: glhf.Vec4},
canvasTexBounds: {Name: "texBounds", Type: glhf.Vec4},
}
var canvasVertexShader = `
#version 330 core
in vec2 position;
in vec4 color;
in vec2 texture;
2017-03-05 17:28:52 -06:00
in float intensity;
out vec4 Color;
out vec2 Texture;
2017-03-05 17:28:52 -06:00
out float Intensity;
uniform mat3 transform;
2017-03-07 10:45:46 -06:00
uniform vec4 borders;
2017-03-06 17:32:58 -06:00
uniform vec4 bounds;
void main() {
2017-03-05 17:28:52 -06:00
vec2 transPos = (transform * vec3(position, 1.0)).xy;
2017-03-07 10:45:46 -06:00
vec2 normPos = (transPos - bounds.xy) / (bounds.zw) * 2 - vec2(1, 1);
2017-03-05 17:28:52 -06:00
gl_Position = vec4(normPos, 0.0, 1.0);
Color = color;
Texture = texture;
2017-03-05 17:28:52 -06:00
Intensity = intensity;
}
`
var canvasFragmentShader = `
#version 330 core
in vec4 Color;
in vec2 Texture;
2017-03-05 17:28:52 -06:00
in float Intensity;
out vec4 color;
2017-03-05 17:28:52 -06:00
uniform vec4 colorMask;
2017-03-06 17:32:58 -06:00
uniform vec4 texBorders;
2017-03-06 09:09:09 -06:00
uniform vec4 texBounds;
uniform sampler2D tex;
void main() {
2017-03-05 17:28:52 -06:00
if (Intensity == 0) {
color = colorMask * Color;
} else {
2017-03-05 17:28:52 -06:00
color = vec4(0, 0, 0, 0);
color += (1 - Intensity) * colorMask * Color;
vec2 t = (Texture - texBounds.xy) / texBounds.zw;
color += Intensity * colorMask * Color * texture(tex, t);
}
}
`