ebiten/internal/opengl/context_js.go

473 lines
13 KiB
Go
Raw Normal View History

2014-12-31 13:55:40 +01:00
// Copyright 2014 Hajime Hoshi
//
// Licensed under the Apache License, Version 2.0 (the "License");
// you may not use this file except in compliance with the License.
// You may obtain a copy of the License at
//
// http://www.apache.org/licenses/LICENSE-2.0
//
// Unless required by applicable law or agreed to in writing, software
// distributed under the License is distributed on an "AS IS" BASIS,
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
// See the License for the specific language governing permissions and
// limitations under the License.
// +build js
package opengl
import (
"errors"
"fmt"
2016-06-14 20:35:35 +02:00
2018-06-24 09:10:24 +02:00
"github.com/gopherjs/gopherwasm/js"
2018-10-28 12:42:57 +01:00
"github.com/hajimehoshi/ebiten/internal/graphics"
2014-12-31 13:55:40 +01:00
)
type (
Texture js.Value
Framebuffer js.Value
2018-10-29 17:52:59 +01:00
shader js.Value
buffer js.Value
uniformLocation js.Value
2014-12-31 13:55:40 +01:00
attribLocation int
programID int
2018-10-29 17:52:59 +01:00
program struct {
value js.Value
id programID
}
)
2015-01-12 15:16:34 +01:00
var InvalidTexture = Texture(js.Null())
2018-10-29 17:52:59 +01:00
func getProgramID(p program) programID {
return p.id
2015-01-12 15:16:34 +01:00
}
var (
blend js.Value
clampToEdge js.Value
colorAttachment0 js.Value
compileStatus js.Value
framebuffer js.Value
framebufferBinding js.Value
framebufferComplete js.Value
linkStatus js.Value
maxTextureSize js.Value
nearest js.Value
noError js.Value
texture2d js.Value
textureMagFilter js.Value
textureMinFilter js.Value
textureWrapS js.Value
textureWrapT js.Value
2018-10-30 14:29:54 +01:00
triangles js.Value
rgba js.Value
unpackAlignment js.Value
unsignedByte js.Value
unsignedShort js.Value
)
2016-07-03 11:11:37 +02:00
func init() {
// Accessing the prototype is rquired on Safari.
c := js.Global().Get("WebGLRenderingContext").Get("prototype")
2018-10-30 14:41:05 +01:00
vertexShader = shaderType(c.Get("VERTEX_SHADER").Int())
fragmentShader = shaderType(c.Get("FRAGMENT_SHADER").Int())
arrayBuffer = bufferType(c.Get("ARRAY_BUFFER").Int())
elementArrayBuffer = bufferType(c.Get("ELEMENT_ARRAY_BUFFER").Int())
dynamicDraw = bufferUsage(c.Get("DYNAMIC_DRAW").Int())
2016-10-22 07:51:23 +02:00
Short = DataType(c.Get("SHORT").Int())
Float = DataType(c.Get("FLOAT").Int())
2016-07-03 11:11:37 +02:00
zero = operation(c.Get("ZERO").Int())
one = operation(c.Get("ONE").Int())
srcAlpha = operation(c.Get("SRC_ALPHA").Int())
dstAlpha = operation(c.Get("DST_ALPHA").Int())
oneMinusSrcAlpha = operation(c.Get("ONE_MINUS_SRC_ALPHA").Int())
oneMinusDstAlpha = operation(c.Get("ONE_MINUS_DST_ALPHA").Int())
blend = c.Get("BLEND")
clampToEdge = c.Get("CLAMP_TO_EDGE")
compileStatus = c.Get("COMPILE_STATUS")
colorAttachment0 = c.Get("COLOR_ATTACHMENT0")
framebuffer = c.Get("FRAMEBUFFER")
framebufferBinding = c.Get("FRAMEBUFFER_BINDING")
framebufferComplete = c.Get("FRAMEBUFFER_COMPLETE")
linkStatus = c.Get("LINK_STATUS")
maxTextureSize = c.Get("MAX_TEXTURE_SIZE")
nearest = c.Get("NEAREST")
noError = c.Get("NO_ERROR")
rgba = c.Get("RGBA")
texture2d = c.Get("TEXTURE_2D")
textureMagFilter = c.Get("TEXTURE_MAG_FILTER")
textureMinFilter = c.Get("TEXTURE_MIN_FILTER")
textureWrapS = c.Get("TEXTURE_WRAP_S")
textureWrapT = c.Get("TEXTURE_WRAP_T")
2018-10-30 14:29:54 +01:00
triangles = c.Get("TRIANGLES")
unpackAlignment = c.Get("UNPACK_ALIGNMENT")
unsignedByte = c.Get("UNSIGNED_BYTE")
unsignedShort = c.Get("UNSIGNED_SHORT")
initializeArrayBuferLayout()
2016-07-03 11:11:37 +02:00
}
2014-12-31 13:55:40 +01:00
type context struct {
2018-05-23 20:04:56 +02:00
gl js.Value
loseContext js.Value
2016-05-31 19:33:31 +02:00
lastProgramID programID
2014-12-31 13:55:40 +01:00
}
func Init() error {
if js.Global().Get("WebGLRenderingContext") == js.Undefined() {
return fmt.Errorf("opengl: WebGL is not supported")
}
2017-12-02 08:46:55 +01:00
// TODO: Define id?
canvas := js.Global().Get("document").Call("querySelector", "canvas")
attr := js.Global().Get("Object").New()
2018-05-23 20:04:56 +02:00
attr.Set("alpha", true)
attr.Set("premultipliedAlpha", true)
gl := canvas.Call("getContext", "webgl", attr)
if gl == js.Null() {
gl = canvas.Call("getContext", "experimental-webgl", attr)
if gl == js.Null() {
return fmt.Errorf("opengl: getContext failed")
}
2015-01-27 14:02:23 +01:00
}
2016-07-03 17:55:04 +02:00
c := &Context{}
2015-01-02 07:20:05 +01:00
c.gl = gl
2017-12-02 08:46:55 +01:00
// Getting an extension might fail after the context is lost, so
// it is required to get the extension here.
c.loseContext = gl.Call("getExtension", "WEBGL_lose_context")
2018-06-30 05:34:40 +02:00
theContext = c
return nil
2014-12-31 13:55:40 +01:00
}
2018-10-29 18:18:10 +01:00
func (c *Context) reset() error {
2016-06-09 18:19:10 +02:00
c.locationCache = newLocationCache()
c.lastTexture = Texture(js.Null())
c.lastFramebuffer = Framebuffer(js.Null())
2016-06-09 18:19:10 +02:00
c.lastViewportWidth = 0
c.lastViewportHeight = 0
2018-10-28 12:42:57 +01:00
c.lastCompositeMode = graphics.CompositeModeUnknown
2016-06-09 18:19:10 +02:00
gl := c.gl
gl.Call("enable", blend)
2018-10-28 12:42:57 +01:00
c.BlendFunc(graphics.CompositeModeSourceOver)
f := gl.Call("getParameter", framebufferBinding)
c.screenFramebuffer = Framebuffer(f)
return nil
2016-06-09 18:19:10 +02:00
}
2018-10-28 12:42:57 +01:00
func (c *Context) BlendFunc(mode graphics.CompositeMode) {
if c.lastCompositeMode == mode {
2016-02-28 17:44:09 +01:00
return
}
c.lastCompositeMode = mode
2018-10-28 12:42:57 +01:00
s, d := mode.Operations()
s2, d2 := convertOperation(s), convertOperation(d)
gl := c.gl
2018-10-28 12:42:57 +01:00
gl.Call("blendFunc", int(s2), int(d2))
2014-12-31 13:55:40 +01:00
}
func (c *Context) NewTexture(width, height int) (Texture, error) {
2014-12-31 13:55:40 +01:00
gl := c.gl
t := gl.Call("createTexture")
if t == js.Null() {
return Texture(js.Null()), errors.New("opengl: glGenTexture failed")
2014-12-31 13:55:40 +01:00
}
gl.Call("pixelStorei", unpackAlignment, 4)
2018-11-01 19:43:42 +01:00
c.bindTexture(Texture(t))
2014-12-31 13:55:40 +01:00
gl.Call("texParameteri", texture2d, textureMagFilter, nearest)
gl.Call("texParameteri", texture2d, textureMinFilter, nearest)
gl.Call("texParameteri", texture2d, textureWrapS, clampToEdge)
gl.Call("texParameteri", texture2d, textureWrapT, clampToEdge)
2014-12-31 13:55:40 +01:00
2018-09-28 19:20:02 +02:00
// Firefox warns the usage of textures without specifying pixels (#629)
//
// Error: WebGL warning: drawElements: This operation requires zeroing texture data. This is slow.
//
// In Ebiten, textures are filled with pixels laster by the filter that ignores destination, so it is fine
// to leave textures as uninitialized here. Rather, extra memory allocating for initialization should be
// avoided.
gl.Call("texImage2D", texture2d, 0, rgba, width, height, 0, rgba, unsignedByte, nil)
2014-12-31 13:55:40 +01:00
return Texture(t), nil
2014-12-31 13:55:40 +01:00
}
func (c *Context) bindFramebufferImpl(f Framebuffer) {
2015-02-19 18:02:23 +01:00
gl := c.gl
gl.Call("bindFramebuffer", framebuffer, js.Value(f))
2015-02-19 18:02:23 +01:00
}
func (c *Context) FramebufferPixels(f *FramebufferStruct, width, height int) ([]byte, error) {
2014-12-31 13:55:40 +01:00
gl := c.gl
c.bindFramebuffer(f.native)
2018-05-23 20:04:56 +02:00
pixels := make([]byte, 4*width*height)
p := js.TypedArrayOf(pixels)
gl.Call("readPixels", 0, 0, width, height, rgba, unsignedByte, p)
p.Release()
if e := gl.Call("getError"); e.Int() != noError.Int() {
2016-02-23 16:31:25 +01:00
return nil, errors.New(fmt.Sprintf("opengl: error: %d", e))
2014-12-31 13:55:40 +01:00
}
2018-05-23 20:04:56 +02:00
return pixels, nil
2014-12-31 13:55:40 +01:00
}
func (c *Context) bindTextureImpl(t Texture) {
2014-12-31 13:55:40 +01:00
gl := c.gl
gl.Call("bindTexture", texture2d, js.Value(t))
2014-12-31 13:55:40 +01:00
}
func (c *Context) DeleteTexture(t Texture) {
gl := c.gl
if !gl.Call("isTexture", js.Value(t)).Bool() {
return
}
if c.lastTexture == t {
c.lastTexture = Texture(js.Null())
}
gl.Call("deleteTexture", js.Value(t))
2014-12-31 13:55:40 +01:00
}
2016-06-12 16:54:36 +02:00
func (c *Context) IsTexture(t Texture) bool {
gl := c.gl
return gl.Call("isTexture", js.Value(t)).Bool()
2016-06-12 16:54:36 +02:00
}
2018-11-01 19:43:42 +01:00
func (c *Context) TexSubImage2D(t Texture, pixels []byte, x, y, width, height int) {
c.bindTexture(t)
2015-01-02 15:30:22 +01:00
gl := c.gl
// void texSubImage2D(GLenum target, GLint level, GLint xoffset, GLint yoffset,
// GLsizei width, GLsizei height,
// GLenum format, GLenum type, ArrayBufferView? pixels);
p := js.TypedArrayOf(pixels)
gl.Call("texSubImage2D", texture2d, 0, x, y, width, height, rgba, unsignedByte, p)
p.Release()
2015-01-02 15:30:22 +01:00
}
func (c *Context) newFramebuffer(t Texture) (Framebuffer, error) {
2014-12-31 13:55:40 +01:00
gl := c.gl
f := gl.Call("createFramebuffer")
c.bindFramebuffer(Framebuffer(f))
2014-12-31 13:55:40 +01:00
gl.Call("framebufferTexture2D", framebuffer, colorAttachment0, texture2d, js.Value(t), 0)
if s := gl.Call("checkFramebufferStatus", framebuffer); s.Int() != framebufferComplete.Int() {
return Framebuffer(js.Null()), errors.New(fmt.Sprintf("opengl: creating framebuffer failed: %d", s.Int()))
2014-12-31 13:55:40 +01:00
}
return Framebuffer(f), nil
2014-12-31 13:55:40 +01:00
}
func (c *Context) setViewportImpl(width, height int) {
gl := c.gl
gl.Call("viewport", 0, 0, width, height)
2014-12-31 13:55:40 +01:00
}
func (c *Context) deleteFramebuffer(f Framebuffer) {
2014-12-31 13:55:40 +01:00
gl := c.gl
if !gl.Call("isFramebuffer", js.Value(f)).Bool() {
return
}
2016-07-06 19:08:28 +02:00
// If a framebuffer to be deleted is bound, a newly bound framebuffer
// will be a default framebuffer.
// https://www.khronos.org/opengles/sdk/docs/man/xhtml/glDeleteFramebuffers.xml
if c.lastFramebuffer == f {
c.lastFramebuffer = Framebuffer(js.Null())
c.lastViewportWidth = 0
c.lastViewportHeight = 0
}
gl.Call("deleteFramebuffer", js.Value(f))
2014-12-31 13:55:40 +01:00
}
2018-10-29 17:52:59 +01:00
func (c *Context) newShader(shaderType shaderType, source string) (shader, error) {
2014-12-31 13:55:40 +01:00
gl := c.gl
s := gl.Call("createShader", int(shaderType))
if s == js.Null() {
2018-10-29 17:52:59 +01:00
return shader(js.Null()), fmt.Errorf("opengl: glCreateShader failed: shader type: %d", shaderType)
2014-12-31 13:55:40 +01:00
}
gl.Call("shaderSource", js.Value(s), source)
gl.Call("compileShader", js.Value(s))
2014-12-31 13:55:40 +01:00
if !gl.Call("getShaderParameter", js.Value(s), compileStatus).Bool() {
log := gl.Call("getShaderInfoLog", js.Value(s))
2018-10-29 17:52:59 +01:00
return shader(js.Null()), fmt.Errorf("opengl: shader compile failed: %s", log)
2014-12-31 13:55:40 +01:00
}
2018-10-29 17:52:59 +01:00
return shader(s), nil
2014-12-31 13:55:40 +01:00
}
2018-10-29 17:52:59 +01:00
func (c *Context) deleteShader(s shader) {
2014-12-31 13:55:40 +01:00
gl := c.gl
gl.Call("deleteShader", js.Value(s))
2014-12-31 13:55:40 +01:00
}
2018-10-29 17:52:59 +01:00
func (c *Context) newProgram(shaders []shader) (program, error) {
2014-12-31 13:55:40 +01:00
gl := c.gl
v := gl.Call("createProgram")
if v == js.Null() {
2018-10-29 17:52:59 +01:00
return program{}, errors.New("opengl: glCreateProgram failed")
2014-12-31 13:55:40 +01:00
}
for _, shader := range shaders {
gl.Call("attachShader", v, js.Value(shader))
2014-12-31 13:55:40 +01:00
}
gl.Call("linkProgram", v)
if !gl.Call("getProgramParameter", v, linkStatus).Bool() {
2018-10-29 17:52:59 +01:00
return program{}, errors.New("opengl: program error")
2014-12-31 13:55:40 +01:00
}
id := c.lastProgramID
c.lastProgramID++
2018-10-29 17:52:59 +01:00
return program{
value: v,
id: id,
}, nil
2014-12-31 13:55:40 +01:00
}
2018-10-29 17:52:59 +01:00
func (c *Context) useProgram(p program) {
2014-12-31 13:55:40 +01:00
gl := c.gl
gl.Call("useProgram", p.value)
2014-12-31 13:55:40 +01:00
}
2018-10-29 17:52:59 +01:00
func (c *Context) deleteProgram(p program) {
gl := c.gl
if !gl.Call("isProgram", p.value).Bool() {
return
}
gl.Call("deleteProgram", p.value)
}
2018-10-29 17:52:59 +01:00
func (c *Context) getUniformLocationImpl(p program, location string) uniformLocation {
2015-01-12 15:16:34 +01:00
gl := c.gl
return uniformLocation(gl.Call("getUniformLocation", p.value, location))
2015-01-12 15:16:34 +01:00
}
2018-10-29 17:52:59 +01:00
func (c *Context) uniformInt(p program, location string, v int) {
2014-12-31 13:55:40 +01:00
gl := c.gl
2016-02-26 19:01:55 +01:00
l := c.locationCache.GetUniformLocation(c, p, location)
gl.Call("uniform1i", js.Value(l), v)
2015-01-03 07:52:02 +01:00
}
2018-10-29 17:52:59 +01:00
func (c *Context) uniformFloat(p program, location string, v float32) {
gl := c.gl
l := c.locationCache.GetUniformLocation(c, p, location)
gl.Call("uniform1f", js.Value(l), v)
}
var (
float32Array = js.Global().Get("Float32Array")
)
2018-10-29 17:52:59 +01:00
func (c *Context) uniformFloats(p program, location string, v []float32) {
2015-01-03 07:52:02 +01:00
gl := c.gl
2016-02-26 19:01:55 +01:00
l := c.locationCache.GetUniformLocation(c, p, location)
2015-01-03 07:52:02 +01:00
switch len(v) {
case 2:
gl.Call("uniform2f", js.Value(l), v[0], v[1])
2015-01-03 07:52:02 +01:00
case 4:
gl.Call("uniform4f", js.Value(l), v[0], v[1], v[2], v[3])
2015-01-03 07:52:02 +01:00
case 16:
arr := js.TypedArrayOf(v)
gl.Call("uniformMatrix4fv", js.Value(l), false, arr)
arr.Release()
default:
panic("not reached")
2014-12-31 13:55:40 +01:00
}
}
2018-10-29 17:52:59 +01:00
func (c *Context) getAttribLocationImpl(p program, location string) attribLocation {
2015-01-12 15:16:34 +01:00
gl := c.gl
return attribLocation(gl.Call("getAttribLocation", p.value, location).Int())
2015-01-12 15:16:34 +01:00
}
2018-10-29 17:52:59 +01:00
func (c *Context) vertexAttribPointer(p program, location string, size int, dataType DataType, stride int, offset int) {
2014-12-31 13:55:40 +01:00
gl := c.gl
2016-02-26 19:01:55 +01:00
l := c.locationCache.GetAttribLocation(c, p, location)
gl.Call("vertexAttribPointer", int(l), size, int(dataType), false, stride, offset)
2014-12-31 13:55:40 +01:00
}
2018-10-29 17:52:59 +01:00
func (c *Context) enableVertexAttribArray(p program, location string) {
2014-12-31 13:55:40 +01:00
gl := c.gl
2016-02-26 19:01:55 +01:00
l := c.locationCache.GetAttribLocation(c, p, location)
gl.Call("enableVertexAttribArray", int(l))
2014-12-31 13:55:40 +01:00
}
2018-10-29 17:52:59 +01:00
func (c *Context) disableVertexAttribArray(p program, location string) {
2014-12-31 13:55:40 +01:00
gl := c.gl
2016-02-26 19:01:55 +01:00
l := c.locationCache.GetAttribLocation(c, p, location)
gl.Call("disableVertexAttribArray", int(l))
2014-12-31 13:55:40 +01:00
}
2018-10-29 17:52:59 +01:00
func (c *Context) newArrayBuffer(size int) buffer {
2014-12-31 13:55:40 +01:00
gl := c.gl
b := gl.Call("createBuffer")
2018-10-30 14:41:05 +01:00
gl.Call("bindBuffer", int(arrayBuffer), js.Value(b))
gl.Call("bufferData", int(arrayBuffer), size, int(dynamicDraw))
2018-10-29 17:52:59 +01:00
return buffer(b)
}
2018-10-29 17:52:59 +01:00
func (c *Context) newElementArrayBuffer(size int) buffer {
gl := c.gl
b := gl.Call("createBuffer")
2018-10-30 14:41:05 +01:00
gl.Call("bindBuffer", int(elementArrayBuffer), js.Value(b))
gl.Call("bufferData", int(elementArrayBuffer), size, int(dynamicDraw))
2018-10-29 17:52:59 +01:00
return buffer(b)
2015-01-17 04:45:19 +01:00
}
2018-10-30 15:57:40 +01:00
func (c *Context) bindBuffer(bufferType bufferType, b buffer) {
2015-01-17 04:45:19 +01:00
gl := c.gl
gl.Call("bindBuffer", int(bufferType), js.Value(b))
2014-12-31 13:55:40 +01:00
}
func (c *Context) arrayBufferSubData(data []float32) {
2014-12-31 13:55:40 +01:00
gl := c.gl
arr := js.TypedArrayOf(data)
2018-10-30 14:41:05 +01:00
gl.Call("bufferSubData", int(arrayBuffer), 0, arr)
arr.Release()
}
func (c *Context) elementArrayBufferSubData(data []uint16) {
gl := c.gl
arr := js.TypedArrayOf(data)
2018-10-30 14:41:05 +01:00
gl.Call("bufferSubData", int(elementArrayBuffer), 0, arr)
arr.Release()
2014-12-31 13:55:40 +01:00
}
2018-10-29 17:52:59 +01:00
func (c *Context) deleteBuffer(b buffer) {
gl := c.gl
gl.Call("deleteBuffer", js.Value(b))
}
2018-10-30 14:29:54 +01:00
func (c *Context) DrawElements(len int, offsetInBytes int) {
2014-12-31 13:55:40 +01:00
gl := c.gl
2018-10-30 14:29:54 +01:00
gl.Call("drawElements", triangles, len, unsignedShort, offsetInBytes)
2014-12-31 13:55:40 +01:00
}
func (c *Context) maxTextureSizeImpl() int {
gl := c.gl
return gl.Call("getParameter", maxTextureSize).Int()
}
func (c *Context) Flush() {
gl := c.gl
gl.Call("flush")
}
2017-01-19 18:20:41 +01:00
func (c *Context) IsContextLost() bool {
2017-01-19 18:20:41 +01:00
gl := c.gl
return gl.Call("isContextLost").Bool()
2017-01-19 18:20:41 +01:00
}
func (c *Context) RestoreContext() {
if c.loseContext != js.Null() {
c.loseContext.Call("restoreContext")
}
}