Skip to content
canvas.go 12.2 KiB
Newer Older
	"math"
faiface's avatar
faiface committed
	"github.com/faiface/glhf"
	"github.com/faiface/mainthread"
	"github.com/faiface/pixel"
	"github.com/go-gl/mathgl/mgl32"
	"github.com/pkg/errors"
)

faiface's avatar
faiface committed
// Canvas is an off-screen rectangular BasicTarget and Picture at the same time, that you can draw
// onto.
// It supports TrianglesPosition, TrianglesColor, TrianglesPicture and PictureColor.
type Canvas struct {
faiface's avatar
faiface committed
	// these should **only** be accessed through orig
	f       *glhf.Frame
	borders pixel.Rect
	pixels  []uint8
	dirty   bool
faiface's avatar
faiface committed
	// these should **never** be accessed through orig
	s      *glhf.Shader
	bounds pixel.Rect
	mat    mgl32.Mat3
	col    mgl32.Vec4
faiface's avatar
faiface committed
	smooth bool

faiface's avatar
faiface committed
	orig *Canvas
faiface's avatar
faiface committed
// NewCanvas creates a new empty, fully transparent Canvas with given bounds. If the smooth flag is
// 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},
	}
faiface's avatar
faiface committed
	c.orig = c
	mainthread.Call(func() {
		var err error
faiface's avatar
faiface committed
		c.s, err = glhf.NewShader(
			canvasVertexFormat,
			canvasUniformFormat,
			canvasVertexShader,
			canvasFragmentShader,
		)
		if err != nil {
			panic(errors.Wrap(err, "failed to create Canvas, there's a bug in the shader"))
	c.SetBounds(bounds)

// 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),
faiface's avatar
faiface committed
	}
faiface's avatar
faiface committed

// 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 {
faiface's avatar
faiface committed
	if cp, ok := p.(*canvasPicture); ok {
		tp := new(canvasPicture)
		*tp = *cp
		return tp
faiface's avatar
faiface committed
	}
	if ccp, ok := p.(*canvasCanvasPicture); ok {
		tp := new(canvasCanvasPicture)
		*tp = *ccp
		tp.dst = c
		return tp
	}
faiface's avatar
faiface committed

	// Canvas special case
	if canvas, ok := p.(*Canvas); ok {
		return &canvasCanvasPicture{
			src:    canvas,
			dst:    c,
			bounds: c.bounds,
		}
	}
faiface's avatar
faiface committed

	bounds := p.Bounds()
faiface's avatar
faiface committed
	bx, by, bw, bh := intBounds(bounds)

	pixels := make([]uint8, 4*bw*bh)

	if pd, ok := p.(*pixel.PictureData); ok {
		// PictureData short path
		for i, nrgba := range pd.Pix {
			pixels[i*4+0] = nrgba.R
			pixels[i*4+1] = nrgba.G
			pixels[i*4+2] = nrgba.B
			pixels[i*4+3] = nrgba.A
		}
	} else if p, ok := p.(pixel.PictureColor); ok {
		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()),
				)
				color := p.Color(at)
				pixels[(y*bw+x)*4+0] = uint8(color.R * 255)
				pixels[(y*bw+x)*4+1] = uint8(color.G * 255)
				pixels[(y*bw+x)*4+2] = uint8(color.B * 255)
				pixels[(y*bw+x)*4+3] = uint8(color.A * 255)
			}
		}
faiface's avatar
faiface committed

	var tex *glhf.Texture
	mainthread.Call(func() {
		tex = glhf.NewTexture(bw, bh, c.smooth, pixels)
faiface's avatar
faiface committed
	cp := &canvasPicture{
		tex:    tex,
		pixels: pixels,
		borders: pixel.R(
			float64(bx), float64(by),
			float64(bw), float64(bh),
		),
		bounds: bounds,
	cp.orig = cp
faiface's avatar
faiface committed
	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])
	}
// 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 {
		nrgba = pixel.NRGBAModel.Convert(col).(pixel.NRGBA)
	}
	c.col = mgl32.Vec4{
		float32(nrgba.R),
		float32(nrgba.G),
		float32(nrgba.B),
		float32(nrgba.A),
	}
// SetBounds resizes the Canvas to the new bounds. Old content will be preserved.
faiface's avatar
faiface committed
//
// If the new Bounds fit into the Original borders, no new Canvas will be allocated.
func (c *Canvas) SetBounds(bounds pixel.Rect) {
faiface's avatar
faiface committed
	c.bounds = bounds

	// if this bounds fit into the original bounds, no need to reallocate
	if c.orig.borders.Contains(bounds.Min) && c.orig.borders.Contains(bounds.Max) {
		return
	}

	mainthread.Call(func() {
faiface's avatar
faiface committed
		oldF := c.orig.f
faiface's avatar
faiface committed
		_, _, w, h := intBounds(bounds)
		c.f = glhf.NewFrame(w, h, c.smooth)

		// preserve old content
		if oldF != nil {
faiface's avatar
faiface committed
			relBounds := bounds
			relBounds = relBounds.Moved(-c.orig.borders.Min)
faiface's avatar
faiface committed
			ox, oy, ow, oh := intBounds(relBounds)
			oldF.Blit(
				c.f,
				ox, oy, ox+ow, oy+oh,
				ox, oy, ox+ow, oy+oh,
			)
		}
	})
faiface's avatar
faiface committed
	// detach from orig
faiface's avatar
faiface committed
	c.borders = bounds
faiface's avatar
faiface committed
	c.pixels = nil
faiface's avatar
faiface committed
	c.dirty = true
faiface's avatar
faiface committed
	c.orig = c
// Bounds returns the rectangular bounds of the Canvas.
func (c *Canvas) Bounds() pixel.Rect {
	return c.bounds
faiface's avatar
faiface committed
// SetSmooth sets whether stretched Pictures drawn onto this Canvas should be drawn smooth or
// pixely.
func (c *Canvas) SetSmooth(smooth bool) {
	c.smooth = smooth
faiface's avatar
faiface committed
// Smooth returns whether stretched Pictures drawn onto this Canvas are set to be drawn smooth or
// pixely.
func (c *Canvas) Smooth() bool {
	return c.smooth
faiface's avatar
faiface committed
// must be manually called inside mainthread
func (c *Canvas) setGlhfBounds() {
	bounds := c.bounds
	bounds.Moved(c.orig.borders.Min)
faiface's avatar
faiface committed
	bx, by, bw, bh := intBounds(bounds)
	glhf.Bounds(bx, by, bw, bh)
}

faiface's avatar
faiface committed
// Clear fills the whole Canvas with a single color.
func (c *Canvas) Clear(color color.Color) {
faiface's avatar
faiface committed
	c.orig.dirty = true

	nrgba := pixel.NRGBAModel.Convert(color).(pixel.NRGBA)
	// 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]),
	})

	mainthread.CallNonBlock(func() {
faiface's avatar
faiface committed
		c.setGlhfBounds()
faiface's avatar
faiface committed
		c.orig.f.Begin()
		glhf.Clear(
			float32(nrgba.R),
			float32(nrgba.G),
			float32(nrgba.B),
			float32(nrgba.A),
		)
faiface's avatar
faiface committed
		c.orig.f.End()
faiface's avatar
faiface committed
// Slice returns a sub-Canvas with the specified Bounds.
//
faiface's avatar
faiface committed
// The type of the returned value is *Canvas, the type of the return value is a general
// pixel.Picture just so that Canvas implements pixel.Picture interface.
faiface's avatar
faiface committed
func (c *Canvas) Slice(bounds pixel.Rect) pixel.Picture {
	sc := new(Canvas)
	*sc = *c
	sc.bounds = bounds
	return sc
}

// Original returns the most original Canvas that this Canvas was created from using Slice-ing.
//
faiface's avatar
faiface committed
// The type of the returned value is *Canvas, the type of the return value is a general
// pixel.Picture just so that Canvas implements pixel.Picture interface.
faiface's avatar
faiface committed
func (c *Canvas) Original() pixel.Picture {
	return c.orig
}

// 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() {
faiface's avatar
faiface committed
			tex := c.orig.f.Texture()
			tex.Begin()
			c.orig.pixels = tex.Pixels(0, 0, tex.Width(), tex.Height())
			tex.End()
faiface's avatar
faiface committed
		})
		c.orig.dirty = false
	}
	if !c.bounds.Contains(at) {
		return pixel.NRGBA{}
	}
faiface's avatar
faiface committed
	bx, by, bw, _ := intBounds(c.orig.borders)
faiface's avatar
faiface committed
	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 {
	*GLTriangles

	dst *Canvas
func (ct *canvasTriangles) draw(tex *glhf.Texture, borders, bounds pixel.Rect) {
	ct.dst.orig.dirty = true
	// save the current state vars to avoid race condition
	mat := ct.dst.mat
	col := ct.dst.col

	mainthread.CallNonBlock(func() {
		ct.dst.setGlhfBounds()
faiface's avatar
faiface committed
		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()),
		ct.dst.s.SetUniformAttr(canvasTransform, mat)
		ct.dst.s.SetUniformAttr(canvasColorMask, col)
		if tex == nil {
			ct.vs.Begin()
			ct.vs.Draw()
			ct.vs.End()
			ct.dst.s.SetUniformAttr(canvasTexBorders, mgl32.Vec4{
				float32(borders.Min.X()),
				float32(borders.Min.Y()),
				float32(borders.W()),
				float32(borders.H()),
faiface's avatar
faiface committed
			})
			ct.dst.s.SetUniformAttr(canvasTexBounds, mgl32.Vec4{
				float32(bounds.Min.X()),
				float32(bounds.Min.Y()),
				float32(bounds.W()),
				float32(bounds.H()),
			if tex.Smooth() != ct.dst.smooth {
				tex.SetSmooth(ct.dst.smooth)
			}

			ct.vs.Begin()
			ct.vs.Draw()
			ct.vs.End()

		ct.dst.s.End()
faiface's avatar
faiface committed
		ct.dst.orig.f.End()
func (ct *canvasTriangles) Draw() {
	ct.draw(nil, pixel.Rect{}, pixel.Rect{})
}

type canvasPicture struct {
	tex     *glhf.Texture
	borders pixel.Rect
	bounds  pixel.Rect
	orig *canvasPicture
}

func (cp *canvasPicture) Bounds() pixel.Rect {
	return cp.bounds
}

func (cp *canvasPicture) Slice(r pixel.Rect) pixel.Picture {
	sp := new(canvasPicture)
	*sp = *cp
	sp.bounds = r
	return sp
faiface's avatar
faiface committed
func (cp *canvasPicture) Original() pixel.Picture {
	return cp.orig
faiface's avatar
faiface committed
}

func (cp *canvasPicture) Color(at pixel.Vec) pixel.NRGBA {
	if !cp.bounds.Contains(at) {
		return pixel.NRGBA{}
	}
	bx, by, bw, _ := intBounds(cp.borders)
	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,
	}
}

func (cp *canvasPicture) Draw(t pixel.TargetTriangles) {
	ct := t.(*canvasTriangles)
	if cp.dst != ct.dst {
		panic(fmt.Errorf("(%T).Draw: TargetTriangles generated by different Canvas", cp))
	ct.draw(cp.tex, cp.borders, cp.bounds)
}

type canvasCanvasPicture struct {
	src, dst *Canvas
	bounds   pixel.Rect
	orig     *canvasCanvasPicture
}

func (ccp *canvasCanvasPicture) Bounds() pixel.Rect {
	return ccp.bounds
}

func (ccp *canvasCanvasPicture) Slice(r pixel.Rect) pixel.Picture {
	sp := new(canvasCanvasPicture)
	*sp = *ccp
	sp.bounds = r
	return sp
}

func (ccp *canvasCanvasPicture) Original() pixel.Picture {
	return ccp.orig
}

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 {
		panic(fmt.Errorf("(%T).Draw: TargetTriangles generated by different Canvas", ccp))
faiface's avatar
faiface committed
	ct.draw(ccp.src.orig.f.Texture(), ccp.src.orig.borders, ccp.bounds)
	canvasPosition int = iota
	canvasColor
	canvasTexture
	canvasIntensity
faiface's avatar
faiface committed
var canvasVertexFormat = glhf.AttrFormat{
	canvasPosition:  {Name: "position", Type: glhf.Vec2},
	canvasColor:     {Name: "color", Type: glhf.Vec4},
	canvasTexture:   {Name: "texture", Type: glhf.Vec2},
	canvasIntensity: {Name: "intensity", Type: glhf.Float},
	canvasTransform int = iota
	canvasColorMask
	canvasBounds
	canvasTexBorders
faiface's avatar
faiface committed
	canvasTexBounds
faiface's avatar
faiface committed
var canvasUniformFormat = glhf.AttrFormat{
	canvasTransform:  {Name: "transform", Type: glhf.Mat3},
	canvasColorMask:  {Name: "colorMask", Type: glhf.Vec4},
	canvasBounds:     {Name: "bounds", Type: glhf.Vec4},
	canvasTexBorders: {Name: "texBorders", Type: glhf.Vec4},
	canvasTexBounds:  {Name: "texBounds", Type: glhf.Vec4},
}

var canvasVertexShader = `
#version 330 core

in vec2 position;
in vec4 color;
in vec2 texture;
in float intensity;

out vec4 Color;
out vec2 Texture;
out float Intensity;

uniform mat3 transform;
faiface's avatar
faiface committed
uniform vec4 borders;
uniform vec4 bounds;
	vec2 transPos = (transform * vec3(position, 1.0)).xy;
faiface's avatar
faiface committed
	vec2 normPos = (transPos - bounds.xy) / (bounds.zw) * 2 - vec2(1, 1);
	gl_Position = vec4(normPos, 0.0, 1.0);
	Color = color;
	Texture = texture;
	Intensity = intensity;
}
`

var canvasFragmentShader = `
#version 330 core

in vec4 Color;
in vec2 Texture;
in float Intensity;
uniform vec4 colorMask;
uniform vec4 texBorders;
faiface's avatar
faiface committed
uniform vec4 texBounds;
uniform sampler2D tex;

void main() {
	if (Intensity == 0) {
		color = colorMask * Color;
		color = vec4(0, 0, 0, 0);
		color += (1 - Intensity) * colorMask * Color;
faiface's avatar
faiface committed

		float bx = texBounds.x;
		float by = texBounds.y;
		float bw = texBounds.z;
		float bh = texBounds.w;
		if (bx <= Texture.x && Texture.x <= bx + bw && by <= Texture.y && Texture.y <= by + bh) {
			vec2 t = (Texture - texBorders.xy) / texBorders.zw;
faiface's avatar
faiface committed
			color += Intensity * colorMask * Color * texture(tex, t);
		}