KHR_draco_mesh_compression.ts 4.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101
  1. /// <reference path="../../../../../dist/preview release/babylon.d.ts"/>
  2. module BABYLON.GLTF2.Extensions {
  3. const NAME = "KHR_draco_mesh_compression";
  4. interface IKHRDracoMeshCompression {
  5. bufferView: number;
  6. attributes: { [name: string]: number };
  7. }
  8. interface ILoaderBufferViewDraco extends _ILoaderBufferView {
  9. _dracoBabylonGeometry?: Promise<Geometry>;
  10. }
  11. /**
  12. * [Specification](https://github.com/KhronosGroup/glTF/tree/master/extensions/2.0/Khronos/KHR_draco_mesh_compression)
  13. */
  14. export class KHR_draco_mesh_compression extends GLTFLoaderExtension {
  15. public readonly name = NAME;
  16. private _dracoCompression: Nullable<DracoCompression> = null;
  17. constructor(loader: GLTFLoader) {
  18. super(loader);
  19. // Disable extension if decoder is not available.
  20. if (!DracoCompression.DecoderAvailable) {
  21. this.enabled = false;
  22. }
  23. }
  24. public dispose(): void {
  25. if (this._dracoCompression) {
  26. this._dracoCompression.dispose();
  27. }
  28. super.dispose();
  29. }
  30. protected _loadVertexDataAsync(context: string, primitive: _ILoaderMeshPrimitive, babylonMesh: Mesh): Nullable<Promise<Geometry>> {
  31. return this._loadExtensionAsync<IKHRDracoMeshCompression, Geometry>(context, primitive, (extensionContext, extension) => {
  32. if (primitive.mode != undefined) {
  33. if (primitive.mode !== MeshPrimitiveMode.TRIANGLE_STRIP &&
  34. primitive.mode !== MeshPrimitiveMode.TRIANGLES) {
  35. throw new Error(`${context}: Unsupported mode ${primitive.mode}`);
  36. }
  37. // TODO: handle triangle strips
  38. if (primitive.mode === MeshPrimitiveMode.TRIANGLE_STRIP) {
  39. throw new Error(`${context}: Mode ${primitive.mode} is not currently supported`);
  40. }
  41. }
  42. const attributes: { [kind: string]: number } = {};
  43. const loadAttribute = (name: string, kind: string) => {
  44. const uniqueId = extension.attributes[name];
  45. if (uniqueId == undefined) {
  46. return;
  47. }
  48. babylonMesh._delayInfo = babylonMesh._delayInfo || [];
  49. if (babylonMesh._delayInfo.indexOf(kind) === -1) {
  50. babylonMesh._delayInfo.push(kind);
  51. }
  52. attributes[kind] = uniqueId;
  53. };
  54. loadAttribute("POSITION", VertexBuffer.PositionKind);
  55. loadAttribute("NORMAL", VertexBuffer.NormalKind);
  56. loadAttribute("TANGENT", VertexBuffer.TangentKind);
  57. loadAttribute("TEXCOORD_0", VertexBuffer.UVKind);
  58. loadAttribute("TEXCOORD_1", VertexBuffer.UV2Kind);
  59. loadAttribute("JOINTS_0", VertexBuffer.MatricesIndicesKind);
  60. loadAttribute("WEIGHTS_0", VertexBuffer.MatricesWeightsKind);
  61. loadAttribute("COLOR_0", VertexBuffer.ColorKind);
  62. var bufferView = GLTFLoader._GetProperty(extensionContext, this._loader._gltf.bufferViews, extension.bufferView) as ILoaderBufferViewDraco;
  63. if (!bufferView._dracoBabylonGeometry) {
  64. bufferView._dracoBabylonGeometry = this._loader._loadBufferViewAsync(`#/bufferViews/${bufferView._index}`, bufferView).then(data => {
  65. if (!this._dracoCompression) {
  66. this._dracoCompression = new DracoCompression();
  67. }
  68. return this._dracoCompression.decodeMeshAsync(data, attributes).then(babylonVertexData => {
  69. const babylonGeometry = new Geometry(babylonMesh.name, this._loader._babylonScene);
  70. babylonVertexData.applyToGeometry(babylonGeometry);
  71. return babylonGeometry;
  72. }).catch(error => {
  73. throw new Error(`${context}: ${error.message}`);
  74. });
  75. });
  76. }
  77. return bufferView._dracoBabylonGeometry;
  78. });
  79. }
  80. }
  81. GLTFLoader._Register(NAME, loader => new KHR_draco_mesh_compression(loader));
  82. }