@lightningjs/renderer 2.12.0 → 2.13.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (338) hide show
  1. package/COPYING +1 -0
  2. package/LICENSE +202 -202
  3. package/NOTICE +3 -3
  4. package/README.md +147 -147
  5. package/dist/src/core/CoreNode.d.ts +12 -1
  6. package/dist/src/core/CoreNode.js +113 -34
  7. package/dist/src/core/CoreNode.js.map +1 -1
  8. package/dist/src/core/CoreTextureManager.d.ts +13 -7
  9. package/dist/src/core/CoreTextureManager.js +92 -105
  10. package/dist/src/core/CoreTextureManager.js.map +1 -1
  11. package/dist/src/core/Stage.d.ts +4 -2
  12. package/dist/src/core/Stage.js +23 -7
  13. package/dist/src/core/Stage.js.map +1 -1
  14. package/dist/src/core/TextureMemoryManager.d.ts +17 -6
  15. package/dist/src/core/TextureMemoryManager.js +78 -18
  16. package/dist/src/core/TextureMemoryManager.js.map +1 -1
  17. package/dist/src/core/lib/ImageWorker.d.ts +1 -1
  18. package/dist/src/core/lib/ImageWorker.js +13 -11
  19. package/dist/src/core/lib/ImageWorker.js.map +1 -1
  20. package/dist/src/core/lib/WebGlContextWrapper.d.ts +7 -0
  21. package/dist/src/core/lib/WebGlContextWrapper.js +9 -0
  22. package/dist/src/core/lib/WebGlContextWrapper.js.map +1 -1
  23. package/dist/src/core/lib/validateImageBitmap.d.ts +6 -0
  24. package/dist/src/core/lib/validateImageBitmap.js +68 -0
  25. package/dist/src/core/lib/validateImageBitmap.js.map +1 -0
  26. package/dist/src/core/platform.js +3 -3
  27. package/dist/src/core/platform.js.map +1 -1
  28. package/dist/src/core/renderers/webgl/WebGlCoreCtxRenderTexture.d.ts +2 -1
  29. package/dist/src/core/renderers/webgl/WebGlCoreCtxRenderTexture.js +9 -5
  30. package/dist/src/core/renderers/webgl/WebGlCoreCtxRenderTexture.js.map +1 -1
  31. package/dist/src/core/renderers/webgl/WebGlCoreCtxTexture.js +7 -12
  32. package/dist/src/core/renderers/webgl/WebGlCoreCtxTexture.js.map +1 -1
  33. package/dist/src/core/renderers/webgl/WebGlCoreRenderer.d.ts +1 -1
  34. package/dist/src/core/renderers/webgl/WebGlCoreRenderer.js +18 -18
  35. package/dist/src/core/renderers/webgl/WebGlCoreRenderer.js.map +1 -1
  36. package/dist/src/core/renderers/webgl/shaders/DefaultShader.js +45 -45
  37. package/dist/src/core/renderers/webgl/shaders/DefaultShaderBatched.js +61 -61
  38. package/dist/src/core/renderers/webgl/shaders/DynamicShader.js +93 -93
  39. package/dist/src/core/renderers/webgl/shaders/RoundedRectangle.js +63 -63
  40. package/dist/src/core/renderers/webgl/shaders/SdfShader.js +62 -62
  41. package/dist/src/core/renderers/webgl/shaders/effects/BorderBottomEffect.js +15 -15
  42. package/dist/src/core/renderers/webgl/shaders/effects/BorderEffect.js +6 -6
  43. package/dist/src/core/renderers/webgl/shaders/effects/BorderLeftEffect.js +15 -15
  44. package/dist/src/core/renderers/webgl/shaders/effects/BorderRightEffect.js +15 -15
  45. package/dist/src/core/renderers/webgl/shaders/effects/BorderTopEffect.js +15 -15
  46. package/dist/src/core/renderers/webgl/shaders/effects/FadeOutEffect.js +42 -42
  47. package/dist/src/core/renderers/webgl/shaders/effects/GlitchEffect.js +44 -44
  48. package/dist/src/core/renderers/webgl/shaders/effects/GrayscaleEffect.js +3 -3
  49. package/dist/src/core/renderers/webgl/shaders/effects/HolePunchEffect.js +22 -22
  50. package/dist/src/core/renderers/webgl/shaders/effects/LinearGradientEffect.js +28 -28
  51. package/dist/src/core/renderers/webgl/shaders/effects/RadialGradientEffect.js +10 -10
  52. package/dist/src/core/renderers/webgl/shaders/effects/RadialProgressEffect.js +37 -37
  53. package/dist/src/core/renderers/webgl/shaders/effects/RadiusEffect.js +19 -19
  54. package/dist/src/core/textures/ImageTexture.js +10 -16
  55. package/dist/src/core/textures/ImageTexture.js.map +1 -1
  56. package/dist/src/main-api/Inspector.js +16 -2
  57. package/dist/src/main-api/Inspector.js.map +1 -1
  58. package/dist/src/main-api/Renderer.d.ts +27 -1
  59. package/dist/src/main-api/Renderer.js +8 -3
  60. package/dist/src/main-api/Renderer.js.map +1 -1
  61. package/dist/tsconfig.dist.tsbuildinfo +1 -1
  62. package/exports/canvas.ts +39 -39
  63. package/exports/index.ts +89 -89
  64. package/exports/inspector.ts +24 -24
  65. package/exports/utils.ts +44 -44
  66. package/exports/webgl.ts +38 -38
  67. package/package.json +1 -2
  68. package/scripts/please-use-pnpm.js +13 -13
  69. package/src/common/CommonTypes.ts +146 -146
  70. package/src/common/EventEmitter.ts +77 -77
  71. package/src/common/IAnimationController.ts +92 -92
  72. package/src/common/IEventEmitter.ts +28 -28
  73. package/src/core/CoreNode.test.ts +202 -199
  74. package/src/core/CoreNode.ts +2455 -2335
  75. package/src/core/CoreShaderManager.ts +292 -292
  76. package/src/core/CoreTextNode.ts +455 -455
  77. package/src/core/CoreTextureManager.ts +597 -596
  78. package/src/core/Stage.ts +743 -722
  79. package/src/core/TextureMemoryManager.ts +395 -310
  80. package/src/core/animations/AnimationManager.ts +38 -38
  81. package/src/core/animations/CoreAnimation.ts +340 -340
  82. package/src/core/animations/CoreAnimationController.ts +157 -157
  83. package/src/core/lib/ContextSpy.ts +41 -41
  84. package/src/core/lib/ImageWorker.ts +279 -271
  85. package/src/core/lib/Matrix3d.ts +244 -244
  86. package/src/core/lib/RenderCoords.ts +86 -86
  87. package/src/core/lib/WebGlContextWrapper.ts +1332 -1322
  88. package/src/core/lib/textureCompression.ts +152 -152
  89. package/src/core/lib/textureSvg.ts +78 -78
  90. package/src/core/lib/utils.ts +310 -310
  91. package/src/core/lib/validateImageBitmap.ts +76 -0
  92. package/src/core/platform.ts +63 -61
  93. package/src/core/renderers/CoreContextTexture.ts +43 -43
  94. package/src/core/renderers/CoreRenderOp.ts +22 -22
  95. package/src/core/renderers/CoreRenderer.ts +115 -115
  96. package/src/core/renderers/CoreShader.ts +41 -41
  97. package/src/core/renderers/canvas/CanvasCoreRenderer.ts +375 -375
  98. package/src/core/renderers/canvas/CanvasCoreTexture.ts +153 -153
  99. package/src/core/renderers/canvas/internal/C2DShaderUtils.ts +231 -231
  100. package/src/core/renderers/canvas/internal/ColorUtils.ts +69 -69
  101. package/src/core/renderers/canvas/shaders/UnsupportedShader.ts +48 -48
  102. package/src/core/renderers/webgl/WebGlCoreCtxRenderTexture.ts +86 -79
  103. package/src/core/renderers/webgl/WebGlCoreCtxSubTexture.ts +50 -50
  104. package/src/core/renderers/webgl/WebGlCoreCtxTexture.ts +301 -303
  105. package/src/core/renderers/webgl/WebGlCoreRenderOp.ts +125 -125
  106. package/src/core/renderers/webgl/WebGlCoreRenderer.ts +815 -814
  107. package/src/core/renderers/webgl/WebGlCoreShader.ts +362 -362
  108. package/src/core/renderers/webgl/internal/BufferCollection.ts +54 -54
  109. package/src/core/renderers/webgl/internal/RendererUtils.ts +155 -155
  110. package/src/core/renderers/webgl/internal/ShaderUtils.ts +143 -143
  111. package/src/core/renderers/webgl/internal/WebGlUtils.ts +35 -35
  112. package/src/core/renderers/webgl/shaders/DefaultShader.ts +93 -93
  113. package/src/core/renderers/webgl/shaders/DefaultShaderBatched.ts +132 -132
  114. package/src/core/renderers/webgl/shaders/DynamicShader.ts +580 -580
  115. package/src/core/renderers/webgl/shaders/RoundedRectangle.ts +167 -167
  116. package/src/core/renderers/webgl/shaders/SdfShader.ts +204 -204
  117. package/src/core/renderers/webgl/shaders/effects/BorderBottomEffect.ts +101 -101
  118. package/src/core/renderers/webgl/shaders/effects/BorderEffect.ts +87 -87
  119. package/src/core/renderers/webgl/shaders/effects/BorderLeftEffect.ts +101 -101
  120. package/src/core/renderers/webgl/shaders/effects/BorderRightEffect.ts +101 -101
  121. package/src/core/renderers/webgl/shaders/effects/BorderTopEffect.ts +101 -101
  122. package/src/core/renderers/webgl/shaders/effects/EffectUtils.ts +159 -159
  123. package/src/core/renderers/webgl/shaders/effects/FadeOutEffect.ts +127 -127
  124. package/src/core/renderers/webgl/shaders/effects/GlitchEffect.ts +148 -148
  125. package/src/core/renderers/webgl/shaders/effects/GrayscaleEffect.ts +67 -67
  126. package/src/core/renderers/webgl/shaders/effects/HolePunchEffect.ts +157 -157
  127. package/src/core/renderers/webgl/shaders/effects/LinearGradientEffect.ts +171 -171
  128. package/src/core/renderers/webgl/shaders/effects/RadialGradientEffect.ts +168 -168
  129. package/src/core/renderers/webgl/shaders/effects/RadialProgressEffect.ts +187 -187
  130. package/src/core/renderers/webgl/shaders/effects/RadiusEffect.ts +110 -110
  131. package/src/core/renderers/webgl/shaders/effects/ShaderEffect.ts +196 -196
  132. package/src/core/text-rendering/TextRenderingUtils.ts +36 -36
  133. package/src/core/text-rendering/TextTextureRendererUtils.ts +263 -263
  134. package/src/core/text-rendering/TrFontManager.ts +183 -183
  135. package/src/core/text-rendering/font-face-types/SdfTrFontFace/SdfTrFontFace.ts +176 -176
  136. package/src/core/text-rendering/font-face-types/SdfTrFontFace/internal/FontShaper.ts +139 -139
  137. package/src/core/text-rendering/font-face-types/SdfTrFontFace/internal/SdfFontShaper.test.ts +173 -173
  138. package/src/core/text-rendering/font-face-types/SdfTrFontFace/internal/SdfFontShaper.ts +171 -171
  139. package/src/core/text-rendering/font-face-types/TrFontFace.ts +187 -187
  140. package/src/core/text-rendering/font-face-types/WebTrFontFace.ts +94 -94
  141. package/src/core/text-rendering/font-face-types/utils.ts +39 -39
  142. package/src/core/text-rendering/renderers/CanvasTextRenderer.ts +509 -509
  143. package/src/core/text-rendering/renderers/LightningTextTextureRenderer.ts +808 -808
  144. package/src/core/text-rendering/renderers/SdfTextRenderer/SdfTextRenderer.ts +853 -853
  145. package/src/core/text-rendering/renderers/SdfTextRenderer/internal/PeekableGenerator.test.ts +48 -48
  146. package/src/core/text-rendering/renderers/SdfTextRenderer/internal/PeekableGenerator.ts +66 -66
  147. package/src/core/text-rendering/renderers/SdfTextRenderer/internal/SpecialCodepoints.ts +52 -52
  148. package/src/core/text-rendering/renderers/SdfTextRenderer/internal/constants.ts +32 -32
  149. package/src/core/text-rendering/renderers/SdfTextRenderer/internal/getStartConditions.ts +117 -117
  150. package/src/core/text-rendering/renderers/SdfTextRenderer/internal/getUnicodeCodepoints.test.ts +133 -133
  151. package/src/core/text-rendering/renderers/SdfTextRenderer/internal/getUnicodeCodepoints.ts +38 -38
  152. package/src/core/text-rendering/renderers/SdfTextRenderer/internal/layoutText.ts +408 -408
  153. package/src/core/text-rendering/renderers/SdfTextRenderer/internal/measureText.test.ts +49 -49
  154. package/src/core/text-rendering/renderers/SdfTextRenderer/internal/measureText.ts +52 -52
  155. package/src/core/text-rendering/renderers/SdfTextRenderer/internal/setRenderWindow.test.ts +205 -205
  156. package/src/core/text-rendering/renderers/SdfTextRenderer/internal/setRenderWindow.ts +93 -93
  157. package/src/core/text-rendering/renderers/SdfTextRenderer/internal/util.ts +40 -40
  158. package/src/core/text-rendering/renderers/TextRenderer.ts +557 -557
  159. package/src/core/textures/ColorTexture.ts +102 -102
  160. package/src/core/textures/ImageTexture.ts +376 -382
  161. package/src/core/textures/NoiseTexture.ts +104 -104
  162. package/src/core/textures/RenderTexture.ts +85 -85
  163. package/src/core/textures/SubTexture.ts +205 -205
  164. package/src/core/textures/Texture.ts +337 -337
  165. package/src/core/utils.ts +227 -227
  166. package/src/env.d.ts +7 -7
  167. package/src/main-api/DynamicShaderController.ts +104 -104
  168. package/src/main-api/INode.ts +101 -101
  169. package/src/main-api/Inspector.ts +522 -505
  170. package/src/main-api/Renderer.ts +751 -720
  171. package/src/main-api/ShaderController.ts +80 -80
  172. package/src/main-api/utils.ts +45 -45
  173. package/src/utils.ts +248 -248
  174. package/dist/exports/core-api.d.ts +0 -74
  175. package/dist/exports/core-api.js +0 -96
  176. package/dist/exports/core-api.js.map +0 -1
  177. package/dist/exports/main-api.d.ts +0 -30
  178. package/dist/exports/main-api.js +0 -45
  179. package/dist/exports/main-api.js.map +0 -1
  180. package/dist/src/core/CoreExtension.d.ts +0 -12
  181. package/dist/src/core/CoreExtension.js +0 -29
  182. package/dist/src/core/CoreExtension.js.map +0 -1
  183. package/dist/src/core/CoreStuff.d.ts +0 -1
  184. package/dist/src/core/CoreStuff.js +0 -138
  185. package/dist/src/core/CoreStuff.js.map +0 -1
  186. package/dist/src/core/CoreTexturizer.d.ts +0 -14
  187. package/dist/src/core/CoreTexturizer.js +0 -47
  188. package/dist/src/core/CoreTexturizer.js.map +0 -1
  189. package/dist/src/core/LngNode.d.ts +0 -736
  190. package/dist/src/core/LngNode.js +0 -1174
  191. package/dist/src/core/LngNode.js.map +0 -1
  192. package/dist/src/core/Matrix2DContext.d.ts +0 -15
  193. package/dist/src/core/Matrix2DContext.js +0 -45
  194. package/dist/src/core/Matrix2DContext.js.map +0 -1
  195. package/dist/src/core/ShaderNode.d.ts +0 -10
  196. package/dist/src/core/ShaderNode.js +0 -30
  197. package/dist/src/core/ShaderNode.js.map +0 -1
  198. package/dist/src/core/TextNode.d.ts +0 -103
  199. package/dist/src/core/TextNode.js +0 -331
  200. package/dist/src/core/TextNode.js.map +0 -1
  201. package/dist/src/core/lib/Coords.d.ts +0 -14
  202. package/dist/src/core/lib/Coords.js +0 -55
  203. package/dist/src/core/lib/Coords.js.map +0 -1
  204. package/dist/src/core/lib/glm/common.d.ts +0 -162
  205. package/dist/src/core/lib/glm/common.js +0 -81
  206. package/dist/src/core/lib/glm/common.js.map +0 -1
  207. package/dist/src/core/lib/glm/index.d.ts +0 -11
  208. package/dist/src/core/lib/glm/index.js +0 -30
  209. package/dist/src/core/lib/glm/index.js.map +0 -1
  210. package/dist/src/core/lib/glm/mat2.d.ts +0 -219
  211. package/dist/src/core/lib/glm/mat2.js +0 -396
  212. package/dist/src/core/lib/glm/mat2.js.map +0 -1
  213. package/dist/src/core/lib/glm/mat2d.d.ts +0 -237
  214. package/dist/src/core/lib/glm/mat2d.js +0 -442
  215. package/dist/src/core/lib/glm/mat2d.js.map +0 -1
  216. package/dist/src/core/lib/glm/mat3.d.ts +0 -283
  217. package/dist/src/core/lib/glm/mat3.js +0 -680
  218. package/dist/src/core/lib/glm/mat3.js.map +0 -1
  219. package/dist/src/core/lib/glm/mat4.d.ts +0 -550
  220. package/dist/src/core/lib/glm/mat4.js +0 -1802
  221. package/dist/src/core/lib/glm/mat4.js.map +0 -1
  222. package/dist/src/core/lib/glm/quat.d.ts +0 -363
  223. package/dist/src/core/lib/glm/quat.js +0 -693
  224. package/dist/src/core/lib/glm/quat.js.map +0 -1
  225. package/dist/src/core/lib/glm/quat2.d.ts +0 -356
  226. package/dist/src/core/lib/glm/quat2.js +0 -754
  227. package/dist/src/core/lib/glm/quat2.js.map +0 -1
  228. package/dist/src/core/lib/glm/vec2.d.ts +0 -365
  229. package/dist/src/core/lib/glm/vec2.js +0 -569
  230. package/dist/src/core/lib/glm/vec2.js.map +0 -1
  231. package/dist/src/core/lib/glm/vec3.d.ts +0 -406
  232. package/dist/src/core/lib/glm/vec3.js +0 -720
  233. package/dist/src/core/lib/glm/vec3.js.map +0 -1
  234. package/dist/src/core/lib/glm/vec4.d.ts +0 -330
  235. package/dist/src/core/lib/glm/vec4.js +0 -608
  236. package/dist/src/core/lib/glm/vec4.js.map +0 -1
  237. package/dist/src/core/renderers/CoreShaderManager.d.ts +0 -19
  238. package/dist/src/core/renderers/CoreShaderManager.js +0 -33
  239. package/dist/src/core/renderers/CoreShaderManager.js.map +0 -1
  240. package/dist/src/core/renderers/webgl/WebGlCoreShaderManager.d.ts +0 -27
  241. package/dist/src/core/renderers/webgl/WebGlCoreShaderManager.js +0 -82
  242. package/dist/src/core/renderers/webgl/WebGlCoreShaderManager.js.map +0 -1
  243. package/dist/src/core/renderers/webgl/WebGlCoreShaderProgram.d.ts +0 -11
  244. package/dist/src/core/renderers/webgl/WebGlCoreShaderProgram.js +0 -34
  245. package/dist/src/core/renderers/webgl/WebGlCoreShaderProgram.js.map +0 -1
  246. package/dist/src/core/scene/Scene.d.ts +0 -59
  247. package/dist/src/core/scene/Scene.js +0 -106
  248. package/dist/src/core/scene/Scene.js.map +0 -1
  249. package/dist/src/core/text-rendering/renderers/SdfTextRenderer/internal/makeRenderWindow.d.ts +0 -20
  250. package/dist/src/core/text-rendering/renderers/SdfTextRenderer/internal/makeRenderWindow.js +0 -55
  251. package/dist/src/core/text-rendering/renderers/SdfTextRenderer/internal/makeRenderWindow.js.map +0 -1
  252. package/dist/src/main-api/ICoreDriver.d.ts +0 -27
  253. package/dist/src/main-api/ICoreDriver.js +0 -20
  254. package/dist/src/main-api/ICoreDriver.js.map +0 -1
  255. package/dist/src/main-api/IRenderDriver.d.ts +0 -20
  256. package/dist/src/main-api/IRenderDriver.js +0 -20
  257. package/dist/src/main-api/IRenderDriver.js.map +0 -1
  258. package/dist/src/main-api/IShaderController.d.ts +0 -14
  259. package/dist/src/main-api/IShaderController.js +0 -30
  260. package/dist/src/main-api/IShaderController.js.map +0 -1
  261. package/dist/src/main-api/IShaderNode.d.ts +0 -17
  262. package/dist/src/main-api/IShaderNode.js +0 -19
  263. package/dist/src/main-api/IShaderNode.js.map +0 -1
  264. package/dist/src/main-api/RendererMain.d.ts +0 -375
  265. package/dist/src/main-api/RendererMain.js +0 -365
  266. package/dist/src/main-api/RendererMain.js.map +0 -1
  267. package/dist/src/main-api/texture-usage-trackers/FinalizationRegistryTextureUsageTracker.d.ts +0 -9
  268. package/dist/src/main-api/texture-usage-trackers/FinalizationRegistryTextureUsageTracker.js +0 -38
  269. package/dist/src/main-api/texture-usage-trackers/FinalizationRegistryTextureUsageTracker.js.map +0 -1
  270. package/dist/src/main-api/texture-usage-trackers/ManualCountTextureUsageTracker.d.ts +0 -56
  271. package/dist/src/main-api/texture-usage-trackers/ManualCountTextureUsageTracker.js +0 -101
  272. package/dist/src/main-api/texture-usage-trackers/ManualCountTextureUsageTracker.js.map +0 -1
  273. package/dist/src/main-api/texture-usage-trackers/TextureUsageTracker.d.ts +0 -32
  274. package/dist/src/main-api/texture-usage-trackers/TextureUsageTracker.js +0 -28
  275. package/dist/src/main-api/texture-usage-trackers/TextureUsageTracker.js.map +0 -1
  276. package/dist/src/render-drivers/main/MainCoreDriver.d.ts +0 -24
  277. package/dist/src/render-drivers/main/MainCoreDriver.js +0 -118
  278. package/dist/src/render-drivers/main/MainCoreDriver.js.map +0 -1
  279. package/dist/src/render-drivers/main/MainOnlyNode.d.ts +0 -99
  280. package/dist/src/render-drivers/main/MainOnlyNode.js +0 -396
  281. package/dist/src/render-drivers/main/MainOnlyNode.js.map +0 -1
  282. package/dist/src/render-drivers/main/MainOnlyShaderController.d.ts +0 -6
  283. package/dist/src/render-drivers/main/MainOnlyShaderController.js +0 -15
  284. package/dist/src/render-drivers/main/MainOnlyShaderController.js.map +0 -1
  285. package/dist/src/render-drivers/main/MainOnlyShaderNode.d.ts +0 -7
  286. package/dist/src/render-drivers/main/MainOnlyShaderNode.js +0 -34
  287. package/dist/src/render-drivers/main/MainOnlyShaderNode.js.map +0 -1
  288. package/dist/src/render-drivers/main/MainOnlyTextNode.d.ts +0 -47
  289. package/dist/src/render-drivers/main/MainOnlyTextNode.js +0 -205
  290. package/dist/src/render-drivers/main/MainOnlyTextNode.js.map +0 -1
  291. package/dist/src/render-drivers/main/MainRenderDriver.d.ts +0 -17
  292. package/dist/src/render-drivers/main/MainRenderDriver.js +0 -88
  293. package/dist/src/render-drivers/main/MainRenderDriver.js.map +0 -1
  294. package/dist/src/render-drivers/threadx/NodeStruct.d.ts +0 -90
  295. package/dist/src/render-drivers/threadx/NodeStruct.js +0 -281
  296. package/dist/src/render-drivers/threadx/NodeStruct.js.map +0 -1
  297. package/dist/src/render-drivers/threadx/SharedNode.d.ts +0 -39
  298. package/dist/src/render-drivers/threadx/SharedNode.js +0 -60
  299. package/dist/src/render-drivers/threadx/SharedNode.js.map +0 -1
  300. package/dist/src/render-drivers/threadx/TextNodeStruct.d.ts +0 -44
  301. package/dist/src/render-drivers/threadx/TextNodeStruct.js +0 -201
  302. package/dist/src/render-drivers/threadx/TextNodeStruct.js.map +0 -1
  303. package/dist/src/render-drivers/threadx/ThreadXCoreDriver.d.ts +0 -28
  304. package/dist/src/render-drivers/threadx/ThreadXCoreDriver.js +0 -234
  305. package/dist/src/render-drivers/threadx/ThreadXCoreDriver.js.map +0 -1
  306. package/dist/src/render-drivers/threadx/ThreadXMainAnimationController.d.ts +0 -20
  307. package/dist/src/render-drivers/threadx/ThreadXMainAnimationController.js +0 -84
  308. package/dist/src/render-drivers/threadx/ThreadXMainAnimationController.js.map +0 -1
  309. package/dist/src/render-drivers/threadx/ThreadXMainNode.d.ts +0 -44
  310. package/dist/src/render-drivers/threadx/ThreadXMainNode.js +0 -154
  311. package/dist/src/render-drivers/threadx/ThreadXMainNode.js.map +0 -1
  312. package/dist/src/render-drivers/threadx/ThreadXMainShaderController.d.ts +0 -6
  313. package/dist/src/render-drivers/threadx/ThreadXMainShaderController.js +0 -16
  314. package/dist/src/render-drivers/threadx/ThreadXMainShaderController.js.map +0 -1
  315. package/dist/src/render-drivers/threadx/ThreadXMainShaderNode.d.ts +0 -7
  316. package/dist/src/render-drivers/threadx/ThreadXMainShaderNode.js +0 -15
  317. package/dist/src/render-drivers/threadx/ThreadXMainShaderNode.js.map +0 -1
  318. package/dist/src/render-drivers/threadx/ThreadXMainTextNode.d.ts +0 -28
  319. package/dist/src/render-drivers/threadx/ThreadXMainTextNode.js +0 -55
  320. package/dist/src/render-drivers/threadx/ThreadXMainTextNode.js.map +0 -1
  321. package/dist/src/render-drivers/threadx/ThreadXRenderDriver.d.ts +0 -21
  322. package/dist/src/render-drivers/threadx/ThreadXRenderDriver.js +0 -198
  323. package/dist/src/render-drivers/threadx/ThreadXRenderDriver.js.map +0 -1
  324. package/dist/src/render-drivers/threadx/ThreadXRendererMessage.d.ts +0 -70
  325. package/dist/src/render-drivers/threadx/ThreadXRendererMessage.js +0 -32
  326. package/dist/src/render-drivers/threadx/ThreadXRendererMessage.js.map +0 -1
  327. package/dist/src/render-drivers/threadx/worker/ThreadXRendererNode.d.ts +0 -19
  328. package/dist/src/render-drivers/threadx/worker/ThreadXRendererNode.js +0 -177
  329. package/dist/src/render-drivers/threadx/worker/ThreadXRendererNode.js.map +0 -1
  330. package/dist/src/render-drivers/threadx/worker/ThreadXRendererTextNode.d.ts +0 -27
  331. package/dist/src/render-drivers/threadx/worker/ThreadXRendererTextNode.js +0 -108
  332. package/dist/src/render-drivers/threadx/worker/ThreadXRendererTextNode.js.map +0 -1
  333. package/dist/src/render-drivers/threadx/worker/renderer.d.ts +0 -1
  334. package/dist/src/render-drivers/threadx/worker/renderer.js +0 -145
  335. package/dist/src/render-drivers/threadx/worker/renderer.js.map +0 -1
  336. package/dist/src/render-drivers/utils.d.ts +0 -12
  337. package/dist/src/render-drivers/utils.js +0 -69
  338. package/dist/src/render-drivers/utils.js.map +0 -1
@@ -1,814 +1,815 @@
1
- /*
2
- * If not stated otherwise in this file or this component's LICENSE file the
3
- * following copyright and licenses apply:
4
- *
5
- * Copyright 2023 Comcast Cable Communications Management, LLC.
6
- *
7
- * Licensed under the Apache License, Version 2.0 (the License);
8
- * you may not use this file except in compliance with the License.
9
- * You may obtain a copy of the License at
10
- *
11
- * http://www.apache.org/licenses/LICENSE-2.0
12
- *
13
- * Unless required by applicable law or agreed to in writing, software
14
- * distributed under the License is distributed on an "AS IS" BASIS,
15
- * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
16
- * See the License for the specific language governing permissions and
17
- * limitations under the License.
18
- */
19
-
20
- import { assertTruthy, createWebGLContext, hasOwn } from '../../../utils.js';
21
- import {
22
- CoreRenderer,
23
- type BufferInfo,
24
- type CoreRendererOptions,
25
- type QuadOptions,
26
- } from '../CoreRenderer.js';
27
- import { WebGlCoreRenderOp } from './WebGlCoreRenderOp.js';
28
- import type { CoreContextTexture } from '../CoreContextTexture.js';
29
- import {
30
- createIndexBuffer,
31
- type CoreWebGlParameters,
32
- type CoreWebGlExtensions,
33
- getWebGlParameters,
34
- getWebGlExtensions,
35
- type WebGlColor,
36
- } from './internal/RendererUtils.js';
37
- import { WebGlCoreCtxTexture } from './WebGlCoreCtxTexture.js';
38
- import { Texture, TextureType } from '../../textures/Texture.js';
39
- import { SubTexture } from '../../textures/SubTexture.js';
40
- import { WebGlCoreCtxSubTexture } from './WebGlCoreCtxSubTexture.js';
41
- import { CoreShaderManager } from '../../CoreShaderManager.js';
42
- import { BufferCollection } from './internal/BufferCollection.js';
43
- import {
44
- compareRect,
45
- getNormalizedRgbaComponents,
46
- type RectWithValid,
47
- } from '../../lib/utils.js';
48
- import type { Dimensions } from '../../../common/CommonTypes.js';
49
- import { WebGlCoreShader } from './WebGlCoreShader.js';
50
- import { WebGlContextWrapper } from '../../lib/WebGlContextWrapper.js';
51
- import { RenderTexture } from '../../textures/RenderTexture.js';
52
- import type { CoreNode } from '../../CoreNode.js';
53
- import { WebGlCoreCtxRenderTexture } from './WebGlCoreCtxRenderTexture.js';
54
- import type { BaseShaderController } from '../../../main-api/ShaderController.js';
55
-
56
- const WORDS_PER_QUAD = 24;
57
- // const BYTES_PER_QUAD = WORDS_PER_QUAD * 4;
58
-
59
- export type WebGlCoreRendererOptions = CoreRendererOptions;
60
-
61
- interface CoreWebGlSystem {
62
- parameters: CoreWebGlParameters;
63
- extensions: CoreWebGlExtensions;
64
- }
65
-
66
- export class WebGlCoreRenderer extends CoreRenderer {
67
- //// WebGL Native Context and Data
68
- glw: WebGlContextWrapper;
69
- system: CoreWebGlSystem;
70
-
71
- //// Persistent data
72
- quadBuffer: ArrayBuffer;
73
- fQuadBuffer: Float32Array;
74
- uiQuadBuffer: Uint32Array;
75
- renderOps: WebGlCoreRenderOp[] = [];
76
-
77
- //// Render Op / Buffer Filling State
78
- curBufferIdx = 0;
79
- curRenderOp: WebGlCoreRenderOp | null = null;
80
- override rttNodes: CoreNode[] = [];
81
- activeRttNode: CoreNode | null = null;
82
-
83
- //// Default Shader
84
- defShaderCtrl: BaseShaderController;
85
- defaultShader: WebGlCoreShader;
86
- quadBufferCollection: BufferCollection;
87
-
88
- clearColor: WebGlColor = {
89
- raw: 0x00000000,
90
- normalized: [0, 0, 0, 0],
91
- };
92
-
93
- /**
94
- * White pixel texture used by default when no texture is specified.
95
- */
96
-
97
- quadBufferUsage = 0;
98
- numQuadsRendered = 0;
99
- /**
100
- * Whether the renderer is currently rendering to a texture.
101
- */
102
- public renderToTextureActive = false;
103
-
104
- constructor(options: WebGlCoreRendererOptions) {
105
- super(options);
106
-
107
- this.quadBuffer = new ArrayBuffer(this.stage.options.quadBufferSize);
108
- this.fQuadBuffer = new Float32Array(this.quadBuffer);
109
- this.uiQuadBuffer = new Uint32Array(this.quadBuffer);
110
-
111
- this.mode = 'webgl';
112
-
113
- const { canvas, clearColor, bufferMemory } = options;
114
-
115
- const gl = createWebGLContext(
116
- canvas,
117
- options.forceWebGL2,
118
- options.contextSpy,
119
- );
120
- const glw = (this.glw = new WebGlContextWrapper(gl));
121
- glw.viewport(0, 0, canvas.width, canvas.height);
122
-
123
- this.updateClearColor(clearColor);
124
-
125
- glw.setBlend(true);
126
- glw.blendFunc(glw.ONE, glw.ONE_MINUS_SRC_ALPHA);
127
-
128
- createIndexBuffer(glw, bufferMemory);
129
-
130
- this.system = {
131
- parameters: getWebGlParameters(this.glw),
132
- extensions: getWebGlExtensions(this.glw),
133
- };
134
- this.shManager.renderer = this;
135
- this.defShaderCtrl = this.shManager.loadShader('DefaultShader');
136
- this.defaultShader = this.defShaderCtrl.shader as WebGlCoreShader;
137
- const quadBuffer = glw.createBuffer();
138
- assertTruthy(quadBuffer);
139
- const stride = 6 * Float32Array.BYTES_PER_ELEMENT;
140
- this.quadBufferCollection = new BufferCollection([
141
- {
142
- buffer: quadBuffer,
143
- attributes: {
144
- a_position: {
145
- name: 'a_position',
146
- size: 2, // 2 components per iteration
147
- type: glw.FLOAT, // the data is 32bit floats
148
- normalized: false, // don't normalize the data
149
- stride, // 0 = move forward size * sizeof(type) each iteration to get the next position
150
- offset: 0, // start at the beginning of the buffer
151
- },
152
- a_textureCoordinate: {
153
- name: 'a_textureCoordinate',
154
- size: 2,
155
- type: glw.FLOAT,
156
- normalized: false,
157
- stride,
158
- offset: 2 * Float32Array.BYTES_PER_ELEMENT,
159
- },
160
- a_color: {
161
- name: 'a_color',
162
- size: 4,
163
- type: glw.UNSIGNED_BYTE,
164
- normalized: true,
165
- stride,
166
- offset: 4 * Float32Array.BYTES_PER_ELEMENT,
167
- },
168
- a_textureIndex: {
169
- name: 'a_textureIndex',
170
- size: 1,
171
- type: glw.FLOAT,
172
- normalized: false,
173
- stride,
174
- offset: 5 * Float32Array.BYTES_PER_ELEMENT,
175
- },
176
- },
177
- },
178
- ]);
179
- }
180
-
181
- reset() {
182
- const { glw } = this;
183
- this.curBufferIdx = 0;
184
- this.curRenderOp = null;
185
- this.renderOps.length = 0;
186
- glw.setScissorTest(false);
187
- glw.clear();
188
- }
189
-
190
- override getShaderManager(): CoreShaderManager {
191
- return this.shManager;
192
- }
193
-
194
- override createCtxTexture(textureSource: Texture): CoreContextTexture {
195
- if (textureSource instanceof SubTexture) {
196
- return new WebGlCoreCtxSubTexture(
197
- this.glw,
198
- this.txMemManager,
199
- textureSource,
200
- );
201
- } else if (textureSource instanceof RenderTexture) {
202
- return new WebGlCoreCtxRenderTexture(
203
- this.glw,
204
- this.txMemManager,
205
- textureSource,
206
- );
207
- }
208
- return new WebGlCoreCtxTexture(this.glw, this.txMemManager, textureSource);
209
- }
210
-
211
- /**
212
- * This function adds a quad (a rectangle composed of two triangles) to the WebGL rendering pipeline.
213
- *
214
- * It takes a set of options that define the quad's properties, such as its dimensions, colors, texture, shader, and transformation matrix.
215
- * The function first updates the shader properties with the current dimensions if necessary, then sets the default texture if none is provided.
216
- * It then checks if a new render operation is needed, based on the current shader and clipping rectangle.
217
- * If a new render operation is needed, it creates one and updates the current render operation.
218
- * The function then adjusts the texture coordinates based on the texture options and adds the texture to the texture manager.
219
- *
220
- * Finally, it calculates the vertices for the quad, taking into account any transformations, and adds them to the quad buffer.
221
- * The function updates the length and number of quads in the current render operation, and updates the current buffer index.
222
- */
223
- addQuad(params: QuadOptions) {
224
- const { fQuadBuffer, uiQuadBuffer } = this;
225
- let texture = params.texture;
226
-
227
- assertTruthy(texture !== null, 'Texture is required');
228
-
229
- /**
230
- * If the shader props contain any automatic properties, update it with the
231
- * current dimensions and or alpha that will be used to render the quad.
232
- */
233
- if (params.shaderProps !== null) {
234
- if (hasOwn(params.shaderProps, '$dimensions') == true) {
235
- const dimensions = params.shaderProps.$dimensions as Dimensions;
236
- dimensions.width = params.width;
237
- dimensions.height = params.height;
238
- }
239
-
240
- if (hasOwn(params.shaderProps, '$alpha') === true) {
241
- params.shaderProps.$alpha = params.alpha;
242
- }
243
- }
244
-
245
- let { curBufferIdx: bufferIdx, curRenderOp } = this;
246
- const targetDims = { width: -1, height: -1 };
247
- targetDims.width = params.width;
248
- targetDims.height = params.height;
249
-
250
- const targetShader =
251
- (params.shader as WebGlCoreShader) || this.defaultShader;
252
- assertTruthy(
253
- targetShader.getUniformLocation !== undefined,
254
- 'Invalid WebGL shader',
255
- );
256
-
257
- if (this.reuseRenderOp(params) === false) {
258
- this.newRenderOp(
259
- targetShader,
260
- params.shaderProps as Record<string, unknown>,
261
- params.alpha,
262
- targetDims,
263
- params.clippingRect,
264
- bufferIdx,
265
- params.rtt,
266
- params.parentHasRenderTexture,
267
- params.framebufferDimensions,
268
- );
269
- curRenderOp = this.curRenderOp;
270
- assertTruthy(curRenderOp);
271
- }
272
-
273
- let texCoordX1 = 0;
274
- let texCoordY1 = 0;
275
- let texCoordX2 = 1;
276
- let texCoordY2 = 1;
277
-
278
- if (texture.type === TextureType.subTexture) {
279
- const {
280
- x: tx,
281
- y: ty,
282
- width: tw,
283
- height: th,
284
- } = (texture as SubTexture).props;
285
- const { width: parentW = 0, height: parentH = 0 } = (
286
- texture as SubTexture
287
- ).parentTexture.dimensions || { width: 0, height: 0 };
288
- texCoordX1 = tx / parentW;
289
- texCoordX2 = texCoordX1 + tw / parentW;
290
- texCoordY1 = ty / parentH;
291
- texCoordY2 = texCoordY1 + th / parentH;
292
- texture = (texture as SubTexture).parentTexture;
293
- }
294
-
295
- if (
296
- texture.type === TextureType.image &&
297
- params.textureOptions !== null &&
298
- params.textureOptions.resizeMode !== undefined &&
299
- texture.dimensions !== null
300
- ) {
301
- const resizeMode = params.textureOptions.resizeMode;
302
- const { width: tw, height: th } = texture.dimensions;
303
- if (resizeMode.type === 'cover') {
304
- const scaleX = params.width / tw;
305
- const scaleY = params.height / th;
306
- const scale = Math.max(scaleX, scaleY);
307
- const precision = 1 / scale;
308
- // Determine based on width
309
- if (scale && scaleX && scaleX < scale) {
310
- const desiredSize = precision * params.width;
311
- texCoordX1 = (1 - desiredSize / tw) * (resizeMode.clipX ?? 0.5);
312
- texCoordX2 = texCoordX1 + desiredSize / tw;
313
- }
314
- // Determine based on height
315
- if (scale && scaleY && scaleY < scale) {
316
- const desiredSize = precision * params.height;
317
- texCoordY1 = (1 - desiredSize / th) * (resizeMode.clipY ?? 0.5);
318
- texCoordY2 = texCoordY1 + desiredSize / th;
319
- }
320
- }
321
- }
322
-
323
- // Flip texture coordinates if dictated by texture options
324
- let flipY = 0;
325
- if (params.textureOptions !== null) {
326
- if (params.textureOptions.flipX === true) {
327
- [texCoordX1, texCoordX2] = [texCoordX2, texCoordX1];
328
- }
329
-
330
- // convert to integer for bitwise operation below
331
- flipY = +(params.textureOptions.flipY || false);
332
- }
333
-
334
- // Eitherone should be true
335
- if (flipY ^ +(texture.type === TextureType.renderToTexture)) {
336
- [texCoordY1, texCoordY2] = [texCoordY2, texCoordY1];
337
- }
338
-
339
- const ctxTexture = texture.ctxTexture as WebGlCoreCtxTexture;
340
- assertTruthy(ctxTexture instanceof WebGlCoreCtxTexture);
341
- const textureIdx = this.addTexture(ctxTexture, bufferIdx);
342
-
343
- assertTruthy(this.curRenderOp !== null);
344
- if (params.renderCoords) {
345
- // Upper-Left
346
- fQuadBuffer[bufferIdx++] = params.renderCoords.x1; // vertexX
347
- fQuadBuffer[bufferIdx++] = params.renderCoords.y1; // vertexY
348
- fQuadBuffer[bufferIdx++] = texCoordX1; // texCoordX
349
- fQuadBuffer[bufferIdx++] = texCoordY1; // texCoordY
350
- uiQuadBuffer[bufferIdx++] = params.colorTl; // color
351
- fQuadBuffer[bufferIdx++] = textureIdx; // texIndex
352
-
353
- // Upper-Right
354
- fQuadBuffer[bufferIdx++] = params.renderCoords.x2;
355
- fQuadBuffer[bufferIdx++] = params.renderCoords.y2;
356
- fQuadBuffer[bufferIdx++] = texCoordX2;
357
- fQuadBuffer[bufferIdx++] = texCoordY1;
358
- uiQuadBuffer[bufferIdx++] = params.colorTr;
359
- fQuadBuffer[bufferIdx++] = textureIdx;
360
-
361
- // Lower-Left
362
- fQuadBuffer[bufferIdx++] = params.renderCoords.x4;
363
- fQuadBuffer[bufferIdx++] = params.renderCoords.y4;
364
- fQuadBuffer[bufferIdx++] = texCoordX1;
365
- fQuadBuffer[bufferIdx++] = texCoordY2;
366
- uiQuadBuffer[bufferIdx++] = params.colorBl;
367
- fQuadBuffer[bufferIdx++] = textureIdx;
368
-
369
- // Lower-Right
370
- fQuadBuffer[bufferIdx++] = params.renderCoords.x3;
371
- fQuadBuffer[bufferIdx++] = params.renderCoords.y3;
372
- fQuadBuffer[bufferIdx++] = texCoordX2;
373
- fQuadBuffer[bufferIdx++] = texCoordY2;
374
- uiQuadBuffer[bufferIdx++] = params.colorBr;
375
- fQuadBuffer[bufferIdx++] = textureIdx;
376
- } else if (params.tb !== 0 || params.tc !== 0) {
377
- // Upper-Left
378
- fQuadBuffer[bufferIdx++] = params.tx; // vertexX
379
- fQuadBuffer[bufferIdx++] = params.ty; // vertexY
380
- fQuadBuffer[bufferIdx++] = texCoordX1; // texCoordX
381
- fQuadBuffer[bufferIdx++] = texCoordY1; // texCoordY
382
- uiQuadBuffer[bufferIdx++] = params.colorTl; // color
383
- fQuadBuffer[bufferIdx++] = textureIdx; // texIndex
384
-
385
- // Upper-Right
386
- fQuadBuffer[bufferIdx++] = params.tx + params.width * params.ta;
387
- fQuadBuffer[bufferIdx++] = params.ty + params.width * params.tc;
388
- fQuadBuffer[bufferIdx++] = texCoordX2;
389
- fQuadBuffer[bufferIdx++] = texCoordY1;
390
- uiQuadBuffer[bufferIdx++] = params.colorTr;
391
- fQuadBuffer[bufferIdx++] = textureIdx;
392
-
393
- // Lower-Left
394
- fQuadBuffer[bufferIdx++] = params.tx + params.height * params.tb;
395
- fQuadBuffer[bufferIdx++] = params.ty + params.height * params.td;
396
- fQuadBuffer[bufferIdx++] = texCoordX1;
397
- fQuadBuffer[bufferIdx++] = texCoordY2;
398
- uiQuadBuffer[bufferIdx++] = params.colorBl;
399
- fQuadBuffer[bufferIdx++] = textureIdx;
400
-
401
- // Lower-Right
402
- fQuadBuffer[bufferIdx++] =
403
- params.tx + params.width * params.ta + params.height * params.tb;
404
- fQuadBuffer[bufferIdx++] =
405
- params.ty + params.width * params.tc + params.height * params.td;
406
- fQuadBuffer[bufferIdx++] = texCoordX2;
407
- fQuadBuffer[bufferIdx++] = texCoordY2;
408
- uiQuadBuffer[bufferIdx++] = params.colorBr;
409
- fQuadBuffer[bufferIdx++] = textureIdx;
410
- } else {
411
- // Calculate the right corner of the quad
412
- // multiplied by the scale
413
- const rightCornerX = params.tx + params.width * params.ta;
414
- const rightCornerY = params.ty + params.height * params.td;
415
-
416
- // Upper-Left
417
- fQuadBuffer[bufferIdx++] = params.tx; // vertexX
418
- fQuadBuffer[bufferIdx++] = params.ty; // vertexY
419
- fQuadBuffer[bufferIdx++] = texCoordX1; // texCoordX
420
- fQuadBuffer[bufferIdx++] = texCoordY1; // texCoordY
421
- uiQuadBuffer[bufferIdx++] = params.colorTl; // color
422
- fQuadBuffer[bufferIdx++] = textureIdx; // texIndex
423
-
424
- // Upper-Right
425
- fQuadBuffer[bufferIdx++] = rightCornerX;
426
- fQuadBuffer[bufferIdx++] = params.ty;
427
- fQuadBuffer[bufferIdx++] = texCoordX2;
428
- fQuadBuffer[bufferIdx++] = texCoordY1;
429
- uiQuadBuffer[bufferIdx++] = params.colorTr;
430
- fQuadBuffer[bufferIdx++] = textureIdx;
431
-
432
- // Lower-Left
433
- fQuadBuffer[bufferIdx++] = params.tx;
434
- fQuadBuffer[bufferIdx++] = rightCornerY;
435
- fQuadBuffer[bufferIdx++] = texCoordX1;
436
- fQuadBuffer[bufferIdx++] = texCoordY2;
437
- uiQuadBuffer[bufferIdx++] = params.colorBl;
438
- fQuadBuffer[bufferIdx++] = textureIdx;
439
-
440
- // Lower-Right
441
- fQuadBuffer[bufferIdx++] = rightCornerX;
442
- fQuadBuffer[bufferIdx++] = rightCornerY;
443
- fQuadBuffer[bufferIdx++] = texCoordX2;
444
- fQuadBuffer[bufferIdx++] = texCoordY2;
445
- uiQuadBuffer[bufferIdx++] = params.colorBr;
446
- fQuadBuffer[bufferIdx++] = textureIdx;
447
- }
448
- // Update the length of the current render op
449
- this.curRenderOp.length += WORDS_PER_QUAD;
450
- this.curRenderOp.numQuads++;
451
- this.curBufferIdx = bufferIdx;
452
- }
453
-
454
- /**
455
- * Replace the existing RenderOp with a new one that uses the specified Shader
456
- * and starts at the specified buffer index.
457
- *
458
- * @param shader
459
- * @param bufferIdx
460
- */
461
- private newRenderOp(
462
- shader: WebGlCoreShader,
463
- shaderProps: Record<string, unknown>,
464
- alpha: number,
465
- dimensions: Dimensions,
466
- clippingRect: RectWithValid,
467
- bufferIdx: number,
468
- renderToTexture?: boolean,
469
- parentHasRenderTexture?: boolean,
470
- framebufferDimensions?: Dimensions,
471
- ) {
472
- const curRenderOp = new WebGlCoreRenderOp(
473
- this.glw,
474
- this.options,
475
- this.quadBufferCollection,
476
- shader,
477
- shaderProps,
478
- alpha,
479
- clippingRect,
480
- dimensions,
481
- bufferIdx,
482
- 0, // Z-Index is only used for explictly added Render Ops
483
- renderToTexture,
484
- parentHasRenderTexture,
485
- framebufferDimensions,
486
- );
487
- this.curRenderOp = curRenderOp;
488
- this.renderOps.push(curRenderOp);
489
- }
490
-
491
- /**
492
- * Add a texture to the current RenderOp. If the texture cannot be added to the
493
- * current RenderOp, a new RenderOp will be created and the texture will be added
494
- * to that one.
495
- *
496
- * If the texture cannot be added to the new RenderOp, an error will be thrown.
497
- *
498
- * @param texture
499
- * @param bufferIdx
500
- * @param recursive
501
- * @returns Assigned Texture Index of the texture in the render op
502
- */
503
- private addTexture(
504
- texture: WebGlCoreCtxTexture,
505
- bufferIdx: number,
506
- recursive?: boolean,
507
- ): number {
508
- const { curRenderOp } = this;
509
- assertTruthy(curRenderOp);
510
- const textureIdx = curRenderOp.addTexture(texture);
511
- // TODO: Refactor to be more DRY
512
- if (textureIdx === 0xffffffff) {
513
- if (recursive) {
514
- throw new Error('Unable to add texture to render op');
515
- }
516
-
517
- this.newRenderOp(
518
- curRenderOp.shader,
519
- curRenderOp.shaderProps,
520
- curRenderOp.alpha,
521
- curRenderOp.dimensions,
522
- curRenderOp.clippingRect,
523
- bufferIdx,
524
- );
525
- return this.addTexture(texture, bufferIdx, true);
526
- }
527
- return textureIdx;
528
- }
529
-
530
- /**
531
- * Test if the current Render operation can be reused for the specified parameters.
532
- * @param params
533
- * @returns
534
- */
535
- reuseRenderOp(params: QuadOptions): boolean {
536
- const { shader, shaderProps, parentHasRenderTexture, rtt, clippingRect } =
537
- params;
538
-
539
- const targetShader = shader || this.defaultShader;
540
-
541
- // Switching shader program will require a new render operation
542
- if (this.curRenderOp?.shader !== targetShader) {
543
- return false;
544
- }
545
-
546
- // Switching clipping rect will require a new render operation
547
- if (!compareRect(this.curRenderOp.clippingRect, clippingRect)) {
548
- return false;
549
- }
550
-
551
- // Force new render operation if rendering to texture
552
- // @todo: This needs to be improved, render operations could also be reused
553
- // for rendering to texture
554
- if (parentHasRenderTexture || rtt) {
555
- return false;
556
- }
557
-
558
- // Check if the shader can batch the shader properties
559
- if (
560
- this.curRenderOp.shader !== this.defaultShader &&
561
- (!shaderProps ||
562
- !this.curRenderOp.shader.canBatchShaderProps(
563
- this.curRenderOp.shaderProps,
564
- shaderProps,
565
- ))
566
- ) {
567
- return false;
568
- }
569
-
570
- // Render operation can be reused
571
- return true;
572
- }
573
-
574
- /**
575
- * add RenderOp to the render pipeline
576
- */
577
- addRenderOp(renderable: WebGlCoreRenderOp) {
578
- this.renderOps.push(renderable);
579
- this.curRenderOp = null;
580
- }
581
-
582
- /**
583
- * Render the current set of RenderOps to render to the specified surface.
584
- *
585
- * TODO: 'screen' is the only supported surface at the moment.
586
- *
587
- * @param surface
588
- */
589
- render(surface: 'screen' | CoreContextTexture = 'screen'): void {
590
- const { glw, quadBuffer } = this;
591
-
592
- const arr = new Float32Array(quadBuffer, 0, this.curBufferIdx);
593
-
594
- const buffer = this.quadBufferCollection.getBuffer('a_position') ?? null;
595
- glw.arrayBufferData(buffer, arr, glw.STATIC_DRAW);
596
-
597
- const doLog = false; // idx++ % 100 === 0;
598
- if (doLog) {
599
- console.log('renderOps', this.renderOps.length);
600
- }
601
-
602
- for (let i = 0, length = this.renderOps.length; i < length; i++) {
603
- const renderOp = this.renderOps[i] as WebGlCoreRenderOp;
604
- if (doLog) {
605
- console.log('Quads per operation', renderOp.numQuads);
606
- }
607
- renderOp.draw();
608
- }
609
- this.quadBufferUsage = this.curBufferIdx * arr.BYTES_PER_ELEMENT;
610
-
611
- // Calculate the size of each quad in bytes (4 vertices per quad) times the size of each vertex in bytes
612
- const QUAD_SIZE_IN_BYTES = 4 * (6 * arr.BYTES_PER_ELEMENT); // 6 attributes per vertex
613
- this.numQuadsRendered = this.quadBufferUsage / QUAD_SIZE_IN_BYTES;
614
- }
615
-
616
- getQuadCount(): number {
617
- return this.numQuadsRendered;
618
- }
619
-
620
- renderToTexture(node: CoreNode) {
621
- for (let i = 0; i < this.rttNodes.length; i++) {
622
- if (this.rttNodes[i] === node) {
623
- return;
624
- }
625
- }
626
-
627
- this.insertRTTNodeInOrder(node);
628
- }
629
-
630
- /**
631
- * Inserts an RTT node into `this.rttNodes` while maintaining the correct rendering order based on hierarchy.
632
- *
633
- * Rendering order for RTT nodes is critical when nested RTT nodes exist in a parent-child relationship.
634
- * Specifically:
635
- * - Child RTT nodes must be rendered before their RTT-enabled parents to ensure proper texture composition.
636
- * - If an RTT node is added and it has existing RTT children, it should be rendered after those children.
637
- *
638
- * This function addresses both cases by:
639
- * 1. **Checking Upwards**: It traverses the node's hierarchy upwards to identify any RTT parent
640
- * already in `rttNodes`. If an RTT parent is found, the new node is placed before this parent.
641
- * 2. **Checking Downwards**: It traverses the node’s children recursively to find any RTT-enabled
642
- * children that are already in `rttNodes`. If such children are found, the new node is inserted
643
- * after the last (highest index) RTT child node.
644
- *
645
- * The final calculated insertion index ensures the new node is positioned in `rttNodes` to respect
646
- * both parent-before-child and child-before-parent rendering rules, preserving the correct order
647
- * for the WebGL renderer.
648
- *
649
- * @param node - The RTT-enabled CoreNode to be added to `rttNodes` in the appropriate hierarchical position.
650
- */
651
- private insertRTTNodeInOrder(node: CoreNode) {
652
- let insertIndex = this.rttNodes.length; // Default to the end of the array
653
-
654
- // 1. Traverse upwards to ensure the node is placed before its RTT parent (if any).
655
- let currentNode: CoreNode = node;
656
- while (currentNode) {
657
- if (!currentNode.parent) {
658
- break;
659
- }
660
-
661
- const parentIndex = this.rttNodes.indexOf(currentNode.parent);
662
- if (parentIndex !== -1) {
663
- // Found an RTT parent in the list; set insertIndex to place node before the parent
664
- insertIndex = parentIndex;
665
- break;
666
- }
667
-
668
- currentNode = currentNode.parent;
669
- }
670
-
671
- // 2. Traverse downwards to ensure the node is placed after any RTT children.
672
- // Look through each child recursively to see if any are already in rttNodes.
673
- const maxChildIndex = this.findMaxChildRTTIndex(node);
674
- if (maxChildIndex !== -1) {
675
- // Adjust insertIndex to be after the last child RTT node
676
- insertIndex = Math.max(insertIndex, maxChildIndex + 1);
677
- }
678
-
679
- // 3. Insert the node at the calculated position
680
- this.rttNodes.splice(insertIndex, 0, node);
681
- }
682
-
683
- // Helper function to find the highest index of any RTT children of a node within rttNodes
684
- private findMaxChildRTTIndex(node: CoreNode): number {
685
- let maxIndex = -1;
686
-
687
- const traverseChildren = (currentNode: CoreNode) => {
688
- const currentIndex = this.rttNodes.indexOf(currentNode);
689
- if (currentIndex !== -1) {
690
- maxIndex = Math.max(maxIndex, currentIndex);
691
- }
692
-
693
- // Recursively check all children of the current node
694
- for (const child of currentNode.children) {
695
- traverseChildren(child);
696
- }
697
- };
698
-
699
- // Start traversal directly with the provided node
700
- traverseChildren(node);
701
-
702
- return maxIndex;
703
- }
704
-
705
- renderRTTNodes() {
706
- const { glw } = this;
707
- const { txManager } = this.stage;
708
-
709
- // Render all associated RTT nodes to their textures
710
- for (let i = 0; i < this.rttNodes.length; i++) {
711
- const node = this.rttNodes[i];
712
-
713
- // Skip nodes that don't have RTT updates
714
- if (!node || !node.hasRTTupdates) {
715
- continue;
716
- }
717
-
718
- if (!node.texture || !node.texture.ctxTexture) {
719
- console.warn('Texture not loaded for RTT node', node);
720
- continue;
721
- }
722
-
723
- // Set the active RTT node to the current node
724
- // So we can prevent rendering children of nested RTT nodes
725
- this.activeRttNode = node;
726
-
727
- assertTruthy(node.texture, 'RTT node missing texture');
728
- const ctxTexture = node.texture.ctxTexture;
729
- assertTruthy(ctxTexture instanceof WebGlCoreCtxRenderTexture);
730
- this.renderToTextureActive = true;
731
-
732
- // Bind the the texture's framebuffer
733
- glw.bindFramebuffer(ctxTexture.framebuffer);
734
-
735
- glw.viewport(0, 0, ctxTexture.w, ctxTexture.h);
736
- // Set the clear color to transparent
737
- glw.clearColor(0, 0, 0, 0);
738
- glw.clear();
739
-
740
- // Render all associated quads to the texture
741
- for (let i = 0; i < node.children.length; i++) {
742
- const child = node.children[i];
743
-
744
- if (!child) {
745
- continue;
746
- }
747
-
748
- this.stage.addQuads(child);
749
- child.hasRTTupdates = false;
750
- }
751
-
752
- // Render all associated quads to the texture
753
- this.render();
754
-
755
- // Reset render operations
756
- this.renderOps.length = 0;
757
- node.hasRTTupdates = false;
758
- }
759
-
760
- const clearColor = this.clearColor.normalized;
761
- // Restore the default clear color
762
- glw.clearColor(clearColor[0], clearColor[1], clearColor[2], clearColor[3]);
763
-
764
- // Bind the default framebuffer
765
- glw.bindFramebuffer(null);
766
-
767
- glw.viewport(0, 0, this.glw.canvas.width, this.glw.canvas.height);
768
- this.renderToTextureActive = false;
769
- }
770
-
771
- removeRTTNode(node: CoreNode) {
772
- const index = this.rttNodes.indexOf(node);
773
- if (index === -1) {
774
- return;
775
- }
776
- this.rttNodes.splice(index, 1);
777
- }
778
-
779
- getBufferInfo(): BufferInfo | null {
780
- const bufferInfo: BufferInfo = {
781
- totalAvailable: this.stage.options.quadBufferSize,
782
- totalUsed: this.quadBufferUsage,
783
- };
784
- return bufferInfo;
785
- }
786
-
787
- override getDefShaderCtr(): BaseShaderController {
788
- return this.defShaderCtrl;
789
- }
790
-
791
- /**
792
- * Updates the WebGL context's clear color and clears the color buffer.
793
- *
794
- * @param color - The color to set as the clear color, represented as a 32-bit integer.
795
- */
796
- updateClearColor(color: number) {
797
- if (this.clearColor.raw === color) {
798
- return;
799
- }
800
- const glw = this.glw;
801
- const normalizedColor = getNormalizedRgbaComponents(color);
802
- glw.clearColor(
803
- normalizedColor[0],
804
- normalizedColor[1],
805
- normalizedColor[2],
806
- normalizedColor[3],
807
- );
808
- this.clearColor = {
809
- raw: color,
810
- normalized: normalizedColor,
811
- };
812
- glw.clear();
813
- }
814
- }
1
+ /*
2
+ * If not stated otherwise in this file or this component's LICENSE file the
3
+ * following copyright and licenses apply:
4
+ *
5
+ * Copyright 2023 Comcast Cable Communications Management, LLC.
6
+ *
7
+ * Licensed under the Apache License, Version 2.0 (the License);
8
+ * you may not use this file except in compliance with the License.
9
+ * You may obtain a copy of the License at
10
+ *
11
+ * http://www.apache.org/licenses/LICENSE-2.0
12
+ *
13
+ * Unless required by applicable law or agreed to in writing, software
14
+ * distributed under the License is distributed on an "AS IS" BASIS,
15
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
16
+ * See the License for the specific language governing permissions and
17
+ * limitations under the License.
18
+ */
19
+
20
+ import { assertTruthy, createWebGLContext, hasOwn } from '../../../utils.js';
21
+ import {
22
+ CoreRenderer,
23
+ type BufferInfo,
24
+ type CoreRendererOptions,
25
+ type QuadOptions,
26
+ } from '../CoreRenderer.js';
27
+ import { WebGlCoreRenderOp } from './WebGlCoreRenderOp.js';
28
+ import type { CoreContextTexture } from '../CoreContextTexture.js';
29
+ import {
30
+ createIndexBuffer,
31
+ type CoreWebGlParameters,
32
+ type CoreWebGlExtensions,
33
+ getWebGlParameters,
34
+ getWebGlExtensions,
35
+ type WebGlColor,
36
+ } from './internal/RendererUtils.js';
37
+ import { WebGlCoreCtxTexture } from './WebGlCoreCtxTexture.js';
38
+ import { Texture, TextureType } from '../../textures/Texture.js';
39
+ import { SubTexture } from '../../textures/SubTexture.js';
40
+ import { WebGlCoreCtxSubTexture } from './WebGlCoreCtxSubTexture.js';
41
+ import { CoreShaderManager } from '../../CoreShaderManager.js';
42
+ import { BufferCollection } from './internal/BufferCollection.js';
43
+ import {
44
+ compareRect,
45
+ getNormalizedRgbaComponents,
46
+ type RectWithValid,
47
+ } from '../../lib/utils.js';
48
+ import type { Dimensions } from '../../../common/CommonTypes.js';
49
+ import { WebGlCoreShader } from './WebGlCoreShader.js';
50
+ import { WebGlContextWrapper } from '../../lib/WebGlContextWrapper.js';
51
+ import { RenderTexture } from '../../textures/RenderTexture.js';
52
+ import { CoreNodeRenderState, type CoreNode } from '../../CoreNode.js';
53
+ import { WebGlCoreCtxRenderTexture } from './WebGlCoreCtxRenderTexture.js';
54
+ import type { BaseShaderController } from '../../../main-api/ShaderController.js';
55
+
56
+ const WORDS_PER_QUAD = 24;
57
+ // const BYTES_PER_QUAD = WORDS_PER_QUAD * 4;
58
+
59
+ export type WebGlCoreRendererOptions = CoreRendererOptions;
60
+
61
+ interface CoreWebGlSystem {
62
+ parameters: CoreWebGlParameters;
63
+ extensions: CoreWebGlExtensions;
64
+ }
65
+
66
+ export class WebGlCoreRenderer extends CoreRenderer {
67
+ //// WebGL Native Context and Data
68
+ glw: WebGlContextWrapper;
69
+ system: CoreWebGlSystem;
70
+
71
+ //// Persistent data
72
+ quadBuffer: ArrayBuffer;
73
+ fQuadBuffer: Float32Array;
74
+ uiQuadBuffer: Uint32Array;
75
+ renderOps: WebGlCoreRenderOp[] = [];
76
+
77
+ //// Render Op / Buffer Filling State
78
+ curBufferIdx = 0;
79
+ curRenderOp: WebGlCoreRenderOp | null = null;
80
+ override rttNodes: CoreNode[] = [];
81
+ activeRttNode: CoreNode | null = null;
82
+
83
+ //// Default Shader
84
+ defShaderCtrl: BaseShaderController;
85
+ defaultShader: WebGlCoreShader;
86
+ quadBufferCollection: BufferCollection;
87
+
88
+ clearColor: WebGlColor = {
89
+ raw: 0x00000000,
90
+ normalized: [0, 0, 0, 0],
91
+ };
92
+
93
+ /**
94
+ * White pixel texture used by default when no texture is specified.
95
+ */
96
+
97
+ quadBufferUsage = 0;
98
+ numQuadsRendered = 0;
99
+ /**
100
+ * Whether the renderer is currently rendering to a texture.
101
+ */
102
+ public renderToTextureActive = false;
103
+
104
+ constructor(options: WebGlCoreRendererOptions) {
105
+ super(options);
106
+
107
+ this.quadBuffer = new ArrayBuffer(this.stage.options.quadBufferSize);
108
+ this.fQuadBuffer = new Float32Array(this.quadBuffer);
109
+ this.uiQuadBuffer = new Uint32Array(this.quadBuffer);
110
+
111
+ this.mode = 'webgl';
112
+
113
+ const { canvas, clearColor, bufferMemory } = options;
114
+
115
+ const gl = createWebGLContext(
116
+ canvas,
117
+ options.forceWebGL2,
118
+ options.contextSpy,
119
+ );
120
+ const glw = (this.glw = new WebGlContextWrapper(gl));
121
+ glw.viewport(0, 0, canvas.width, canvas.height);
122
+
123
+ this.updateClearColor(clearColor);
124
+
125
+ glw.setBlend(true);
126
+ glw.blendFunc(glw.ONE, glw.ONE_MINUS_SRC_ALPHA);
127
+
128
+ createIndexBuffer(glw, bufferMemory);
129
+
130
+ this.system = {
131
+ parameters: getWebGlParameters(this.glw),
132
+ extensions: getWebGlExtensions(this.glw),
133
+ };
134
+ this.shManager.renderer = this;
135
+ this.defShaderCtrl = this.shManager.loadShader('DefaultShader');
136
+ this.defaultShader = this.defShaderCtrl.shader as WebGlCoreShader;
137
+ const quadBuffer = glw.createBuffer();
138
+ assertTruthy(quadBuffer);
139
+ const stride = 6 * Float32Array.BYTES_PER_ELEMENT;
140
+ this.quadBufferCollection = new BufferCollection([
141
+ {
142
+ buffer: quadBuffer,
143
+ attributes: {
144
+ a_position: {
145
+ name: 'a_position',
146
+ size: 2, // 2 components per iteration
147
+ type: glw.FLOAT, // the data is 32bit floats
148
+ normalized: false, // don't normalize the data
149
+ stride, // 0 = move forward size * sizeof(type) each iteration to get the next position
150
+ offset: 0, // start at the beginning of the buffer
151
+ },
152
+ a_textureCoordinate: {
153
+ name: 'a_textureCoordinate',
154
+ size: 2,
155
+ type: glw.FLOAT,
156
+ normalized: false,
157
+ stride,
158
+ offset: 2 * Float32Array.BYTES_PER_ELEMENT,
159
+ },
160
+ a_color: {
161
+ name: 'a_color',
162
+ size: 4,
163
+ type: glw.UNSIGNED_BYTE,
164
+ normalized: true,
165
+ stride,
166
+ offset: 4 * Float32Array.BYTES_PER_ELEMENT,
167
+ },
168
+ a_textureIndex: {
169
+ name: 'a_textureIndex',
170
+ size: 1,
171
+ type: glw.FLOAT,
172
+ normalized: false,
173
+ stride,
174
+ offset: 5 * Float32Array.BYTES_PER_ELEMENT,
175
+ },
176
+ },
177
+ },
178
+ ]);
179
+ }
180
+
181
+ reset() {
182
+ const { glw } = this;
183
+ this.curBufferIdx = 0;
184
+ this.curRenderOp = null;
185
+ this.renderOps.length = 0;
186
+ glw.setScissorTest(false);
187
+ glw.clear();
188
+ }
189
+
190
+ override getShaderManager(): CoreShaderManager {
191
+ return this.shManager;
192
+ }
193
+
194
+ override createCtxTexture(textureSource: Texture): CoreContextTexture {
195
+ if (textureSource instanceof SubTexture) {
196
+ return new WebGlCoreCtxSubTexture(
197
+ this.glw,
198
+ this.txMemManager,
199
+ textureSource,
200
+ );
201
+ } else if (textureSource instanceof RenderTexture) {
202
+ return new WebGlCoreCtxRenderTexture(
203
+ this.glw,
204
+ this.txMemManager,
205
+ textureSource,
206
+ );
207
+ }
208
+ return new WebGlCoreCtxTexture(this.glw, this.txMemManager, textureSource);
209
+ }
210
+
211
+ /**
212
+ * This function adds a quad (a rectangle composed of two triangles) to the WebGL rendering pipeline.
213
+ *
214
+ * It takes a set of options that define the quad's properties, such as its dimensions, colors, texture, shader, and transformation matrix.
215
+ * The function first updates the shader properties with the current dimensions if necessary, then sets the default texture if none is provided.
216
+ * It then checks if a new render operation is needed, based on the current shader and clipping rectangle.
217
+ * If a new render operation is needed, it creates one and updates the current render operation.
218
+ * The function then adjusts the texture coordinates based on the texture options and adds the texture to the texture manager.
219
+ *
220
+ * Finally, it calculates the vertices for the quad, taking into account any transformations, and adds them to the quad buffer.
221
+ * The function updates the length and number of quads in the current render operation, and updates the current buffer index.
222
+ */
223
+ addQuad(params: QuadOptions) {
224
+ const { fQuadBuffer, uiQuadBuffer } = this;
225
+ let texture = params.texture;
226
+
227
+ assertTruthy(texture !== null, 'Texture is required');
228
+
229
+ /**
230
+ * If the shader props contain any automatic properties, update it with the
231
+ * current dimensions and or alpha that will be used to render the quad.
232
+ */
233
+ if (params.shaderProps !== null) {
234
+ if (hasOwn(params.shaderProps, '$dimensions') == true) {
235
+ const dimensions = params.shaderProps.$dimensions as Dimensions;
236
+ dimensions.width = params.width;
237
+ dimensions.height = params.height;
238
+ }
239
+
240
+ if (hasOwn(params.shaderProps, '$alpha') === true) {
241
+ params.shaderProps.$alpha = params.alpha;
242
+ }
243
+ }
244
+
245
+ let { curBufferIdx: bufferIdx, curRenderOp } = this;
246
+ const targetDims = { width: -1, height: -1 };
247
+ targetDims.width = params.width;
248
+ targetDims.height = params.height;
249
+
250
+ const targetShader =
251
+ (params.shader as WebGlCoreShader) || this.defaultShader;
252
+ assertTruthy(
253
+ targetShader.getUniformLocation !== undefined,
254
+ 'Invalid WebGL shader',
255
+ );
256
+
257
+ if (this.reuseRenderOp(params) === false) {
258
+ this.newRenderOp(
259
+ targetShader,
260
+ params.shaderProps as Record<string, unknown>,
261
+ params.alpha,
262
+ targetDims,
263
+ params.clippingRect,
264
+ bufferIdx,
265
+ params.rtt,
266
+ params.parentHasRenderTexture,
267
+ params.framebufferDimensions,
268
+ );
269
+ curRenderOp = this.curRenderOp;
270
+ assertTruthy(curRenderOp);
271
+ }
272
+
273
+ let texCoordX1 = 0;
274
+ let texCoordY1 = 0;
275
+ let texCoordX2 = 1;
276
+ let texCoordY2 = 1;
277
+
278
+ if (texture.type === TextureType.subTexture) {
279
+ const {
280
+ x: tx,
281
+ y: ty,
282
+ width: tw,
283
+ height: th,
284
+ } = (texture as SubTexture).props;
285
+ const { width: parentW = 0, height: parentH = 0 } = (
286
+ texture as SubTexture
287
+ ).parentTexture.dimensions || { width: 0, height: 0 };
288
+ texCoordX1 = tx / parentW;
289
+ texCoordX2 = texCoordX1 + tw / parentW;
290
+ texCoordY1 = ty / parentH;
291
+ texCoordY2 = texCoordY1 + th / parentH;
292
+ texture = (texture as SubTexture).parentTexture;
293
+ }
294
+
295
+ if (
296
+ texture.type === TextureType.image &&
297
+ params.textureOptions !== null &&
298
+ params.textureOptions.resizeMode !== undefined &&
299
+ texture.dimensions !== null
300
+ ) {
301
+ const resizeMode = params.textureOptions.resizeMode;
302
+ const { width: tw, height: th } = texture.dimensions;
303
+ if (resizeMode.type === 'cover') {
304
+ const scaleX = params.width / tw;
305
+ const scaleY = params.height / th;
306
+ const scale = Math.max(scaleX, scaleY);
307
+ const precision = 1 / scale;
308
+ // Determine based on width
309
+ if (scale && scaleX && scaleX < scale) {
310
+ const desiredSize = precision * params.width;
311
+ texCoordX1 = (1 - desiredSize / tw) * (resizeMode.clipX ?? 0.5);
312
+ texCoordX2 = texCoordX1 + desiredSize / tw;
313
+ }
314
+ // Determine based on height
315
+ if (scale && scaleY && scaleY < scale) {
316
+ const desiredSize = precision * params.height;
317
+ texCoordY1 = (1 - desiredSize / th) * (resizeMode.clipY ?? 0.5);
318
+ texCoordY2 = texCoordY1 + desiredSize / th;
319
+ }
320
+ }
321
+ }
322
+
323
+ // Flip texture coordinates if dictated by texture options
324
+ let flipY = 0;
325
+ if (params.textureOptions !== null) {
326
+ if (params.textureOptions.flipX === true) {
327
+ [texCoordX1, texCoordX2] = [texCoordX2, texCoordX1];
328
+ }
329
+
330
+ // convert to integer for bitwise operation below
331
+ flipY = +(params.textureOptions.flipY || false);
332
+ }
333
+
334
+ // Eitherone should be true
335
+ if (flipY ^ +(texture.type === TextureType.renderToTexture)) {
336
+ [texCoordY1, texCoordY2] = [texCoordY2, texCoordY1];
337
+ }
338
+
339
+ const ctxTexture = texture.ctxTexture as WebGlCoreCtxTexture;
340
+ assertTruthy(ctxTexture instanceof WebGlCoreCtxTexture);
341
+ const textureIdx = this.addTexture(ctxTexture, bufferIdx);
342
+
343
+ assertTruthy(this.curRenderOp !== null);
344
+ if (params.renderCoords) {
345
+ // Upper-Left
346
+ fQuadBuffer[bufferIdx++] = params.renderCoords.x1; // vertexX
347
+ fQuadBuffer[bufferIdx++] = params.renderCoords.y1; // vertexY
348
+ fQuadBuffer[bufferIdx++] = texCoordX1; // texCoordX
349
+ fQuadBuffer[bufferIdx++] = texCoordY1; // texCoordY
350
+ uiQuadBuffer[bufferIdx++] = params.colorTl; // color
351
+ fQuadBuffer[bufferIdx++] = textureIdx; // texIndex
352
+
353
+ // Upper-Right
354
+ fQuadBuffer[bufferIdx++] = params.renderCoords.x2;
355
+ fQuadBuffer[bufferIdx++] = params.renderCoords.y2;
356
+ fQuadBuffer[bufferIdx++] = texCoordX2;
357
+ fQuadBuffer[bufferIdx++] = texCoordY1;
358
+ uiQuadBuffer[bufferIdx++] = params.colorTr;
359
+ fQuadBuffer[bufferIdx++] = textureIdx;
360
+
361
+ // Lower-Left
362
+ fQuadBuffer[bufferIdx++] = params.renderCoords.x4;
363
+ fQuadBuffer[bufferIdx++] = params.renderCoords.y4;
364
+ fQuadBuffer[bufferIdx++] = texCoordX1;
365
+ fQuadBuffer[bufferIdx++] = texCoordY2;
366
+ uiQuadBuffer[bufferIdx++] = params.colorBl;
367
+ fQuadBuffer[bufferIdx++] = textureIdx;
368
+
369
+ // Lower-Right
370
+ fQuadBuffer[bufferIdx++] = params.renderCoords.x3;
371
+ fQuadBuffer[bufferIdx++] = params.renderCoords.y3;
372
+ fQuadBuffer[bufferIdx++] = texCoordX2;
373
+ fQuadBuffer[bufferIdx++] = texCoordY2;
374
+ uiQuadBuffer[bufferIdx++] = params.colorBr;
375
+ fQuadBuffer[bufferIdx++] = textureIdx;
376
+ } else if (params.tb !== 0 || params.tc !== 0) {
377
+ // Upper-Left
378
+ fQuadBuffer[bufferIdx++] = params.tx; // vertexX
379
+ fQuadBuffer[bufferIdx++] = params.ty; // vertexY
380
+ fQuadBuffer[bufferIdx++] = texCoordX1; // texCoordX
381
+ fQuadBuffer[bufferIdx++] = texCoordY1; // texCoordY
382
+ uiQuadBuffer[bufferIdx++] = params.colorTl; // color
383
+ fQuadBuffer[bufferIdx++] = textureIdx; // texIndex
384
+
385
+ // Upper-Right
386
+ fQuadBuffer[bufferIdx++] = params.tx + params.width * params.ta;
387
+ fQuadBuffer[bufferIdx++] = params.ty + params.width * params.tc;
388
+ fQuadBuffer[bufferIdx++] = texCoordX2;
389
+ fQuadBuffer[bufferIdx++] = texCoordY1;
390
+ uiQuadBuffer[bufferIdx++] = params.colorTr;
391
+ fQuadBuffer[bufferIdx++] = textureIdx;
392
+
393
+ // Lower-Left
394
+ fQuadBuffer[bufferIdx++] = params.tx + params.height * params.tb;
395
+ fQuadBuffer[bufferIdx++] = params.ty + params.height * params.td;
396
+ fQuadBuffer[bufferIdx++] = texCoordX1;
397
+ fQuadBuffer[bufferIdx++] = texCoordY2;
398
+ uiQuadBuffer[bufferIdx++] = params.colorBl;
399
+ fQuadBuffer[bufferIdx++] = textureIdx;
400
+
401
+ // Lower-Right
402
+ fQuadBuffer[bufferIdx++] =
403
+ params.tx + params.width * params.ta + params.height * params.tb;
404
+ fQuadBuffer[bufferIdx++] =
405
+ params.ty + params.width * params.tc + params.height * params.td;
406
+ fQuadBuffer[bufferIdx++] = texCoordX2;
407
+ fQuadBuffer[bufferIdx++] = texCoordY2;
408
+ uiQuadBuffer[bufferIdx++] = params.colorBr;
409
+ fQuadBuffer[bufferIdx++] = textureIdx;
410
+ } else {
411
+ // Calculate the right corner of the quad
412
+ // multiplied by the scale
413
+ const rightCornerX = params.tx + params.width * params.ta;
414
+ const rightCornerY = params.ty + params.height * params.td;
415
+
416
+ // Upper-Left
417
+ fQuadBuffer[bufferIdx++] = params.tx; // vertexX
418
+ fQuadBuffer[bufferIdx++] = params.ty; // vertexY
419
+ fQuadBuffer[bufferIdx++] = texCoordX1; // texCoordX
420
+ fQuadBuffer[bufferIdx++] = texCoordY1; // texCoordY
421
+ uiQuadBuffer[bufferIdx++] = params.colorTl; // color
422
+ fQuadBuffer[bufferIdx++] = textureIdx; // texIndex
423
+
424
+ // Upper-Right
425
+ fQuadBuffer[bufferIdx++] = rightCornerX;
426
+ fQuadBuffer[bufferIdx++] = params.ty;
427
+ fQuadBuffer[bufferIdx++] = texCoordX2;
428
+ fQuadBuffer[bufferIdx++] = texCoordY1;
429
+ uiQuadBuffer[bufferIdx++] = params.colorTr;
430
+ fQuadBuffer[bufferIdx++] = textureIdx;
431
+
432
+ // Lower-Left
433
+ fQuadBuffer[bufferIdx++] = params.tx;
434
+ fQuadBuffer[bufferIdx++] = rightCornerY;
435
+ fQuadBuffer[bufferIdx++] = texCoordX1;
436
+ fQuadBuffer[bufferIdx++] = texCoordY2;
437
+ uiQuadBuffer[bufferIdx++] = params.colorBl;
438
+ fQuadBuffer[bufferIdx++] = textureIdx;
439
+
440
+ // Lower-Right
441
+ fQuadBuffer[bufferIdx++] = rightCornerX;
442
+ fQuadBuffer[bufferIdx++] = rightCornerY;
443
+ fQuadBuffer[bufferIdx++] = texCoordX2;
444
+ fQuadBuffer[bufferIdx++] = texCoordY2;
445
+ uiQuadBuffer[bufferIdx++] = params.colorBr;
446
+ fQuadBuffer[bufferIdx++] = textureIdx;
447
+ }
448
+ // Update the length of the current render op
449
+ this.curRenderOp.length += WORDS_PER_QUAD;
450
+ this.curRenderOp.numQuads++;
451
+ this.curBufferIdx = bufferIdx;
452
+ }
453
+
454
+ /**
455
+ * Replace the existing RenderOp with a new one that uses the specified Shader
456
+ * and starts at the specified buffer index.
457
+ *
458
+ * @param shader
459
+ * @param bufferIdx
460
+ */
461
+ private newRenderOp(
462
+ shader: WebGlCoreShader,
463
+ shaderProps: Record<string, unknown>,
464
+ alpha: number,
465
+ dimensions: Dimensions,
466
+ clippingRect: RectWithValid,
467
+ bufferIdx: number,
468
+ renderToTexture?: boolean,
469
+ parentHasRenderTexture?: boolean,
470
+ framebufferDimensions?: Dimensions,
471
+ ) {
472
+ const curRenderOp = new WebGlCoreRenderOp(
473
+ this.glw,
474
+ this.options,
475
+ this.quadBufferCollection,
476
+ shader,
477
+ shaderProps,
478
+ alpha,
479
+ clippingRect,
480
+ dimensions,
481
+ bufferIdx,
482
+ 0, // Z-Index is only used for explictly added Render Ops
483
+ renderToTexture,
484
+ parentHasRenderTexture,
485
+ framebufferDimensions,
486
+ );
487
+ this.curRenderOp = curRenderOp;
488
+ this.renderOps.push(curRenderOp);
489
+ }
490
+
491
+ /**
492
+ * Add a texture to the current RenderOp. If the texture cannot be added to the
493
+ * current RenderOp, a new RenderOp will be created and the texture will be added
494
+ * to that one.
495
+ *
496
+ * If the texture cannot be added to the new RenderOp, an error will be thrown.
497
+ *
498
+ * @param texture
499
+ * @param bufferIdx
500
+ * @param recursive
501
+ * @returns Assigned Texture Index of the texture in the render op
502
+ */
503
+ private addTexture(
504
+ texture: WebGlCoreCtxTexture,
505
+ bufferIdx: number,
506
+ recursive?: boolean,
507
+ ): number {
508
+ const { curRenderOp } = this;
509
+ assertTruthy(curRenderOp);
510
+ const textureIdx = curRenderOp.addTexture(texture);
511
+ // TODO: Refactor to be more DRY
512
+ if (textureIdx === 0xffffffff) {
513
+ if (recursive) {
514
+ throw new Error('Unable to add texture to render op');
515
+ }
516
+
517
+ this.newRenderOp(
518
+ curRenderOp.shader,
519
+ curRenderOp.shaderProps,
520
+ curRenderOp.alpha,
521
+ curRenderOp.dimensions,
522
+ curRenderOp.clippingRect,
523
+ bufferIdx,
524
+ );
525
+ return this.addTexture(texture, bufferIdx, true);
526
+ }
527
+ return textureIdx;
528
+ }
529
+
530
+ /**
531
+ * Test if the current Render operation can be reused for the specified parameters.
532
+ * @param params
533
+ * @returns
534
+ */
535
+ reuseRenderOp(params: QuadOptions): boolean {
536
+ const { shader, shaderProps, parentHasRenderTexture, rtt, clippingRect } =
537
+ params;
538
+
539
+ const targetShader = shader || this.defaultShader;
540
+
541
+ // Switching shader program will require a new render operation
542
+ if (this.curRenderOp?.shader !== targetShader) {
543
+ return false;
544
+ }
545
+
546
+ // Switching clipping rect will require a new render operation
547
+ if (compareRect(this.curRenderOp.clippingRect, clippingRect) === false) {
548
+ return false;
549
+ }
550
+
551
+ // Force new render operation if rendering to texture
552
+ // @todo: This needs to be improved, render operations could also be reused
553
+ // for rendering to texture
554
+ if (parentHasRenderTexture !== undefined || rtt !== undefined) {
555
+ return false;
556
+ }
557
+
558
+ // Check if the shader can batch the shader properties
559
+ if (
560
+ this.curRenderOp.shader !== this.defaultShader &&
561
+ (shaderProps === null ||
562
+ this.curRenderOp.shader.canBatchShaderProps(
563
+ this.curRenderOp.shaderProps,
564
+ shaderProps,
565
+ ) === false)
566
+ ) {
567
+ return false;
568
+ }
569
+
570
+ // Render operation can be reused
571
+ return true;
572
+ }
573
+
574
+ /**
575
+ * add RenderOp to the render pipeline
576
+ */
577
+ addRenderOp(renderable: WebGlCoreRenderOp) {
578
+ this.renderOps.push(renderable);
579
+ this.curRenderOp = null;
580
+ }
581
+
582
+ /**
583
+ * Render the current set of RenderOps to render to the specified surface.
584
+ *
585
+ * TODO: 'screen' is the only supported surface at the moment.
586
+ *
587
+ * @param surface
588
+ */
589
+ render(surface: 'screen' | CoreContextTexture = 'screen'): void {
590
+ const { glw, quadBuffer } = this;
591
+
592
+ const arr = new Float32Array(quadBuffer, 0, this.curBufferIdx);
593
+
594
+ const buffer = this.quadBufferCollection.getBuffer('a_position') || null;
595
+ glw.arrayBufferData(buffer, arr, glw.STATIC_DRAW);
596
+
597
+ for (let i = 0, length = this.renderOps.length; i < length; i++) {
598
+ // eslint-disable-next-line @typescript-eslint/no-non-null-assertion
599
+ this.renderOps[i]!.draw();
600
+ }
601
+ this.quadBufferUsage = this.curBufferIdx * arr.BYTES_PER_ELEMENT;
602
+
603
+ // Calculate the size of each quad in bytes (4 vertices per quad) times the size of each vertex in bytes
604
+ const QUAD_SIZE_IN_BYTES = 4 * (6 * arr.BYTES_PER_ELEMENT); // 6 attributes per vertex
605
+ this.numQuadsRendered = this.quadBufferUsage / QUAD_SIZE_IN_BYTES;
606
+ }
607
+
608
+ getQuadCount(): number {
609
+ return this.numQuadsRendered;
610
+ }
611
+
612
+ renderToTexture(node: CoreNode) {
613
+ for (let i = 0; i < this.rttNodes.length; i++) {
614
+ if (this.rttNodes[i] === node) {
615
+ return;
616
+ }
617
+ }
618
+
619
+ this.insertRTTNodeInOrder(node);
620
+ }
621
+
622
+ /**
623
+ * Inserts an RTT node into `this.rttNodes` while maintaining the correct rendering order based on hierarchy.
624
+ *
625
+ * Rendering order for RTT nodes is critical when nested RTT nodes exist in a parent-child relationship.
626
+ * Specifically:
627
+ * - Child RTT nodes must be rendered before their RTT-enabled parents to ensure proper texture composition.
628
+ * - If an RTT node is added and it has existing RTT children, it should be rendered after those children.
629
+ *
630
+ * This function addresses both cases by:
631
+ * 1. **Checking Upwards**: It traverses the node's hierarchy upwards to identify any RTT parent
632
+ * already in `rttNodes`. If an RTT parent is found, the new node is placed before this parent.
633
+ * 2. **Checking Downwards**: It traverses the node’s children recursively to find any RTT-enabled
634
+ * children that are already in `rttNodes`. If such children are found, the new node is inserted
635
+ * after the last (highest index) RTT child node.
636
+ *
637
+ * The final calculated insertion index ensures the new node is positioned in `rttNodes` to respect
638
+ * both parent-before-child and child-before-parent rendering rules, preserving the correct order
639
+ * for the WebGL renderer.
640
+ *
641
+ * @param node - The RTT-enabled CoreNode to be added to `rttNodes` in the appropriate hierarchical position.
642
+ */
643
+ private insertRTTNodeInOrder(node: CoreNode) {
644
+ let insertIndex = this.rttNodes.length; // Default to the end of the array
645
+
646
+ // 1. Traverse upwards to ensure the node is placed before its RTT parent (if any).
647
+ let currentNode: CoreNode = node;
648
+ while (currentNode) {
649
+ if (!currentNode.parent) {
650
+ break;
651
+ }
652
+
653
+ const parentIndex = this.rttNodes.indexOf(currentNode.parent);
654
+ if (parentIndex !== -1) {
655
+ // Found an RTT parent in the list; set insertIndex to place node before the parent
656
+ insertIndex = parentIndex;
657
+ break;
658
+ }
659
+
660
+ currentNode = currentNode.parent;
661
+ }
662
+
663
+ // 2. Traverse downwards to ensure the node is placed after any RTT children.
664
+ // Look through each child recursively to see if any are already in rttNodes.
665
+ const maxChildIndex = this.findMaxChildRTTIndex(node);
666
+ if (maxChildIndex !== -1) {
667
+ // Adjust insertIndex to be after the last child RTT node
668
+ insertIndex = Math.max(insertIndex, maxChildIndex + 1);
669
+ }
670
+
671
+ // 3. Insert the node at the calculated position
672
+ this.rttNodes.splice(insertIndex, 0, node);
673
+ }
674
+
675
+ // Helper function to find the highest index of any RTT children of a node within rttNodes
676
+ private findMaxChildRTTIndex(node: CoreNode): number {
677
+ let maxIndex = -1;
678
+
679
+ const traverseChildren = (currentNode: CoreNode) => {
680
+ const currentIndex = this.rttNodes.indexOf(currentNode);
681
+ if (currentIndex !== -1) {
682
+ maxIndex = Math.max(maxIndex, currentIndex);
683
+ }
684
+
685
+ // Recursively check all children of the current node
686
+ for (const child of currentNode.children) {
687
+ traverseChildren(child);
688
+ }
689
+ };
690
+
691
+ // Start traversal directly with the provided node
692
+ traverseChildren(node);
693
+
694
+ return maxIndex;
695
+ }
696
+
697
+ renderRTTNodes() {
698
+ const { glw } = this;
699
+ const { txManager } = this.stage;
700
+
701
+ // Render all associated RTT nodes to their textures
702
+ for (let i = 0; i < this.rttNodes.length; i++) {
703
+ const node = this.rttNodes[i];
704
+
705
+ // Skip nodes that don't have RTT updates
706
+ if (node === undefined || node.hasRTTupdates === false) {
707
+ continue;
708
+ }
709
+
710
+ // Skip nodes that are not visible
711
+ if (
712
+ node.worldAlpha === 0 ||
713
+ (node.strictBounds === true &&
714
+ node.renderState === CoreNodeRenderState.OutOfBounds)
715
+ ) {
716
+ continue;
717
+ }
718
+
719
+ // Skip nodes that do not have a loaded texture
720
+ if (node.texture === null || node.texture.state !== 'loaded') {
721
+ continue;
722
+ }
723
+
724
+ // Set the active RTT node to the current node
725
+ // So we can prevent rendering children of nested RTT nodes
726
+ this.activeRttNode = node;
727
+
728
+ assertTruthy(node.texture, 'RTT node missing texture');
729
+ const ctxTexture = node.texture.ctxTexture;
730
+ assertTruthy(ctxTexture instanceof WebGlCoreCtxRenderTexture);
731
+ this.renderToTextureActive = true;
732
+
733
+ // Bind the the texture's framebuffer
734
+ glw.bindFramebuffer(ctxTexture.framebuffer);
735
+
736
+ glw.viewport(0, 0, ctxTexture.w, ctxTexture.h);
737
+ // Set the clear color to transparent
738
+ glw.clearColor(0, 0, 0, 0);
739
+ glw.clear();
740
+
741
+ // Render all associated quads to the texture
742
+ for (let i = 0; i < node.children.length; i++) {
743
+ const child = node.children[i];
744
+
745
+ if (child === undefined) {
746
+ continue;
747
+ }
748
+
749
+ this.stage.addQuads(child);
750
+ child.hasRTTupdates = false;
751
+ }
752
+
753
+ // Render all associated quads to the texture
754
+ this.render();
755
+
756
+ // Reset render operations
757
+ this.renderOps.length = 0;
758
+ node.hasRTTupdates = false;
759
+ }
760
+
761
+ const clearColor = this.clearColor.normalized;
762
+ // Restore the default clear color
763
+ glw.clearColor(clearColor[0], clearColor[1], clearColor[2], clearColor[3]);
764
+
765
+ // Bind the default framebuffer
766
+ glw.bindFramebuffer(null);
767
+
768
+ glw.viewport(0, 0, this.glw.canvas.width, this.glw.canvas.height);
769
+ this.renderToTextureActive = false;
770
+ }
771
+
772
+ removeRTTNode(node: CoreNode) {
773
+ const index = this.rttNodes.indexOf(node);
774
+ if (index === -1) {
775
+ return;
776
+ }
777
+ this.rttNodes.splice(index, 1);
778
+ }
779
+
780
+ getBufferInfo(): BufferInfo | null {
781
+ const bufferInfo: BufferInfo = {
782
+ totalAvailable: this.stage.options.quadBufferSize,
783
+ totalUsed: this.quadBufferUsage,
784
+ };
785
+ return bufferInfo;
786
+ }
787
+
788
+ override getDefShaderCtr(): BaseShaderController {
789
+ return this.defShaderCtrl;
790
+ }
791
+
792
+ /**
793
+ * Updates the WebGL context's clear color and clears the color buffer.
794
+ *
795
+ * @param color - The color to set as the clear color, represented as a 32-bit integer.
796
+ */
797
+ updateClearColor(color: number) {
798
+ if (this.clearColor.raw === color) {
799
+ return;
800
+ }
801
+ const glw = this.glw;
802
+ const normalizedColor = getNormalizedRgbaComponents(color);
803
+ glw.clearColor(
804
+ normalizedColor[0],
805
+ normalizedColor[1],
806
+ normalizedColor[2],
807
+ normalizedColor[3],
808
+ );
809
+ this.clearColor = {
810
+ raw: color,
811
+ normalized: normalizedColor,
812
+ };
813
+ glw.clear();
814
+ }
815
+ }