texture.ts 21 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574
  1. /**
  2. * Copyright (c) 2018-2022 mol* contributors, licensed under MIT, See LICENSE file for more info.
  3. *
  4. * @author Alexander Rose <alexander.rose@weirdbyte.de>
  5. */
  6. import { WebGLContext } from './context';
  7. import { TextureImage, TextureVolume } from '../renderable/util';
  8. import { ValueCell } from '../../mol-util';
  9. import { RenderableSchema } from '../renderable/schema';
  10. import { idFactory } from '../../mol-util/id-factory';
  11. import { Framebuffer } from './framebuffer';
  12. import { isWebGL2, GLRenderingContext } from './compat';
  13. import { isPromiseLike, ValueOf } from '../../mol-util/type-helpers';
  14. import { WebGLExtensions } from './extensions';
  15. import { objectForEach } from '../../mol-util/object';
  16. const getNextTextureId = idFactory();
  17. export type TextureKindValue = {
  18. 'image-uint8': TextureImage<Uint8Array>
  19. 'image-float32': TextureImage<Float32Array>
  20. 'image-float16': TextureImage<Float32Array>
  21. 'image-int32': TextureImage<Int32Array>
  22. 'image-depth': TextureImage<Uint8Array> // TODO should be Uint32Array
  23. 'volume-uint8': TextureVolume<Uint8Array>
  24. 'volume-float32': TextureVolume<Float32Array>
  25. 'volume-float16': TextureVolume<Float32Array>
  26. 'texture': Texture
  27. }
  28. export type TextureValueType = ValueOf<TextureKindValue>
  29. export type TextureKind = keyof TextureKindValue
  30. export type TextureType = 'ubyte' | 'ushort' | 'float' | 'fp16' | 'int'
  31. export type TextureFormat = 'alpha' | 'rgb' | 'rgba' | 'depth'
  32. /** Numbers are shortcuts for color attachment */
  33. export type TextureAttachment = 'depth' | 'stencil' | 'color0' | 'color1' | 'color2' | 'color3' | 'color4' | 'color5' | 'color6' | 'color7' | 0 | 1 | 2 | 3 | 4 | 5 | 6 | 7
  34. export type TextureFilter = 'nearest' | 'linear'
  35. export function getTarget(gl: GLRenderingContext, kind: TextureKind): number {
  36. switch (kind) {
  37. case 'image-uint8': return gl.TEXTURE_2D;
  38. case 'image-float32': return gl.TEXTURE_2D;
  39. case 'image-float16': return gl.TEXTURE_2D;
  40. case 'image-depth': return gl.TEXTURE_2D;
  41. }
  42. if (isWebGL2(gl)) {
  43. switch (kind) {
  44. case 'image-int32': return gl.TEXTURE_2D;
  45. case 'volume-uint8': return gl.TEXTURE_3D;
  46. case 'volume-float32': return gl.TEXTURE_3D;
  47. case 'volume-float16': return gl.TEXTURE_3D;
  48. }
  49. }
  50. throw new Error(`unknown texture kind '${kind}'`);
  51. }
  52. export function getFormat(gl: GLRenderingContext, format: TextureFormat, type: TextureType): number {
  53. switch (format) {
  54. case 'alpha':
  55. if (isWebGL2(gl) && type === 'float') return gl.RED;
  56. else if (isWebGL2(gl) && type === 'int') return gl.RED_INTEGER;
  57. else return gl.ALPHA;
  58. case 'rgb':
  59. if (isWebGL2(gl) && type === 'int') return gl.RGB_INTEGER;
  60. return gl.RGB;
  61. case 'rgba':
  62. if (isWebGL2(gl) && type === 'int') return gl.RGBA_INTEGER;
  63. return gl.RGBA;
  64. case 'depth': return gl.DEPTH_COMPONENT;
  65. }
  66. }
  67. export function getInternalFormat(gl: GLRenderingContext, format: TextureFormat, type: TextureType): number {
  68. if (isWebGL2(gl)) {
  69. switch (format) {
  70. case 'alpha':
  71. switch (type) {
  72. case 'ubyte': return gl.ALPHA;
  73. case 'float': return gl.R32F;
  74. case 'fp16': return gl.R16F;
  75. case 'int': return gl.R32I;
  76. }
  77. case 'rgb':
  78. switch (type) {
  79. case 'ubyte': return gl.RGB;
  80. case 'float': return gl.RGB32F;
  81. case 'fp16': return gl.RGB16F;
  82. case 'int': return gl.RGB32I;
  83. }
  84. case 'rgba':
  85. switch (type) {
  86. case 'ubyte': return gl.RGBA;
  87. case 'float': return gl.RGBA32F;
  88. case 'fp16': return gl.RGBA16F;
  89. case 'int': return gl.RGBA32I;
  90. }
  91. case 'depth':
  92. switch (type) {
  93. case 'ushort': return gl.DEPTH_COMPONENT16;
  94. case 'float': return gl.DEPTH_COMPONENT32F;
  95. }
  96. }
  97. }
  98. return getFormat(gl, format, type);
  99. }
  100. function getByteCount(format: TextureFormat, type: TextureType, width: number, height: number, depth: number): number {
  101. const bpe = getFormatSize(format) * getTypeSize(type);
  102. return bpe * width * height * (depth || 1);
  103. }
  104. function getFormatSize(format: TextureFormat) {
  105. switch (format) {
  106. case 'alpha': return 1;
  107. case 'rgb': return 3;
  108. case 'rgba': return 4;
  109. case 'depth': return 4;
  110. }
  111. }
  112. function getTypeSize(type: TextureType): number {
  113. switch (type) {
  114. case 'ubyte': return 1;
  115. case 'ushort': return 2;
  116. case 'float': return 4;
  117. case 'fp16': return 2;
  118. case 'int': return 4;
  119. }
  120. }
  121. export function getType(gl: GLRenderingContext, extensions: WebGLExtensions, type: TextureType): number {
  122. switch (type) {
  123. case 'ubyte': return gl.UNSIGNED_BYTE;
  124. case 'ushort': return gl.UNSIGNED_SHORT;
  125. case 'float': return gl.FLOAT;
  126. case 'fp16':
  127. if (extensions.textureHalfFloat) return extensions.textureHalfFloat.HALF_FLOAT;
  128. else throw new Error('extension "texture_half_float" unavailable');
  129. case 'int':
  130. if (isWebGL2(gl)) return gl.INT;
  131. else throw new Error('texture type "int" requires webgl2');
  132. }
  133. }
  134. export function getFilter(gl: GLRenderingContext, type: TextureFilter): number {
  135. switch (type) {
  136. case 'nearest': return gl.NEAREST;
  137. case 'linear': return gl.LINEAR;
  138. }
  139. }
  140. export function getAttachment(gl: GLRenderingContext, extensions: WebGLExtensions, attachment: TextureAttachment): number {
  141. switch (attachment) {
  142. case 'depth': return gl.DEPTH_ATTACHMENT;
  143. case 'stencil': return gl.STENCIL_ATTACHMENT;
  144. case 'color0': case 0: return gl.COLOR_ATTACHMENT0;
  145. }
  146. if (extensions.drawBuffers) {
  147. switch (attachment) {
  148. case 'color1': case 1: return extensions.drawBuffers.COLOR_ATTACHMENT1;
  149. case 'color2': case 2: return extensions.drawBuffers.COLOR_ATTACHMENT2;
  150. case 'color3': case 3: return extensions.drawBuffers.COLOR_ATTACHMENT3;
  151. case 'color4': case 4: return extensions.drawBuffers.COLOR_ATTACHMENT4;
  152. case 'color5': case 5: return extensions.drawBuffers.COLOR_ATTACHMENT5;
  153. case 'color6': case 6: return extensions.drawBuffers.COLOR_ATTACHMENT6;
  154. case 'color7': case 7: return extensions.drawBuffers.COLOR_ATTACHMENT7;
  155. }
  156. }
  157. throw new Error('unknown texture attachment');
  158. }
  159. function isImage(x: TextureImage<any> | TextureVolume<any> | HTMLImageElement): x is HTMLImageElement {
  160. return typeof HTMLImageElement !== 'undefined' && (x instanceof HTMLImageElement);
  161. }
  162. function isTexture2d(x: TextureImage<any> | TextureVolume<any>, target: number, gl: GLRenderingContext): x is TextureImage<any> {
  163. return target === gl.TEXTURE_2D;
  164. }
  165. function isTexture3d(x: TextureImage<any> | TextureVolume<any>, target: number, gl: WebGL2RenderingContext): x is TextureVolume<any> {
  166. return target === gl.TEXTURE_3D;
  167. }
  168. export interface Texture {
  169. readonly id: number
  170. readonly target: number
  171. readonly format: number
  172. readonly internalFormat: number
  173. readonly type: number
  174. readonly filter: number
  175. getWidth: () => number
  176. getHeight: () => number
  177. getDepth: () => number
  178. getByteCount: () => number
  179. define: (width: number, height: number, depth?: number) => void
  180. /**
  181. * The `sub` option requires an existing allocation on the GPU, that is, either
  182. * `define` or `load` without `sub` must have been called before.
  183. */
  184. load: (image: TextureImage<any> | TextureVolume<any> | HTMLImageElement, sub?: boolean) => void
  185. bind: (id: TextureId) => void
  186. unbind: (id: TextureId) => void
  187. /** Use `layer` to attach a z-slice of a 3D texture */
  188. attachFramebuffer: (framebuffer: Framebuffer, attachment: TextureAttachment, layer?: number) => void
  189. detachFramebuffer: (framebuffer: Framebuffer, attachment: TextureAttachment) => void
  190. reset: () => void
  191. destroy: () => void
  192. }
  193. export type TextureId = 0 | 1 | 2 | 3 | 4 | 5 | 6 | 7 | 8 | 9 | 10 | 11 | 12 | 13 | 14 | 15
  194. export type TextureValues = { [k: string]: ValueCell<TextureValueType> }
  195. export type Textures = [string, Texture][]
  196. function getTexture(gl: GLRenderingContext) {
  197. const texture = gl.createTexture();
  198. if (texture === null) {
  199. throw new Error('Could not create WebGL texture');
  200. }
  201. return texture;
  202. }
  203. export function createTexture(gl: GLRenderingContext, extensions: WebGLExtensions, kind: TextureKind, _format: TextureFormat, _type: TextureType, _filter: TextureFilter): Texture {
  204. const id = getNextTextureId();
  205. let texture = getTexture(gl);
  206. // check texture kind and type compatability
  207. if (
  208. (kind.endsWith('float32') && _type !== 'float') ||
  209. (kind.endsWith('float16') && _type !== 'fp16') ||
  210. (kind.endsWith('uint8') && _type !== 'ubyte') ||
  211. (kind.endsWith('int32') && _type !== 'int') ||
  212. (kind.endsWith('depth') && _type !== 'ushort' && _type !== 'float')
  213. ) {
  214. throw new Error(`texture kind '${kind}' and type '${_type}' are incompatible`);
  215. }
  216. if (!extensions.depthTexture && _format === 'depth') {
  217. throw new Error(`extension 'WEBGL_depth_texture' needed for 'depth' texture format`);
  218. }
  219. const target = getTarget(gl, kind);
  220. const filter = getFilter(gl, _filter);
  221. const format = getFormat(gl, _format, _type);
  222. const internalFormat = getInternalFormat(gl, _format, _type);
  223. const type = getType(gl, extensions, _type);
  224. function init() {
  225. gl.bindTexture(target, texture);
  226. gl.texParameteri(target, gl.TEXTURE_MAG_FILTER, filter);
  227. gl.texParameteri(target, gl.TEXTURE_MIN_FILTER, filter);
  228. // clamp-to-edge needed for non-power-of-two textures in webgl
  229. gl.texParameteri(target, gl.TEXTURE_WRAP_S, gl.CLAMP_TO_EDGE);
  230. gl.texParameteri(target, gl.TEXTURE_WRAP_T, gl.CLAMP_TO_EDGE);
  231. gl.bindTexture(target, null);
  232. }
  233. init();
  234. let width = 0, height = 0, depth = 0;
  235. let loadedData: undefined | TextureImage<any> | TextureVolume<any> | HTMLImageElement;
  236. let destroyed = false;
  237. function define(_width: number, _height: number, _depth?: number) {
  238. if (_width === 0 || _height === 0 || (isWebGL2(gl) && target === gl.TEXTURE_3D && _depth === 0)) {
  239. throw new Error('empty textures are not allowed');
  240. }
  241. if (width === _width && height === _height && depth === (_depth || 0)) return;
  242. width = _width, height = _height, depth = _depth || 0;
  243. gl.bindTexture(target, texture);
  244. if (target === gl.TEXTURE_2D) {
  245. gl.texImage2D(target, 0, internalFormat, width, height, 0, format, type, null);
  246. } else if (isWebGL2(gl) && target === gl.TEXTURE_3D && depth !== undefined) {
  247. gl.texImage3D(target, 0, internalFormat, width, height, depth, 0, format, type, null);
  248. } else {
  249. throw new Error('unknown texture target');
  250. }
  251. }
  252. define(1, 1, isWebGL2(gl) && target === gl.TEXTURE_3D ? 1 : 0);
  253. function load(data: TextureImage<any> | TextureVolume<any> | HTMLImageElement, sub = false) {
  254. if (data.width === 0 || data.height === 0 || (!isImage(data) && isWebGL2(gl) && isTexture3d(data, target, gl) && data.depth === 0)) {
  255. throw new Error('empty textures are not allowed');
  256. }
  257. gl.bindTexture(target, texture);
  258. // unpack alignment of 1 since we use textures only for data
  259. gl.pixelStorei(gl.UNPACK_ALIGNMENT, 1);
  260. gl.pixelStorei(gl.UNPACK_COLORSPACE_CONVERSION_WEBGL, gl.NONE);
  261. gl.pixelStorei(gl.UNPACK_PREMULTIPLY_ALPHA_WEBGL, 0);
  262. if (isImage(data)) {
  263. width = data.width, height = data.height;
  264. gl.pixelStorei(gl.UNPACK_FLIP_Y_WEBGL, false);
  265. gl.bindTexture(gl.TEXTURE_2D, texture);
  266. gl.texImage2D(gl.TEXTURE_2D, 0, internalFormat, format, type, data);
  267. } else if (isTexture2d(data, target, gl)) {
  268. const _filter = data.filter ? getFilter(gl, data.filter) : filter;
  269. gl.texParameteri(target, gl.TEXTURE_MAG_FILTER, _filter);
  270. gl.texParameteri(target, gl.TEXTURE_MIN_FILTER, _filter);
  271. gl.pixelStorei(gl.UNPACK_FLIP_Y_WEBGL, !!data.flipY);
  272. if (sub) {
  273. gl.texSubImage2D(target, 0, 0, 0, data.width, data.height, format, type, data.array);
  274. } else {
  275. width = data.width, height = data.height;
  276. gl.texImage2D(target, 0, internalFormat, width, height, 0, format, type, data.array);
  277. }
  278. } else if (isWebGL2(gl) && isTexture3d(data, target, gl)) {
  279. gl.pixelStorei(gl.UNPACK_FLIP_Y_WEBGL, false);
  280. if (sub) {
  281. gl.texSubImage3D(target, 0, 0, 0, 0, data.width, data.height, data.depth, format, type, data.array);
  282. } else {
  283. width = data.width, height = data.height, depth = data.depth;
  284. gl.texImage3D(target, 0, internalFormat, width, height, depth, 0, format, type, data.array);
  285. }
  286. } else {
  287. throw new Error('unknown texture target');
  288. }
  289. gl.bindTexture(target, null);
  290. loadedData = data;
  291. }
  292. function attachFramebuffer(framebuffer: Framebuffer, attachment: TextureAttachment, layer?: number) {
  293. framebuffer.bind();
  294. if (target === gl.TEXTURE_2D) {
  295. gl.framebufferTexture2D(gl.FRAMEBUFFER, getAttachment(gl, extensions, attachment), gl.TEXTURE_2D, texture, 0);
  296. } else if (isWebGL2(gl) && target === gl.TEXTURE_3D) {
  297. if (layer === undefined) throw new Error('need `layer` to attach 3D texture');
  298. gl.framebufferTextureLayer(gl.FRAMEBUFFER, getAttachment(gl, extensions, attachment), texture, 0, layer);
  299. } else {
  300. throw new Error('unknown/unsupported texture target');
  301. }
  302. }
  303. return {
  304. id,
  305. target,
  306. format,
  307. internalFormat,
  308. type,
  309. filter,
  310. getWidth: () => width,
  311. getHeight: () => height,
  312. getDepth: () => depth,
  313. getByteCount: () => getByteCount(_format, _type, width, height, depth),
  314. define,
  315. load,
  316. bind: (id: TextureId) => {
  317. gl.activeTexture(gl.TEXTURE0 + id);
  318. gl.bindTexture(target, texture);
  319. },
  320. unbind: (id: TextureId) => {
  321. gl.activeTexture(gl.TEXTURE0 + id);
  322. gl.bindTexture(target, null);
  323. },
  324. attachFramebuffer,
  325. detachFramebuffer: (framebuffer: Framebuffer, attachment: TextureAttachment) => {
  326. framebuffer.bind();
  327. if (target === gl.TEXTURE_2D) {
  328. gl.framebufferTexture2D(gl.FRAMEBUFFER, getAttachment(gl, extensions, attachment), gl.TEXTURE_2D, null, 0);
  329. } else if (isWebGL2(gl) && target === gl.TEXTURE_3D) {
  330. gl.framebufferTextureLayer(gl.FRAMEBUFFER, getAttachment(gl, extensions, attachment), null, 0, 0);
  331. } else {
  332. throw new Error('unknown texture target');
  333. }
  334. },
  335. reset: () => {
  336. texture = getTexture(gl);
  337. init();
  338. const [_width, _height, _depth] = [width, height, depth];
  339. width = 0, height = 0, depth = 0; // set to zero to trigger resize
  340. define(_width, _height, _depth);
  341. if (loadedData) load(loadedData);
  342. },
  343. destroy: () => {
  344. if (destroyed) return;
  345. gl.deleteTexture(texture);
  346. destroyed = true;
  347. }
  348. };
  349. }
  350. export function createTextures(ctx: WebGLContext, schema: RenderableSchema, values: TextureValues) {
  351. const { resources } = ctx;
  352. const textures: Textures = [];
  353. Object.keys(schema).forEach(k => {
  354. const spec = schema[k];
  355. if (spec.type === 'texture') {
  356. const value = values[k];
  357. if (value) {
  358. if (spec.kind === 'texture') {
  359. textures[textures.length] = [k, value.ref.value as Texture];
  360. } else {
  361. const texture = resources.texture(spec.kind, spec.format, spec.dataType, spec.filter);
  362. texture.load(value.ref.value as TextureImage<any> | TextureVolume<any>);
  363. textures[textures.length] = [k, texture];
  364. }
  365. }
  366. }
  367. });
  368. return textures;
  369. }
  370. /**
  371. * Loads an image from a url to a textures and triggers update asynchronously.
  372. * This will not work on node.js without a polyfill for `HTMLImageElement`.
  373. */
  374. export function loadImageTexture(src: string, cell: ValueCell<Texture>, texture: Texture) {
  375. const img = new Image();
  376. img.onload = function () {
  377. texture.load(img);
  378. ValueCell.update(cell, texture);
  379. };
  380. img.src = src;
  381. }
  382. //
  383. export type CubeSide = 'nx' | 'ny' | 'nz' | 'px' | 'py' | 'pz';
  384. export type CubeFaces = {
  385. [k in CubeSide]: string | File | Promise<Blob>;
  386. }
  387. export function getCubeTarget(gl: GLRenderingContext, side: CubeSide): number {
  388. switch (side) {
  389. case 'nx': return gl.TEXTURE_CUBE_MAP_NEGATIVE_X;
  390. case 'ny': return gl.TEXTURE_CUBE_MAP_NEGATIVE_Y;
  391. case 'nz': return gl.TEXTURE_CUBE_MAP_NEGATIVE_Z;
  392. case 'px': return gl.TEXTURE_CUBE_MAP_POSITIVE_X;
  393. case 'py': return gl.TEXTURE_CUBE_MAP_POSITIVE_Y;
  394. case 'pz': return gl.TEXTURE_CUBE_MAP_POSITIVE_Z;
  395. }
  396. }
  397. export function createCubeTexture(gl: GLRenderingContext, faces: CubeFaces, mipmaps: boolean, onload?: () => void): Texture {
  398. const target = gl.TEXTURE_CUBE_MAP;
  399. const filter = gl.LINEAR;
  400. const internalFormat = gl.RGBA;
  401. const format = gl.RGBA;
  402. const type = gl.UNSIGNED_BYTE;
  403. let size = 0;
  404. const texture = gl.createTexture();
  405. gl.bindTexture(target, texture);
  406. let loadedCount = 0;
  407. objectForEach(faces, (source, side) => {
  408. if (!source) return;
  409. const level = 0;
  410. const cubeTarget = getCubeTarget(gl, side as CubeSide);
  411. const image = new Image();
  412. if (source instanceof File) {
  413. image.src = URL.createObjectURL(source);
  414. } else if (isPromiseLike(source)) {
  415. source.then(blob => {
  416. image.src = URL.createObjectURL(blob);
  417. });
  418. } else {
  419. image.src = source;
  420. }
  421. image.addEventListener('load', () => {
  422. if (size === 0) size = image.width;
  423. gl.texImage2D(cubeTarget, level, internalFormat, size, size, 0, format, type, null);
  424. gl.pixelStorei(gl.UNPACK_ALIGNMENT, 4);
  425. gl.pixelStorei(gl.UNPACK_COLORSPACE_CONVERSION_WEBGL, gl.NONE);
  426. gl.pixelStorei(gl.UNPACK_PREMULTIPLY_ALPHA_WEBGL, 0);
  427. gl.pixelStorei(gl.UNPACK_FLIP_Y_WEBGL, false);
  428. gl.bindTexture(target, texture);
  429. gl.texImage2D(cubeTarget, level, internalFormat, format, type, image);
  430. loadedCount += 1;
  431. if (loadedCount === 6 && !destroyed) {
  432. if (mipmaps) {
  433. gl.generateMipmap(target);
  434. gl.texParameteri(target, gl.TEXTURE_MIN_FILTER, gl.LINEAR_MIPMAP_LINEAR);
  435. } else {
  436. gl.texParameteri(target, gl.TEXTURE_MIN_FILTER, filter);
  437. }
  438. gl.texParameteri(target, gl.TEXTURE_MAG_FILTER, filter);
  439. if (onload) onload();
  440. }
  441. });
  442. });
  443. let destroyed = false;
  444. return {
  445. id: getNextTextureId(),
  446. target,
  447. format,
  448. internalFormat,
  449. type,
  450. filter,
  451. getWidth: () => size,
  452. getHeight: () => size,
  453. getDepth: () => 0,
  454. getByteCount: () => {
  455. return getByteCount('rgba', 'ubyte', size, size, 0) * 6 * (mipmaps ? 2 : 1);
  456. },
  457. define: () => {},
  458. load: () => {},
  459. bind: (id: TextureId) => {
  460. gl.activeTexture(gl.TEXTURE0 + id);
  461. gl.bindTexture(target, texture);
  462. },
  463. unbind: (id: TextureId) => {
  464. gl.activeTexture(gl.TEXTURE0 + id);
  465. gl.bindTexture(target, null);
  466. },
  467. attachFramebuffer: () => {},
  468. detachFramebuffer: () => {},
  469. reset: () => {},
  470. destroy: () => {
  471. if (destroyed) return;
  472. gl.deleteTexture(texture);
  473. destroyed = true;
  474. },
  475. };
  476. }
  477. //
  478. export function createNullTexture(gl?: GLRenderingContext): Texture {
  479. const target = gl?.TEXTURE_2D ?? 3553;
  480. return {
  481. id: getNextTextureId(),
  482. target,
  483. format: 0,
  484. internalFormat: 0,
  485. type: 0,
  486. filter: 0,
  487. getWidth: () => 0,
  488. getHeight: () => 0,
  489. getDepth: () => 0,
  490. getByteCount: () => 0,
  491. define: () => {},
  492. load: () => {},
  493. bind: (id: TextureId) => {
  494. if (gl) {
  495. gl.activeTexture(gl.TEXTURE0 + id);
  496. gl.bindTexture(target, null);
  497. }
  498. },
  499. unbind: (id: TextureId) => {
  500. if (gl) {
  501. gl.activeTexture(gl.TEXTURE0 + id);
  502. gl.bindTexture(target, null);
  503. }
  504. },
  505. attachFramebuffer: () => {},
  506. detachFramebuffer: () => {},
  507. reset: () => {},
  508. destroy: () => {},
  509. };
  510. }