mirror of
https://github.com/hajimehoshi/ebiten.git
synced 2024-11-10 04:57:26 +01:00
128 lines
3.2 KiB
Go
128 lines
3.2 KiB
Go
package opengl
|
|
|
|
// #cgo LDFLAGS: -framework OpenGL
|
|
//
|
|
// #include <OpenGL/gl.h>
|
|
import "C"
|
|
import (
|
|
"fmt"
|
|
"github.com/hajimehoshi/go-ebiten/graphics"
|
|
"github.com/hajimehoshi/go-ebiten/graphics/matrix"
|
|
"github.com/hajimehoshi/go-ebiten/graphics/opengl/shader"
|
|
"math"
|
|
)
|
|
|
|
func glMatrix(matrix [4][4]float64) [16]float32 {
|
|
result := [16]float32{}
|
|
for j := 0; j < 4; j++ {
|
|
for i := 0; i < 4; i++ {
|
|
result[i+j*4] = float32(matrix[i][j])
|
|
}
|
|
}
|
|
return result
|
|
}
|
|
|
|
type RenderTarget struct {
|
|
framebuffer C.GLuint
|
|
width int
|
|
height int
|
|
flipY bool
|
|
}
|
|
|
|
func newRTWithCurrentFramebuffer(width, height int) *RenderTarget {
|
|
framebuffer := C.GLint(0)
|
|
C.glGetIntegerv(C.GL_FRAMEBUFFER_BINDING, &framebuffer)
|
|
return &RenderTarget{C.GLuint(framebuffer), width, height, true}
|
|
}
|
|
|
|
func createFramebuffer(nativeTexture C.GLuint) C.GLuint {
|
|
framebuffer := C.GLuint(0)
|
|
C.glGenFramebuffers(1, &framebuffer)
|
|
|
|
origFramebuffer := C.GLint(0)
|
|
C.glGetIntegerv(C.GL_FRAMEBUFFER_BINDING, &origFramebuffer)
|
|
|
|
C.glBindFramebuffer(C.GL_FRAMEBUFFER, framebuffer)
|
|
defer C.glBindFramebuffer(C.GL_FRAMEBUFFER, C.GLuint(origFramebuffer))
|
|
|
|
C.glFramebufferTexture2D(C.GL_FRAMEBUFFER, C.GL_COLOR_ATTACHMENT0,
|
|
C.GL_TEXTURE_2D, nativeTexture, 0)
|
|
if C.glCheckFramebufferStatus(C.GL_FRAMEBUFFER) !=
|
|
C.GL_FRAMEBUFFER_COMPLETE {
|
|
panic("creating framebuffer failed")
|
|
}
|
|
|
|
// Set this framebuffer opaque because alpha values on a target might be
|
|
// confusing.
|
|
C.glClearColor(0, 0, 0, 1)
|
|
C.glClear(C.GL_COLOR_BUFFER_BIT)
|
|
|
|
return framebuffer
|
|
}
|
|
|
|
func (r *RenderTarget) setAsViewport() {
|
|
current := C.GLint(0)
|
|
C.glGetIntegerv(C.GL_FRAMEBUFFER_BINDING, ¤t)
|
|
if C.GLuint(current) == r.framebuffer {
|
|
return
|
|
}
|
|
|
|
C.glFlush()
|
|
|
|
C.glBindFramebuffer(C.GL_FRAMEBUFFER, C.GLuint(r.framebuffer))
|
|
err := C.glCheckFramebufferStatus(C.GL_FRAMEBUFFER)
|
|
if err != C.GL_FRAMEBUFFER_COMPLETE {
|
|
panic(fmt.Sprintf("glBindFramebuffer failed: %d", err))
|
|
}
|
|
|
|
C.glBlendFuncSeparate(C.GL_SRC_ALPHA, C.GL_ONE_MINUS_SRC_ALPHA,
|
|
C.GL_ZERO, C.GL_ONE)
|
|
|
|
width := graphics.AdjustSizeForTexture(r.width)
|
|
height := graphics.AdjustSizeForTexture(r.height)
|
|
C.glViewport(0, 0, C.GLsizei(width), C.GLsizei(height))
|
|
}
|
|
|
|
func (r *RenderTarget) projectionMatrix() [4][4]float64 {
|
|
width := graphics.AdjustSizeForTexture(r.width)
|
|
height := graphics.AdjustSizeForTexture(r.height)
|
|
matrix := graphics.OrthoProjectionMatrix(0, width, 0, height)
|
|
if r.flipY {
|
|
matrix[1][1] *= -1
|
|
matrix[1][3] += float64(r.height) /
|
|
float64(graphics.AdjustSizeForTexture(r.height)) * 2
|
|
}
|
|
return matrix
|
|
}
|
|
|
|
func (r *RenderTarget) dispose() {
|
|
C.glDeleteFramebuffers(1, &r.framebuffer)
|
|
}
|
|
|
|
func (r *RenderTarget) fill(red, green, blue uint8) {
|
|
r.setAsViewport()
|
|
const max = float64(math.MaxUint8)
|
|
C.glClearColor(
|
|
C.GLclampf(float64(red)/max),
|
|
C.GLclampf(float64(green)/max),
|
|
C.GLclampf(float64(blue)/max),
|
|
1)
|
|
C.glClear(C.GL_COLOR_BUFFER_BIT)
|
|
}
|
|
|
|
func (r *RenderTarget) drawTexture(
|
|
texture *Texture,
|
|
parts []graphics.TexturePart,
|
|
geometryMatrix matrix.Geometry,
|
|
colorMatrix matrix.Color) {
|
|
r.setAsViewport()
|
|
projectionMatrix := r.projectionMatrix()
|
|
quads := graphics.TextureQuads(parts, texture.width, texture.height)
|
|
shader.DrawTexture(
|
|
shader.NativeTexture(texture.native),
|
|
glMatrix(projectionMatrix),
|
|
quads,
|
|
geometryMatrix,
|
|
colorMatrix)
|
|
}
|