import { BufferLoader, clone, CompressedTextureBuffer, GltfBufferViewLoader, GltfImageLoader, FeatureDetection, Resource, ResourceCache, when, } from "../../Source/Cesium.js"; import createContext from "../createContext.js"; import dataUriToBuffer from "../dataUriToBuffer.js"; import pollToPromise from "../pollToPromise.js"; describe("Scene/GltfImageLoader", function () { var image = new Image(); image.src = ""; var pngBuffer = dataUriToBuffer( "" ); var jpgBuffer = dataUriToBuffer( "" ); var webpBuffer = dataUriToBuffer( "" ); var gifBuffer = dataUriToBuffer( "" ); var ktx2BasisBuffer; var ktx2BasisMipmapBuffer; var gltfUri = "https://example.com/model.glb"; var gltfResource = new Resource({ url: gltfUri, }); var gltf = { buffers: [ { uri: "external.bin", byteLength: 0, // updated in getGltf }, ], bufferViews: [ { buffer: 0, byteOffset: 0, byteLength: 0, // updated in getGltf }, ], images: [ { mimeType: "image/png", bufferView: 0, }, { uri: "image.png", }, { mimeType: "image/ktx2", bufferView: 0, }, { uri: "image.ktx2", }, ], }; var context; function getGltf(imageBuffer) { var clonedGltf = clone(gltf, true); clonedGltf.buffers[0].byteLength = imageBuffer.byteLength; clonedGltf.bufferViews[0].byteLength = imageBuffer.byteLength; return clonedGltf; } beforeAll(function () { context = createContext(); var ktx2BasisBufferPromise = Resource.fetchArrayBuffer({ url: "./Data/Images/Green4x4_ETC1S.ktx2", }).then(function (arrayBuffer) { ktx2BasisBuffer = new Uint8Array(arrayBuffer); }); var ktx2BasisMipmapBufferPromise = Resource.fetchArrayBuffer({ url: "./Data/Images/Green4x4Mipmap_ETC1S.ktx2", }).then(function (arrayBuffer) { ktx2BasisMipmapBuffer = new Uint8Array(arrayBuffer); }); return when.all([ktx2BasisBufferPromise, ktx2BasisMipmapBufferPromise]); }); afterAll(function () { context.destroyForSpecs(); }); afterEach(function () { ResourceCache.clearForSpecs(); }); it("throws if resourceCache is undefined", function () { expect(function () { return new GltfImageLoader({ resourceCache: undefined, gltf: gltf, imageId: 0, gltfResource: gltfResource, baseResource: gltfResource, }); }).toThrowDeveloperError(); }); it("throws if gltf is undefined", function () { expect(function () { return new GltfImageLoader({ resourceCache: ResourceCache, gltf: undefined, imageId: 0, gltfResource: gltfResource, baseResource: gltfResource, }); }).toThrowDeveloperError(); }); it("throws if imageId is undefined", function () { expect(function () { return new GltfImageLoader({ resourceCache: ResourceCache, gltf: gltf, imageId: undefined, gltfResource: gltfResource, baseResource: gltfResource, }); }).toThrowDeveloperError(); }); it("throws if gltfResource is undefined", function () { expect(function () { return new GltfImageLoader({ resourceCache: ResourceCache, gltf: gltf, imageId: 0, gltfResource: undefined, baseResource: gltfResource, }); }).toThrowDeveloperError(); }); it("throws if baseResource is undefined", function () { expect(function () { return new GltfImageLoader({ resourceCache: ResourceCache, gltf: gltf, imageId: 0, gltfResource: gltfResource, baseResource: undefined, }); }).toThrowDeveloperError(); }); it("rejects promise if buffer view fails to load", function () { var error = new Error("404 Not Found"); spyOn(Resource.prototype, "fetchArrayBuffer").and.returnValue( when.reject(error) ); var imageLoader = new GltfImageLoader({ resourceCache: ResourceCache, gltf: getGltf(pngBuffer), imageId: 0, gltfResource: gltfResource, baseResource: gltfResource, }); imageLoader.load(); return imageLoader.promise .then(function (imageLoader) { fail(); }) .otherwise(function (runtimeError) { expect(runtimeError.message).toBe( "Failed to load embedded image\nFailed to load buffer view\nFailed to load external buffer: https://example.com/external.bin\n404 Not Found" ); }); }); it("rejects promise if image format is not recognized", function () { spyOn(Resource.prototype, "fetchArrayBuffer").and.returnValue( when.resolve(gifBuffer) ); var imageLoader = new GltfImageLoader({ resourceCache: ResourceCache, gltf: getGltf(gifBuffer), imageId: 0, gltfResource: gltfResource, baseResource: gltfResource, }); imageLoader.load(); return imageLoader.promise .then(function (imageLoader) { fail(); }) .otherwise(function (runtimeError) { expect(runtimeError.message).toBe( "Failed to load embedded image\nImage format is not recognized" ); }); }); it("rejects promise if uri fails to load", function () { var error = new Error("404 Not Found"); spyOn(Resource.prototype, "fetchImage").and.returnValue(when.reject(error)); var imageLoader = new GltfImageLoader({ resourceCache: ResourceCache, gltf: getGltf(pngBuffer), imageId: 1, gltfResource: gltfResource, baseResource: gltfResource, }); imageLoader.load(); return imageLoader.promise .then(function (imageLoader) { fail(); }) .otherwise(function (runtimeError) { expect(runtimeError.message).toBe( "Failed to load image: image.png\n404 Not Found" ); }); }); function loadsFromBufferView(imageBuffer) { spyOn(Resource.prototype, "fetchArrayBuffer").and.returnValue( when.resolve(imageBuffer) ); var imageLoader = new GltfImageLoader({ resourceCache: ResourceCache, gltf: getGltf(imageBuffer), imageId: 0, gltfResource: gltfResource, baseResource: gltfResource, }); imageLoader.load(); return imageLoader.promise.then(function (imageLoader) { expect(imageLoader.image.width).toBe(1); expect(imageLoader.image.height).toBe(1); }); } it("loads PNG from buffer view", function () { return loadsFromBufferView(pngBuffer); }); it("loads JPEG from buffer view", function () { return loadsFromBufferView(jpgBuffer); }); it("loads WebP from buffer view", function () { return pollToPromise(function () { FeatureDetection.supportsWebP.initialize(); return FeatureDetection.supportsWebP.initialized; }).then(function () { if (!FeatureDetection.supportsWebP()) { return; } return loadsFromBufferView(webpBuffer); }); }); it("loads KTX2/Basis from buffer view", function () { if (!context.supportsBasis) { return; } spyOn(BufferLoader, "_fetchArrayBuffer").and.returnValue( when.resolve(ktx2BasisBuffer) ); var imageLoader = new GltfImageLoader({ resourceCache: ResourceCache, gltf: getGltf(ktx2BasisBuffer), imageId: 2, gltfResource: gltfResource, baseResource: gltfResource, }); imageLoader.load(); return imageLoader.promise.then(function (imageLoader) { expect(imageLoader.image instanceof CompressedTextureBuffer).toBe(true); expect(imageLoader.image.width).toBe(4); expect(imageLoader.image.height).toBe(4); expect(imageLoader.mipLevels).toBeUndefined(); }); }); it("loads KTX2/Basis with mipmap from buffer view", function () { if (!context.supportsBasis) { return; } spyOn(BufferLoader, "_fetchArrayBuffer").and.returnValue( when.resolve(ktx2BasisMipmapBuffer) ); var imageLoader = new GltfImageLoader({ resourceCache: ResourceCache, gltf: getGltf(ktx2BasisMipmapBuffer), imageId: 2, gltfResource: gltfResource, baseResource: gltfResource, }); imageLoader.load(); return imageLoader.promise.then(function (imageLoader) { expect(imageLoader.image instanceof CompressedTextureBuffer).toBe(true); expect(imageLoader.image.width).toBe(4); expect(imageLoader.image.height).toBe(4); expect(imageLoader.mipLevels.length).toBe(2); }); }); it("loads from uri", function () { spyOn(Resource.prototype, "fetchImage").and.returnValue( when.resolve(image) ); var imageLoader = new GltfImageLoader({ resourceCache: ResourceCache, gltf: clone(gltf, true), imageId: 1, gltfResource: gltfResource, baseResource: gltfResource, }); imageLoader.load(); return imageLoader.promise.then(function (imageLoader) { expect(imageLoader.image.width).toBe(1); expect(imageLoader.image.height).toBe(1); }); }); it("loads KTX2/Basis from uri ", function () { if (!context.supportsBasis) { return; } var baseResource = new Resource({ url: "./Data/Images/", }); var clonedGltf = clone(gltf, true); clonedGltf.images[3].uri = "Green4x4_ETC1S.ktx2"; var imageLoader = new GltfImageLoader({ resourceCache: ResourceCache, gltf: clonedGltf, imageId: 3, gltfResource: gltfResource, baseResource: baseResource, }); imageLoader.load(); return imageLoader.promise.then(function (imageLoader) { expect(imageLoader.image instanceof CompressedTextureBuffer).toBe(true); expect(imageLoader.image.width).toBe(4); expect(imageLoader.image.height).toBe(4); expect(imageLoader.mipLevels).toBeUndefined(); }); }); it("destroys image loader", function () { spyOn(Resource.prototype, "fetchArrayBuffer").and.returnValue( when.resolve(pngBuffer) ); var unloadBufferView = spyOn( GltfBufferViewLoader.prototype, "unload" ).and.callThrough(); var imageLoader = new GltfImageLoader({ resourceCache: ResourceCache, gltf: getGltf(pngBuffer), imageId: 0, gltfResource: gltfResource, baseResource: gltfResource, }); expect(imageLoader.image).not.toBeDefined(); imageLoader.load(); return imageLoader.promise.then(function (imageLoader) { expect(imageLoader.image).toBeDefined(); expect(imageLoader.isDestroyed()).toBe(false); imageLoader.destroy(); expect(imageLoader.image).not.toBeDefined(); expect(imageLoader.isDestroyed()).toBe(true); expect(unloadBufferView).toHaveBeenCalled(); }); }); function resolveBufferViewAfterDestroy(reject) { var deferredPromise = when.defer(); spyOn(Resource.prototype, "fetchArrayBuffer").and.returnValue( deferredPromise.promise ); // Load a copy of the buffer view into the cache so that the buffer view // promise resolves even if the image loader is destroyed var bufferViewLoaderCopy = ResourceCache.loadBufferView({ gltf: getGltf(pngBuffer), bufferViewId: 0, gltfResource: gltfResource, baseResource: gltfResource, }); var imageLoader = new GltfImageLoader({ resourceCache: ResourceCache, gltf: getGltf(pngBuffer), imageId: 0, gltfResource: gltfResource, baseResource: gltfResource, }); expect(imageLoader.image).not.toBeDefined(); imageLoader.load(); imageLoader.destroy(); if (reject) { deferredPromise.reject(new Error()); } else { deferredPromise.resolve(pngBuffer); } expect(imageLoader.image).not.toBeDefined(); expect(imageLoader.isDestroyed()).toBe(true); ResourceCache.unload(bufferViewLoaderCopy); } it("handles resolving buffer view after destroy", function () { resolveBufferViewAfterDestroy(false); }); it("handles rejecting buffer view after destroy", function () { resolveBufferViewAfterDestroy(true); }); function resolveImageFromTypedArrayAfterDestroy(reject) { spyOn(Resource.prototype, "fetchArrayBuffer").and.returnValue( when.resolve(pngBuffer) ); var deferredPromise = when.defer(); spyOn(GltfImageLoader, "_loadImageFromTypedArray").and.returnValue( deferredPromise.promise ); // Load a copy of the buffer view into the cache so that the buffer view // promise resolves even if the image loader is destroyed var bufferViewLoaderCopy = ResourceCache.loadBufferView({ gltf: getGltf(pngBuffer), bufferViewId: 0, gltfResource: gltfResource, baseResource: gltfResource, }); var imageLoader = new GltfImageLoader({ resourceCache: ResourceCache, gltf: getGltf(pngBuffer), imageId: 0, gltfResource: gltfResource, baseResource: gltfResource, }); expect(imageLoader.image).not.toBeDefined(); imageLoader.load(); imageLoader.destroy(); if (reject) { deferredPromise.reject(new Error()); } else { deferredPromise.resolve(image); } expect(imageLoader.image).not.toBeDefined(); expect(imageLoader.isDestroyed()).toBe(true); ResourceCache.unload(bufferViewLoaderCopy); } it("handles resolving image from typed array after destroy", function () { resolveImageFromTypedArrayAfterDestroy(false); }); it("handles rejecting image from typed array after destroy", function () { resolveImageFromTypedArrayAfterDestroy(true); }); function resolveUriAfterDestroy(reject) { var deferredPromise = when.defer(); spyOn(Resource.prototype, "fetchImage").and.returnValue( deferredPromise.promise ); var imageLoader = new GltfImageLoader({ resourceCache: ResourceCache, gltf: getGltf(pngBuffer), imageId: 1, gltfResource: gltfResource, baseResource: gltfResource, }); expect(imageLoader.image).not.toBeDefined(); imageLoader.load(); imageLoader.destroy(); if (reject) { deferredPromise.reject(new Error()); } else { deferredPromise.resolve(image); } expect(imageLoader.image).not.toBeDefined(); expect(imageLoader.isDestroyed()).toBe(true); } it("handles resolving uri after destroy", function () { resolveUriAfterDestroy(false); }); it("handles rejecting uri after destroy", function () { resolveUriAfterDestroy(true); }); });