"use strict"; const { DeveloperError } = require("cesium"); const fsExtra = require("fs-extra"); const path = require("path"); const Promise = require("bluebird"); const createGltf = require("../../lib/createGltf"); const obj2gltf = require("../../lib/obj2gltf"); const texturedObjPath = "specs/data/box-textured/box-textured.obj"; const complexObjPath = "specs/data/box-complex-material/box-complex-material.obj"; const missingMtllibObjPath = "specs/data/box-missing-mtllib/box-missing-mtllib.obj"; const outputDirectory = "output"; const textureUrl = "specs/data/box-textured/cesium.png"; describe("obj2gltf", () => { beforeEach(() => { spyOn(fsExtra, "outputFile").and.returnValue(Promise.resolve()); }); it("converts obj to gltf", async () => { const gltf = await obj2gltf(texturedObjPath); expect(gltf).toBeDefined(); expect(gltf.images.length).toBe(1); }); it("converts obj to glb", async () => { const options = { binary: true, }; const glb = await obj2gltf(texturedObjPath, options); const magic = glb.toString("utf8", 0, 4); expect(magic).toBe("glTF"); }); it("convert obj to gltf with separate resources", async () => { const options = { separate: true, separateTextures: true, outputDirectory: outputDirectory, }; await obj2gltf(texturedObjPath, options); expect(fsExtra.outputFile.calls.count()).toBe(2); // Saves out .png and .bin }); it("convert obj to gltf with separate resources when buffer exceeds Node limit", async () => { spyOn(createGltf, "_getBufferMaxByteLength").and.returnValue(0); const options = { separate: true, separateTextures: true, outputDirectory: outputDirectory, }; await obj2gltf(texturedObjPath, options); expect(fsExtra.outputFile.calls.count()).toBe(5); // Saves out .png and four .bin for positions, normals, uvs, and indices }); it("converts obj to glb with separate resources", async () => { const options = { separate: true, separateTextures: true, outputDirectory: outputDirectory, binary: true, }; await obj2gltf(texturedObjPath, options); expect(fsExtra.outputFile.calls.count()).toBe(2); // Saves out .png and .bin }); it("converts obj with multiple textures", async () => { const options = { separateTextures: true, outputDirectory: outputDirectory, }; await obj2gltf(complexObjPath, options); expect(fsExtra.outputFile.calls.count()).toBe(5); // baseColor, metallicRoughness, occlusion, emission, normal }); it("sets overriding textures (1)", async () => { const options = { overridingTextures: { metallicRoughnessOcclusionTexture: textureUrl, normalTexture: textureUrl, baseColorTexture: textureUrl, emissiveTexture: textureUrl, alphaTexture: textureUrl, }, separateTextures: true, outputDirectory: outputDirectory, }; await obj2gltf(complexObjPath, options); const args = fsExtra.outputFile.calls.allArgs(); const length = args.length; for (let i = 0; i < length; ++i) { expect(path.basename(args[i][0])).toBe(path.basename(textureUrl)); } }); it("sets overriding textures (2)", async () => { const options = { overridingTextures: { specularGlossinessTexture: textureUrl, occlusionTexture: textureUrl, normalTexture: textureUrl, baseColorTexture: textureUrl, emissiveTexture: textureUrl, alphaTexture: textureUrl, }, separateTextures: true, outputDirectory: outputDirectory, }; await obj2gltf(complexObjPath, options); const args = fsExtra.outputFile.calls.allArgs(); const length = args.length; for (let i = 0; i < length; ++i) { expect(path.basename(args[i][0])).toBe(path.basename(textureUrl)); } }); it("uses a custom logger", async () => { let lastMessage; const options = { logger: (message) => { lastMessage = message; }, }; await obj2gltf(missingMtllibObjPath, options); expect(lastMessage.indexOf("Could not read material file") >= 0).toBe(true); }); it("uses a custom writer", async () => { const filePaths = []; const fileContents = []; const options = { separate: true, writer: (relativePath, contents) => { filePaths.push(relativePath); fileContents.push(contents); }, }; await obj2gltf(texturedObjPath, options); expect(filePaths).toEqual(["cesium.png", "box-textured.bin"]); expect(fileContents[0]).toBeDefined(); expect(fileContents[1]).toBeDefined(); }); it("throws if objPath is undefined", () => { let thrownError; try { obj2gltf(undefined); } catch (e) { thrownError = e; } expect(thrownError).toEqual(new DeveloperError("objPath is required")); }); it("throws if both options.writer and options.outputDirectory are undefined when writing separate resources", () => { const options = { separateTextures: true, }; let thrownError; try { obj2gltf(texturedObjPath, options); } catch (e) { thrownError = e; } expect(thrownError).toEqual( new DeveloperError( "Either options.writer or options.outputDirectory must be defined when writing separate resources.", ), ); }); it("throws if more than one material type is set", () => { const options = { metallicRoughness: true, specularGlossiness: true, }; let thrownError; try { obj2gltf(texturedObjPath, options); } catch (e) { thrownError = e; } expect(thrownError).toEqual( new DeveloperError( "Only one material type may be set from [metallicRoughness, specularGlossiness, unlit].", ), ); }); it("throws if metallicRoughnessOcclusionTexture and specularGlossinessTexture are both defined", () => { const options = { overridingTextures: { metallicRoughnessOcclusionTexture: textureUrl, specularGlossinessTexture: textureUrl, }, }; let thrownError; try { obj2gltf(texturedObjPath, options); } catch (e) { thrownError = e; } expect(thrownError).toEqual( new DeveloperError( "metallicRoughnessOcclusionTexture and specularGlossinessTexture cannot both be defined.", ), ); }); });