129 lines
2.8 KiB
Go
129 lines
2.8 KiB
Go
package pixelgl
|
|
|
|
import (
|
|
"runtime"
|
|
|
|
"github.com/faiface/mainthread"
|
|
"github.com/go-gl/gl/v3.3-core/gl"
|
|
"github.com/go-gl/mathgl/mgl32"
|
|
)
|
|
|
|
// Texture is an OpenGL texture.
|
|
type Texture struct {
|
|
tex binder
|
|
width, height int
|
|
}
|
|
|
|
// NewTexture creates a new texture with the specified width and height with some initial
|
|
// pixel values. The pixels must be a sequence of RGBA values.
|
|
func NewTexture(width, height int, smooth bool, pixels []uint8) *Texture {
|
|
tex := &Texture{
|
|
tex: binder{
|
|
restoreLoc: gl.TEXTURE_BINDING_2D,
|
|
bindFunc: func(obj uint32) {
|
|
gl.BindTexture(gl.TEXTURE_2D, obj)
|
|
},
|
|
},
|
|
width: width,
|
|
height: height,
|
|
}
|
|
|
|
gl.GenTextures(1, &tex.tex.obj)
|
|
|
|
tex.Begin()
|
|
defer tex.End()
|
|
|
|
// initial data
|
|
gl.TexImage2D(
|
|
gl.TEXTURE_2D,
|
|
0,
|
|
gl.RGBA,
|
|
int32(width),
|
|
int32(height),
|
|
0,
|
|
gl.RGBA,
|
|
gl.UNSIGNED_BYTE,
|
|
gl.Ptr(pixels),
|
|
)
|
|
|
|
borderColor := mgl32.Vec4{0, 0, 0, 0}
|
|
gl.TexParameterfv(gl.TEXTURE_2D, gl.TEXTURE_BORDER_COLOR, &borderColor[0])
|
|
gl.TexParameteri(gl.TEXTURE_2D, gl.TEXTURE_WRAP_S, gl.CLAMP_TO_BORDER)
|
|
gl.TexParameteri(gl.TEXTURE_2D, gl.TEXTURE_WRAP_T, gl.CLAMP_TO_BORDER)
|
|
|
|
if smooth {
|
|
gl.TexParameteri(gl.TEXTURE_2D, gl.TEXTURE_MIN_FILTER, gl.LINEAR)
|
|
gl.TexParameteri(gl.TEXTURE_2D, gl.TEXTURE_MAG_FILTER, gl.LINEAR)
|
|
} else {
|
|
gl.TexParameteri(gl.TEXTURE_2D, gl.TEXTURE_MIN_FILTER, gl.NEAREST)
|
|
gl.TexParameteri(gl.TEXTURE_2D, gl.TEXTURE_MAG_FILTER, gl.NEAREST)
|
|
}
|
|
|
|
runtime.SetFinalizer(tex, (*Texture).delete)
|
|
|
|
return tex
|
|
}
|
|
|
|
func (t *Texture) delete() {
|
|
mainthread.CallNonBlock(func() {
|
|
gl.DeleteTextures(1, &t.tex.obj)
|
|
})
|
|
}
|
|
|
|
// Width returns the width of a texture in pixels.
|
|
func (t *Texture) Width() int {
|
|
return t.width
|
|
}
|
|
|
|
// Height returns the height of a texture in pixels.
|
|
func (t *Texture) Height() int {
|
|
return t.height
|
|
}
|
|
|
|
// SetPixels sets the content of a sub-region of the Texture. Pixels must be an RGBA byte sequence.
|
|
func (t *Texture) SetPixels(x, y, w, h int, pixels []uint8) {
|
|
if len(pixels) != w*h*4 {
|
|
panic("set pixels: wrong number of pixels")
|
|
}
|
|
gl.TexSubImage2D(
|
|
gl.TEXTURE_2D,
|
|
0,
|
|
int32(x),
|
|
int32(y),
|
|
int32(w),
|
|
int32(h),
|
|
gl.RGBA,
|
|
gl.UNSIGNED_BYTE,
|
|
gl.Ptr(pixels),
|
|
)
|
|
}
|
|
|
|
// Pixels returns the content of a sub-region of the Texture as an RGBA byte sequence.
|
|
func (t *Texture) Pixels(x, y, w, h int) []uint8 {
|
|
pixels := make([]uint8, t.width*t.height*4)
|
|
gl.GetTexImage(
|
|
gl.TEXTURE_2D,
|
|
0,
|
|
gl.RGBA,
|
|
gl.UNSIGNED_BYTE,
|
|
gl.Ptr(pixels),
|
|
)
|
|
subPixels := make([]uint8, w*h*4)
|
|
for i := 0; i < h; i++ {
|
|
row := pixels[(i+y)*t.width*4+x*4 : (i+y)*t.width*4+(x+w)*4]
|
|
subRow := subPixels[i*w*4 : (i+1)*w*4]
|
|
copy(subRow, row)
|
|
}
|
|
return subPixels
|
|
}
|
|
|
|
// Begin binds a texture. This is necessary before using the texture.
|
|
func (t *Texture) Begin() {
|
|
t.tex.bind()
|
|
}
|
|
|
|
// End unbinds a texture and restores the previous one.
|
|
func (t *Texture) End() {
|
|
t.tex.restore()
|
|
}
|