UNPKG

@esotericsoftware/spine-webgl

Version:
165 lines 25.1 kB
/****************************************************************************** * Spine Runtimes License Agreement * Last updated April 5, 2025. Replaces all prior versions. * * Copyright (c) 2013-2025, Esoteric Software LLC * * Integration of the Spine Runtimes into software or otherwise creating * derivative works of the Spine Runtimes is permitted under the terms and * conditions of Section 2 of the Spine Editor License Agreement: * http://esotericsoftware.com/spine-editor-license * * Otherwise, it is permitted to integrate the Spine Runtimes into software * or otherwise create derivative works of the Spine Runtimes (collectively, * "Products"), provided that each user of the Products must obtain their own * Spine Editor license and redistribution of the Products in any form must * include this license and copyright notice. * * THE SPINE RUNTIMES ARE PROVIDED BY ESOTERIC SOFTWARE LLC "AS IS" AND ANY * EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED * WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE * DISCLAIMED. IN NO EVENT SHALL ESOTERIC SOFTWARE LLC BE LIABLE FOR ANY * DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES * (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES, * BUSINESS INTERRUPTION, OR LOSS OF USE, DATA, OR PROFITS) HOWEVER CAUSED AND * ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF * THE SPINE RUNTIMES, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. *****************************************************************************/ import { Mesh, Position2Attribute, ColorAttribute, TexCoordAttribute, Color2Attribute } from "./Mesh.js"; import { ManagedWebGLRenderingContext } from "./WebGL.js"; const GL_ONE = 1; const GL_ONE_MINUS_SRC_COLOR = 0x0301; const GL_SRC_ALPHA = 0x0302; const GL_ONE_MINUS_SRC_ALPHA = 0x0303; const GL_ONE_MINUS_DST_ALPHA = 0x0305; const GL_DST_COLOR = 0x0306; export class PolygonBatcher { static disableCulling = false; context; drawCalls = 0; static globalDrawCalls = 0; isDrawing = false; mesh; shader = null; lastTexture = null; verticesLength = 0; indicesLength = 0; srcColorBlend; srcAlphaBlend; dstBlend; cullWasEnabled = false; constructor(context, twoColorTint = true, maxVertices = 10920) { if (maxVertices > 10920) throw new Error("Can't have more than 10920 triangles per batch: " + maxVertices); this.context = context instanceof ManagedWebGLRenderingContext ? context : new ManagedWebGLRenderingContext(context); let attributes = twoColorTint ? [new Position2Attribute(), new ColorAttribute(), new TexCoordAttribute(), new Color2Attribute()] : [new Position2Attribute(), new ColorAttribute(), new TexCoordAttribute()]; this.mesh = new Mesh(context, attributes, maxVertices, maxVertices * 3); let gl = this.context.gl; this.srcColorBlend = gl.SRC_ALPHA; this.srcAlphaBlend = gl.ONE; this.dstBlend = gl.ONE_MINUS_SRC_ALPHA; } begin(shader) { if (this.isDrawing) throw new Error("PolygonBatch is already drawing. Call PolygonBatch.end() before calling PolygonBatch.begin()"); this.drawCalls = 0; this.shader = shader; this.lastTexture = null; this.isDrawing = true; let gl = this.context.gl; gl.enable(gl.BLEND); gl.blendFuncSeparate(this.srcColorBlend, this.dstBlend, this.srcAlphaBlend, this.dstBlend); if (PolygonBatcher.disableCulling) { this.cullWasEnabled = gl.isEnabled(gl.CULL_FACE); if (this.cullWasEnabled) gl.disable(gl.CULL_FACE); } } static blendModesGL = [ { srcRgb: GL_SRC_ALPHA, srcRgbPma: GL_ONE, dstRgb: GL_ONE_MINUS_SRC_ALPHA, srcAlpha: GL_ONE }, { srcRgb: GL_SRC_ALPHA, srcRgbPma: GL_ONE, dstRgb: GL_ONE, srcAlpha: GL_ONE }, { srcRgb: GL_DST_COLOR, srcRgbPma: GL_DST_COLOR, dstRgb: GL_ONE_MINUS_SRC_ALPHA, srcAlpha: GL_ONE }, { srcRgb: GL_ONE, srcRgbPma: GL_ONE, dstRgb: GL_ONE_MINUS_SRC_COLOR, srcAlpha: GL_ONE } ]; setBlendMode(blendMode, premultipliedAlpha) { const blendModeGL = PolygonBatcher.blendModesGL[blendMode]; const srcColorBlend = premultipliedAlpha ? blendModeGL.srcRgbPma : blendModeGL.srcRgb; const srcAlphaBlend = blendModeGL.srcAlpha; const dstBlend = blendModeGL.dstRgb; if (this.srcColorBlend == srcColorBlend && this.srcAlphaBlend == srcAlphaBlend && this.dstBlend == dstBlend) return; this.srcColorBlend = srcColorBlend; this.srcAlphaBlend = srcAlphaBlend; this.dstBlend = dstBlend; if (this.isDrawing) { this.flush(); } let gl = this.context.gl; gl.blendFuncSeparate(srcColorBlend, dstBlend, srcAlphaBlend, dstBlend); } draw(texture, vertices, indices) { if (texture != this.lastTexture) { this.flush(); this.lastTexture = texture; } else if (this.verticesLength + vertices.length > this.mesh.getVertices().length || this.indicesLength + indices.length > this.mesh.getIndices().length) { this.flush(); } let indexStart = this.mesh.numVertices(); this.mesh.getVertices().set(vertices, this.verticesLength); this.verticesLength += vertices.length; this.mesh.setVerticesLength(this.verticesLength); let indicesArray = this.mesh.getIndices(); for (let i = this.indicesLength, j = 0; j < indices.length; i++, j++) indicesArray[i] = indices[j] + indexStart; this.indicesLength += indices.length; this.mesh.setIndicesLength(this.indicesLength); } flush() { if (this.verticesLength == 0) return; if (!this.lastTexture) throw new Error("No texture set."); if (!this.shader) throw new Error("No shader set."); this.lastTexture.bind(); this.mesh.draw(this.shader, this.context.gl.TRIANGLES); this.verticesLength = 0; this.indicesLength = 0; this.mesh.setVerticesLength(0); this.mesh.setIndicesLength(0); this.drawCalls++; PolygonBatcher.globalDrawCalls++; } end() { if (!this.isDrawing) throw new Error("PolygonBatch is not drawing. Call PolygonBatch.begin() before calling PolygonBatch.end()"); if (this.verticesLength > 0 || this.indicesLength > 0) this.flush(); this.shader = null; this.lastTexture = null; this.isDrawing = false; let gl = this.context.gl; gl.disable(gl.BLEND); if (PolygonBatcher.disableCulling) { if (this.cullWasEnabled) gl.enable(gl.CULL_FACE); } } getDrawCalls() { return this.drawCalls; } static getAndResetGlobalDrawCalls() { let result = PolygonBatcher.globalDrawCalls; PolygonBatcher.globalDrawCalls = 0; return result; } dispose() { this.mesh.dispose(); } } //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"PolygonBatcher.js","sourceRoot":"","sources":["../src/PolygonBatcher.ts"],"names":[],"mappings":"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","sourcesContent":["/******************************************************************************\n * Spine Runtimes License Agreement\n * Last updated April 5, 2025. Replaces all prior versions.\n *\n * Copyright (c) 2013-2025, Esoteric Software LLC\n *\n * Integration of the Spine Runtimes into software or otherwise creating\n * derivative works of the Spine Runtimes is permitted under the terms and\n * conditions of Section 2 of the Spine Editor License Agreement:\n * http://esotericsoftware.com/spine-editor-license\n *\n * Otherwise, it is permitted to integrate the Spine Runtimes into software\n * or otherwise create derivative works of the Spine Runtimes (collectively,\n * \"Products\"), provided that each user of the Products must obtain their own\n * Spine Editor license and redistribution of the Products in any form must\n * include this license and copyright notice.\n *\n * THE SPINE RUNTIMES ARE PROVIDED BY ESOTERIC SOFTWARE LLC \"AS IS\" AND ANY\n * EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED\n * WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE\n * DISCLAIMED. IN NO EVENT SHALL ESOTERIC SOFTWARE LLC BE LIABLE FOR ANY\n * DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES\n * (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES,\n * BUSINESS INTERRUPTION, OR LOSS OF USE, DATA, OR PROFITS) HOWEVER CAUSED AND\n * ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT\n * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF\n * THE SPINE RUNTIMES, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.\n *****************************************************************************/\n\nimport { BlendMode, Disposable } from \"@esotericsoftware/spine-core\";\nimport { GLTexture } from \"./GLTexture.js\";\nimport { Mesh, Position2Attribute, ColorAttribute, TexCoordAttribute, Color2Attribute } from \"./Mesh.js\";\nimport { Shader } from \"./Shader.js\";\nimport { ManagedWebGLRenderingContext } from \"./WebGL.js\";\n\nconst GL_ONE = 1;\nconst GL_ONE_MINUS_SRC_COLOR = 0x0301;\nconst GL_SRC_ALPHA = 0x0302;\nconst GL_ONE_MINUS_SRC_ALPHA = 0x0303;\nconst GL_ONE_MINUS_DST_ALPHA = 0x0305;\nconst GL_DST_COLOR = 0x0306;\n\nexport class PolygonBatcher implements Disposable {\n\tpublic static disableCulling = false;\n\n\tprivate context: ManagedWebGLRenderingContext;\n\tprivate drawCalls = 0;\n\tprivate static globalDrawCalls = 0;\n\tisDrawing = false;\n\tprivate mesh: Mesh;\n\tprivate shader: Shader | null = null;\n\tprivate lastTexture: GLTexture | null = null;\n\tprivate verticesLength = 0;\n\tprivate indicesLength = 0;\n\tprivate srcColorBlend: number;\n\tprivate srcAlphaBlend: number;\n\tprivate dstBlend: number;\n\tprivate cullWasEnabled = false;\n\n\tconstructor (context: ManagedWebGLRenderingContext | WebGLRenderingContext, twoColorTint: boolean = true, maxVertices: number = 10920) {\n\t\tif (maxVertices > 10920) throw new Error(\"Can't have more than 10920 triangles per batch: \" + maxVertices);\n\t\tthis.context = context instanceof ManagedWebGLRenderingContext ? context : new ManagedWebGLRenderingContext(context);\n\t\tlet attributes = twoColorTint ?\n\t\t\t[new Position2Attribute(), new ColorAttribute(), new TexCoordAttribute(), new Color2Attribute()] :\n\t\t\t[new Position2Attribute(), new ColorAttribute(), new TexCoordAttribute()];\n\t\tthis.mesh = new Mesh(context, attributes, maxVertices, maxVertices * 3);\n\t\tlet gl = this.context.gl;\n\t\tthis.srcColorBlend = gl.SRC_ALPHA;\n\t\tthis.srcAlphaBlend = gl.ONE;\n\t\tthis.dstBlend = gl.ONE_MINUS_SRC_ALPHA;\n\t}\n\n\tbegin (shader: Shader) {\n\t\tif (this.isDrawing) throw new Error(\"PolygonBatch is already drawing. Call PolygonBatch.end() before calling PolygonBatch.begin()\");\n\t\tthis.drawCalls = 0;\n\t\tthis.shader = shader;\n\t\tthis.lastTexture = null;\n\t\tthis.isDrawing = true;\n\n\t\tlet gl = this.context.gl;\n\t\tgl.enable(gl.BLEND);\n\t\tgl.blendFuncSeparate(this.srcColorBlend, this.dstBlend, this.srcAlphaBlend, this.dstBlend);\n\n\t\tif (PolygonBatcher.disableCulling) {\n\t\t\tthis.cullWasEnabled = gl.isEnabled(gl.CULL_FACE);\n\t\t\tif (this.cullWasEnabled) gl.disable(gl.CULL_FACE);\n\t\t}\n\t}\n\n\tprivate static blendModesGL: { srcRgb: number, srcRgbPma: number, dstRgb: number, srcAlpha: number }[] = [\n\t\t{ srcRgb: GL_SRC_ALPHA, srcRgbPma: GL_ONE, dstRgb: GL_ONE_MINUS_SRC_ALPHA, srcAlpha: GL_ONE },\n\t\t{ srcRgb: GL_SRC_ALPHA, srcRgbPma: GL_ONE, dstRgb: GL_ONE, srcAlpha: GL_ONE },\n\t\t{ srcRgb: GL_DST_COLOR, srcRgbPma: GL_DST_COLOR, dstRgb: GL_ONE_MINUS_SRC_ALPHA, srcAlpha: GL_ONE },\n\t\t{ srcRgb: GL_ONE, srcRgbPma: GL_ONE, dstRgb: GL_ONE_MINUS_SRC_COLOR, srcAlpha: GL_ONE }\n\t]\n\n\tsetBlendMode (blendMode: BlendMode, premultipliedAlpha: boolean) {\n\t\tconst blendModeGL = PolygonBatcher.blendModesGL[blendMode];\n\t\tconst srcColorBlend = premultipliedAlpha ? blendModeGL.srcRgbPma : blendModeGL.srcRgb;\n\t\tconst srcAlphaBlend = blendModeGL.srcAlpha;\n\t\tconst dstBlend = blendModeGL.dstRgb;\n\n\t\tif (this.srcColorBlend == srcColorBlend && this.srcAlphaBlend == srcAlphaBlend && this.dstBlend == dstBlend) return;\n\t\tthis.srcColorBlend = srcColorBlend;\n\t\tthis.srcAlphaBlend = srcAlphaBlend;\n\t\tthis.dstBlend = dstBlend;\n\t\tif (this.isDrawing) {\n\t\t\tthis.flush();\n\t\t}\n\t\tlet gl = this.context.gl;\n\t\tgl.blendFuncSeparate(srcColorBlend, dstBlend, srcAlphaBlend, dstBlend);\n\t}\n\n\tdraw (texture: GLTexture, vertices: ArrayLike<number>, indices: Array<number>) {\n\t\tif (texture != this.lastTexture) {\n\t\t\tthis.flush();\n\t\t\tthis.lastTexture = texture;\n\t\t} else if (this.verticesLength + vertices.length > this.mesh.getVertices().length ||\n\t\t\tthis.indicesLength + indices.length > this.mesh.getIndices().length) {\n\t\t\tthis.flush();\n\t\t}\n\n\t\tlet indexStart = this.mesh.numVertices();\n\t\tthis.mesh.getVertices().set(vertices, this.verticesLength);\n\t\tthis.verticesLength += vertices.length;\n\t\tthis.mesh.setVerticesLength(this.verticesLength)\n\n\t\tlet indicesArray = this.mesh.getIndices();\n\t\tfor (let i = this.indicesLength, j = 0; j < indices.length; i++, j++)\n\t\t\tindicesArray[i] = indices[j] + indexStart;\n\t\tthis.indicesLength += indices.length;\n\t\tthis.mesh.setIndicesLength(this.indicesLength);\n\t}\n\n\tflush () {\n\t\tif (this.verticesLength == 0) return;\n\t\tif (!this.lastTexture) throw new Error(\"No texture set.\");\n\t\tif (!this.shader) throw new Error(\"No shader set.\");\n\t\tthis.lastTexture.bind();\n\t\tthis.mesh.draw(this.shader, this.context.gl.TRIANGLES);\n\n\t\tthis.verticesLength = 0;\n\t\tthis.indicesLength = 0;\n\t\tthis.mesh.setVerticesLength(0);\n\t\tthis.mesh.setIndicesLength(0);\n\t\tthis.drawCalls++;\n\t\tPolygonBatcher.globalDrawCalls++;\n\t}\n\n\tend () {\n\t\tif (!this.isDrawing) throw new Error(\"PolygonBatch is not drawing. Call PolygonBatch.begin() before calling PolygonBatch.end()\");\n\t\tif (this.verticesLength > 0 || this.indicesLength > 0) this.flush();\n\t\tthis.shader = null;\n\t\tthis.lastTexture = null;\n\t\tthis.isDrawing = false;\n\n\t\tlet gl = this.context.gl;\n\t\tgl.disable(gl.BLEND);\n\t\tif (PolygonBatcher.disableCulling) {\n\t\t\tif (this.cullWasEnabled) gl.enable(gl.CULL_FACE);\n\t\t}\n\t}\n\n\tgetDrawCalls () {\n\t\treturn this.drawCalls;\n\t}\n\n\tstatic getAndResetGlobalDrawCalls () {\n\t\tlet result = PolygonBatcher.globalDrawCalls;\n\t\tPolygonBatcher.globalDrawCalls = 0;\n\t\treturn result;\n\t}\n\n\tdispose () {\n\t\tthis.mesh.dispose();\n\t}\n}\n"]}