├── .appveyor.yml
├── .eslintignore
├── .eslintrc.json
├── .gitattributes
├── .github
└── workflows
│ └── build.yml
├── .gitignore
├── .husky
└── pre-commit
├── .npmignore
├── .npmrc
├── .prettierignore
├── CHANGES.md
├── LICENSE.md
├── README.md
├── ThirdParty.json
├── bin
└── gltf-pipeline.js
├── doc
└── gltf.png
├── gulpfile.js
├── index.js
├── lib
├── FileUrl.js
├── ForEach.js
├── addBuffer.js
├── addDefaults.js
├── addExtensionsRequired.js
├── addExtensionsUsed.js
├── addPipelineExtras.js
├── addToArray.js
├── compressDracoMeshes.js
├── dataUriToBuffer.js
├── findAccessorMinMax.js
├── forEachTextureInMaterial.js
├── getAccessorByteStride.js
├── getBufferPadded.js
├── getComponentReader.js
├── getImageExtension.js
├── getJsonBufferPadded.js
├── getStatistics.js
├── glbToGltf.js
├── gltfToGlb.js
├── mergeBuffers.js
├── moveTechniqueRenderStates.js
├── moveTechniquesToExtension.js
├── numberOfComponentsForType.js
├── parseGlb.js
├── processGlb.js
├── processGltf.js
├── readAccessorPacked.js
├── readResources.js
├── removeDefaults.js
├── removeExtension.js
├── removeExtensionsRequired.js
├── removeExtensionsUsed.js
├── removePipelineExtras.js
├── removeUnusedElements.js
├── replaceWithDecompressedPrimitive.js
├── splitPrimitives.js
├── updateAccessorComponentTypes.js
├── updateVersion.js
├── usesExtension.js
└── writeResources.js
├── package.json
└── specs
├── .eslintrc.json
├── data
├── 1.0
│ ├── box-materials-common
│ │ ├── box-materials-common.bin
│ │ └── box-materials-common.gltf
│ ├── box-textured-binary-separate
│ │ ├── box-textured-binary-separate-fs.glsl
│ │ ├── box-textured-binary-separate-vs.glsl
│ │ ├── box-textured-binary-separate.glb
│ │ └── cesium.png
│ ├── box-textured-binary
│ │ └── box-textured-binary.glb
│ ├── box-textured-embedded
│ │ └── box-textured-embedded.gltf
│ ├── box-textured-materials-common
│ │ ├── CesiumLogoFlat.png
│ │ ├── box-textured-materials-common.bin
│ │ └── box-textured-materials-common.gltf
│ ├── box-textured-separate
│ │ ├── box-textured-separate-fs.glsl
│ │ ├── box-textured-separate-vs.glsl
│ │ ├── box-textured-separate.bin
│ │ ├── box-textured-separate.gltf
│ │ └── cesium.png
│ └── box
│ │ └── box.gltf
└── 2.0
│ ├── box-morph
│ └── box-morph.gltf
│ ├── box-shared-image-references-separate
│ ├── CesiumTexturedBoxTest0FS.glsl
│ ├── CesiumTexturedBoxTest0VS.glsl
│ ├── Image.png
│ ├── box-shared-image-references-separate.bin
│ └── box-shared-image-references-separate.gltf
│ ├── box-shared-image-references
│ └── box-shared-image-references.gltf
│ ├── box-techniques-embedded
│ └── box-techniques-embedded.gltf
│ ├── box-techniques-separate
│ ├── CesiumTexturedBoxTest.bin
│ ├── CesiumTexturedBoxTest0FS.glsl
│ ├── CesiumTexturedBoxTest0VS.glsl
│ ├── Image0001.png
│ └── box-techniques-separate.gltf
│ ├── box-textured-binary-separate
│ ├── box-textured-binary-separate.glb
│ └── cesium.png
│ ├── box-textured-binary
│ └── box-textured-binary.glb
│ ├── box-textured-embedded
│ └── box-textured-embedded.gltf
│ ├── box-textured-separate
│ ├── box-textured-separate.bin
│ ├── box-textured-separate.gltf
│ └── cesium logo.png
│ ├── extensions
│ ├── EXT_meshopt_compression
│ │ ├── meshopt-fallback
│ │ │ ├── meshopt-fallback.bin
│ │ │ ├── meshopt-fallback.fallback.bin
│ │ │ └── meshopt-fallback.gltf
│ │ └── meshopt-no-fallback
│ │ │ ├── meshopt-no-fallback.bin
│ │ │ └── meshopt-no-fallback.gltf
│ └── EXT_texture_webp
│ │ ├── box-textured-embedded
│ │ └── box-textured-embedded.gltf
│ │ └── box-textured-separate
│ │ ├── box-textured-separate.bin
│ │ ├── box-textured-separate.gltf
│ │ ├── box-textured-with-fallback.gltf
│ │ ├── cesium logo.png
│ │ └── cesium logo.webp
│ ├── multiple-boxes
│ └── multiple-boxes.gltf
│ └── triangle-without-indices
│ └── triangle-without-indices.gltf
├── jasmine.json
└── lib
├── ForEachSpec.js
├── addBufferSpec.js
├── addDefaultsSpec.js
├── addExtensionsRequiredSpec.js
├── addExtensionsUsedSpec.js
├── addPipelineExtrasSpec.js
├── addToArraySpec.js
├── compressDracoMeshesSpec.js
├── dataUriToBufferSpec.js
├── findAccessorMinMaxSpec.js
├── getAccessorByteStrideSpec.js
├── getBufferPaddedSpec.js
├── getComponentReaderSpec.js
├── getImageExtensionSpec.js
├── getJsonBufferPaddedSpec.js
├── getStatisticsSpec.js
├── glbToGltfSpec.js
├── gltfToGlbSpec.js
├── mergeBuffersSpec.js
├── moveTechniqueRenderStatesSpec.js
├── moveTechniquesToExtensionSpec.js
├── numberOfComponentsForTypeSpec.js
├── parseGlbSpec.js
├── processGlbSpec.js
├── processGltfSpec.js
├── readAccessorPackedSpec.js
├── readResourcesSpec.js
├── removeDefaultsSpec.js
├── removeExtensionSpec.js
├── removeExtensionsRequiredSpec.js
├── removeExtensionsUsedSpec.js
├── removePipelineExtrasSpec.js
├── removeUnusedElementsSpec.js
├── splitPrimitivesSpec.js
├── updateAccessorComponentTypeSpec.js
├── updateVersionSpec.js
├── usesExtensionSpec.js
└── writeResourcesSpec.js
/.appveyor.yml:
--------------------------------------------------------------------------------
1 | # Builds are whole numbered
2 | version: '{build}'
3 |
4 | clone_depth: 50
5 |
6 | environment:
7 | matrix:
8 | - nodejs_version: "12"
9 |
10 | install:
11 | # Get the latest stable version of Node.js
12 | - ps: Install-Product node $env:nodejs_version
13 | # install modules
14 | - npm install
15 |
16 | test_script:
17 | - node --version
18 | - npm --version
19 | - npm run eslint
20 | - npm run test -- --failTaskOnError --suppressPassed
21 |
22 | # Don't actually build.
23 | build: off
24 |
--------------------------------------------------------------------------------
/.eslintignore:
--------------------------------------------------------------------------------
1 | node_modules/**
2 | coverage/**
3 | doc/**
4 | dist/**
5 |
--------------------------------------------------------------------------------
/.eslintrc.json:
--------------------------------------------------------------------------------
1 | {
2 | "extends": ["cesium/node"],
3 | "rules": {
4 | "no-unused-vars": ["error", {"args": "none"}]
5 | }
6 | }
7 |
--------------------------------------------------------------------------------
/.gitattributes:
--------------------------------------------------------------------------------
1 | # Auto detect text files and perform LF normalization
2 | * text=auto
3 | package.json text eol=lf
--------------------------------------------------------------------------------
/.github/workflows/build.yml:
--------------------------------------------------------------------------------
1 | run-name: build
2 | on:
3 | push:
4 | concurrency:
5 | group: ${{ github.ref }}
6 | cancel-in-progress: true
7 | jobs:
8 | build:
9 | runs-on: ubuntu-latest
10 | strategy:
11 | matrix:
12 | node: [ 16, 20 ]
13 | name: Node ${{ matrix.node }}
14 | steps:
15 | - uses: actions/checkout@v3
16 | - uses: actions/setup-node@v3
17 | with:
18 | node-version: ${{ matrix.node }}
19 | - run: npm install
20 | - run: npm run eslint
21 | - run: npm run prettier-check
22 | - run: npm run test -- --failTaskOnError --suppressPassed
23 |
--------------------------------------------------------------------------------
/.gitignore:
--------------------------------------------------------------------------------
1 | # NPM
2 | node_modules
3 | npm-debug.log
4 | package-lock.json
5 | yarn.lock
6 |
7 | # WebStorm user-specific
8 | .idea/workspace.xml
9 | .idea/tasks.xml
10 |
11 | # VSCode user-specific
12 | .vscode
13 |
14 | # Generate data
15 | .eslintcache
16 | coverage
17 | dist
18 | output
19 | test
20 | *.tgz
21 | .nyc_output
22 | doc/*
23 | !doc/gltf.png
24 |
--------------------------------------------------------------------------------
/.husky/pre-commit:
--------------------------------------------------------------------------------
1 | #!/usr/bin/env sh
2 | . "$(dirname -- "$0")/_/husky.sh"
3 |
4 | npm run pre-commit
5 |
--------------------------------------------------------------------------------
/.npmignore:
--------------------------------------------------------------------------------
1 | /.husky
2 | /.vscode
3 | /.idea
4 | /dist
5 | /coverage
6 | /doc/*
7 | /output
8 | /specs
9 | /test
10 | .editorconfig
11 | .eslintcache
12 | .eslintignore
13 | .eslintrc.json
14 | .gitattributes
15 | .nyc_output
16 | .npmignore
17 | .prettierignore
18 | .travis.yml
19 | .appveyor.yml
20 | gulpfile.js
21 | *.tgz
22 | !doc/gltf.png
23 |
--------------------------------------------------------------------------------
/.npmrc:
--------------------------------------------------------------------------------
1 | package-lock=false
2 |
--------------------------------------------------------------------------------
/.prettierignore:
--------------------------------------------------------------------------------
1 | # Ignore everything
2 | *
3 |
4 | # Unignore directories (to all depths) and unignore code in these directories
5 | !bin/**/
6 | !doc/**/
7 | !lib/**/
8 | !specs/**/
9 |
10 | !**/*.js
11 | !**/*.md
12 |
--------------------------------------------------------------------------------
/ThirdParty.json:
--------------------------------------------------------------------------------
1 | [
2 | {
3 | "name": "bluebird",
4 | "license": [
5 | "MIT"
6 | ],
7 | "version": "3.7.2",
8 | "url": "https://www.npmjs.com/package/bluebird"
9 | },
10 | {
11 | "name": "cesium",
12 | "license": [
13 | "Apache-2.0"
14 | ],
15 | "version": "1.109.0",
16 | "url": "https://www.npmjs.com/package/cesium"
17 | },
18 | {
19 | "name": "draco3d",
20 | "license": [
21 | "Apache-2.0"
22 | ],
23 | "version": "1.5.6",
24 | "url": "https://www.npmjs.com/package/draco3d"
25 | },
26 | {
27 | "name": "fs-extra",
28 | "license": [
29 | "MIT"
30 | ],
31 | "version": "11.1.1",
32 | "url": "https://www.npmjs.com/package/fs-extra"
33 | },
34 | {
35 | "name": "mime",
36 | "license": [
37 | "MIT"
38 | ],
39 | "version": "3.0.0",
40 | "url": "https://www.npmjs.com/package/mime"
41 | },
42 | {
43 | "name": "object-hash",
44 | "license": [
45 | "MIT"
46 | ],
47 | "version": "3.0.0",
48 | "url": "https://www.npmjs.com/package/object-hash"
49 | },
50 | {
51 | "name": "yargs",
52 | "license": [
53 | "MIT"
54 | ],
55 | "version": "17.7.2",
56 | "url": "https://www.npmjs.com/package/yargs"
57 | }
58 | ]
--------------------------------------------------------------------------------
/doc/gltf.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/CesiumGS/gltf-pipeline/cbbff2f212b5aa6163c7152d46ffcaf3a2ff3ac3/doc/gltf.png
--------------------------------------------------------------------------------
/index.js:
--------------------------------------------------------------------------------
1 | /*eslint-disable n/global-require*/
2 | "use strict";
3 | module.exports = {
4 | getStatistics: require("./lib/getStatistics"),
5 | glbToGltf: require("./lib/glbToGltf"),
6 | gltfToGlb: require("./lib/gltfToGlb"),
7 | processGlb: require("./lib/processGlb"),
8 | processGltf: require("./lib/processGltf"),
9 | removeExtension: require("./lib/removeExtension"),
10 | };
11 |
--------------------------------------------------------------------------------
/lib/FileUrl.js:
--------------------------------------------------------------------------------
1 | "use strict";
2 |
3 | const { Check, RuntimeError } = require("cesium");
4 | const os = require("os");
5 | const path = require("path");
6 | const { domainToUnicode, URL } = require("url");
7 |
8 | module.exports = {
9 | fileURLToPath: fileURLToPath,
10 | pathToFileURL: pathToFileURL,
11 | };
12 |
13 | const isWindows = os.platform() === "win32";
14 | const forwardSlashRegEx = /\//g;
15 |
16 | const CHAR_LOWERCASE_A = 97;
17 | const CHAR_LOWERCASE_Z = 122;
18 | const CHAR_FORWARD_SLASH = 47;
19 | const CHAR_BACKWARD_SLASH = 92;
20 |
21 | const percentRegEx = /%/g;
22 | const backslashRegEx = /\\/g;
23 | const newlineRegEx = /\n/g;
24 | const carriageReturnRegEx = /\r/g;
25 | const tabRegEx = /\t/g;
26 |
27 | // The following function is copied from Node.js implementation of url module
28 | // https://github.com/nodejs/node/blob/7237eaa3353aacf284289c8b59b0a5e0fa5744bb/lib/internal/url.js#L1345-L1383
29 | // pathToFileURL & fileURLToPath were added in 10.12 so we want to maintain ability run under older versions.
30 | function fileURLToPath(path) {
31 | Check.defined("path", path);
32 |
33 | if (typeof path === "string") {
34 | path = new URL(path);
35 | }
36 |
37 | if (path.protocol !== "file:") {
38 | throw new RuntimeError("Expected path.protocol to start with file:");
39 | }
40 | return isWindows ? getPathFromURLWin32(path) : getPathFromURLPosix(path);
41 | }
42 |
43 | function pathToFileURL(filepath) {
44 | let resolved = path.resolve(filepath);
45 | // path.resolve strips trailing slashes so we must add them back
46 | const filePathLast = filepath.charCodeAt(filepath.length - 1);
47 | if (
48 | (filePathLast === CHAR_FORWARD_SLASH ||
49 | (isWindows && filePathLast === CHAR_BACKWARD_SLASH)) &&
50 | resolved[resolved.length - 1] !== path.sep
51 | ) {
52 | resolved += "/";
53 | }
54 | const outURL = new URL("file://");
55 | if (resolved.includes("%")) {
56 | resolved = resolved.replace(percentRegEx, "%25");
57 | }
58 | // in posix, "/" is a valid character in paths
59 | if (!isWindows && resolved.includes("\\")) {
60 | resolved = resolved.replace(backslashRegEx, "%5C");
61 | }
62 | if (resolved.includes("\n")) {
63 | resolved = resolved.replace(newlineRegEx, "%0A");
64 | }
65 | if (resolved.includes("\r")) {
66 | resolved = resolved.replace(carriageReturnRegEx, "%0D");
67 | }
68 | if (resolved.includes("\t")) {
69 | resolved = resolved.replace(tabRegEx, "%09");
70 | }
71 | outURL.pathname = resolved;
72 | return outURL;
73 | }
74 |
75 | function getPathFromURLWin32(url) {
76 | const hostname = url.hostname;
77 | let pathname = url.pathname;
78 | for (let n = 0; n < pathname.length; n++) {
79 | if (pathname[n] === "%") {
80 | const third = pathname.codePointAt(n + 2) | 0x20;
81 | if (
82 | (pathname[n + 1] === "2" && third === 102) || // 2f 2F /
83 | (pathname[n + 1] === "5" && third === 99)
84 | ) {
85 | // 5c 5C \
86 | throw new RuntimeError(
87 | "file URL must not include encoded \\ or / characters",
88 | );
89 | }
90 | }
91 | }
92 | pathname = pathname.replace(forwardSlashRegEx, "\\");
93 | pathname = decodeURIComponent(pathname);
94 | if (hostname !== "") {
95 | // If hostname is set, then we have a UNC path
96 | // Pass the hostname through domainToUnicode just in case
97 | // it is an IDN using punycode encoding. We do not need to worry
98 | // about percent encoding because the URL parser will have
99 | // already taken care of that for us. Note that this only
100 | // causes IDNs with an appropriate `xn--` prefix to be decoded.
101 | return `\\\\${domainToUnicode(hostname)}${pathname}`;
102 | }
103 |
104 | // Otherwise, it's a local path that requires a drive letter
105 | const letter = pathname.codePointAt(1) | 0x20;
106 | const sep = pathname[2];
107 | if (
108 | letter < CHAR_LOWERCASE_A ||
109 | letter > CHAR_LOWERCASE_Z || // a..z A..Z
110 | sep !== ":"
111 | ) {
112 | throw new RuntimeError("file URL must be absolute");
113 | }
114 | return pathname.slice(1);
115 | }
116 |
117 | function getPathFromURLPosix(url) {
118 | if (url.hostname !== "") {
119 | throw new RuntimeError("Invalid platform");
120 | }
121 | const pathname = url.pathname;
122 | for (let n = 0; n < pathname.length; n++) {
123 | if (pathname[n] === "%") {
124 | const third = pathname.codePointAt(n + 2) | 0x20;
125 | if (pathname[n + 1] === "2" && third === 102) {
126 | throw new RuntimeError(
127 | "file URL must not include encoded \\ or / characters",
128 | );
129 | }
130 | }
131 | }
132 | return decodeURIComponent(pathname);
133 | }
134 |
--------------------------------------------------------------------------------
/lib/addBuffer.js:
--------------------------------------------------------------------------------
1 | "use strict";
2 | const addToArray = require("./addToArray");
3 |
4 | module.exports = addBuffer;
5 |
6 | /**
7 | * Adds buffer to gltf.
8 | *
9 | * @param {object} gltf A javascript object containing a glTF asset.
10 | * @param {Buffer} buffer A Buffer object which will be added to gltf.buffers.
11 | * @returns {number} The bufferView id of the newly added bufferView.
12 | *
13 | * @private
14 | */
15 | function addBuffer(gltf, buffer) {
16 | const newBuffer = {
17 | byteLength: buffer.length,
18 | extras: {
19 | _pipeline: {
20 | source: buffer,
21 | },
22 | },
23 | };
24 | const bufferId = addToArray(gltf.buffers, newBuffer);
25 | const bufferView = {
26 | buffer: bufferId,
27 | byteOffset: 0,
28 | byteLength: buffer.length,
29 | };
30 | return addToArray(gltf.bufferViews, bufferView);
31 | }
32 |
--------------------------------------------------------------------------------
/lib/addExtensionsRequired.js:
--------------------------------------------------------------------------------
1 | "use strict";
2 | const Cesium = require("cesium");
3 | const addExtensionsUsed = require("./addExtensionsUsed");
4 | const addToArray = require("./addToArray");
5 |
6 | const defined = Cesium.defined;
7 |
8 | module.exports = addExtensionsRequired;
9 |
10 | /**
11 | * Adds an extension to gltf.extensionsRequired if it does not already exist.
12 | * Initializes extensionsRequired if it is not defined.
13 | *
14 | * @param {object} gltf A javascript object containing a glTF asset.
15 | * @param {string} extension The extension to add.
16 | *
17 | * @private
18 | */
19 | function addExtensionsRequired(gltf, extension) {
20 | let extensionsRequired = gltf.extensionsRequired;
21 | if (!defined(extensionsRequired)) {
22 | extensionsRequired = [];
23 | gltf.extensionsRequired = extensionsRequired;
24 | }
25 | addToArray(extensionsRequired, extension, true);
26 | addExtensionsUsed(gltf, extension);
27 | }
28 |
--------------------------------------------------------------------------------
/lib/addExtensionsUsed.js:
--------------------------------------------------------------------------------
1 | "use strict";
2 | const Cesium = require("cesium");
3 | const addToArray = require("./addToArray");
4 |
5 | const defined = Cesium.defined;
6 |
7 | module.exports = addExtensionsUsed;
8 |
9 | /**
10 | * Adds an extension to gltf.extensionsUsed if it does not already exist.
11 | * Initializes extensionsUsed if it is not defined.
12 | *
13 | * @param {object} gltf A javascript object containing a glTF asset.
14 | * @param {string} extension The extension to add.
15 | *
16 | * @private
17 | */
18 | function addExtensionsUsed(gltf, extension) {
19 | let extensionsUsed = gltf.extensionsUsed;
20 | if (!defined(extensionsUsed)) {
21 | extensionsUsed = [];
22 | gltf.extensionsUsed = extensionsUsed;
23 | }
24 | addToArray(extensionsUsed, extension, true);
25 | }
26 |
--------------------------------------------------------------------------------
/lib/addPipelineExtras.js:
--------------------------------------------------------------------------------
1 | "use strict";
2 | const Cesium = require("cesium");
3 | const ForEach = require("./ForEach");
4 |
5 | const defined = Cesium.defined;
6 |
7 | module.exports = addPipelineExtras;
8 |
9 | /**
10 | * Adds extras._pipeline to each object that can have extras in the glTF asset.
11 | * This stage runs before updateVersion and handles both glTF 1.0 and glTF 2.0 assets.
12 | *
13 | * @param {object} gltf A javascript object containing a glTF asset.
14 | * @returns {object} The glTF asset with the added pipeline extras.
15 | *
16 | * @private
17 | */
18 | function addPipelineExtras(gltf) {
19 | ForEach.shader(gltf, function (shader) {
20 | addExtras(shader);
21 | });
22 | ForEach.buffer(gltf, function (buffer) {
23 | addExtras(buffer);
24 | });
25 | ForEach.image(gltf, function (image) {
26 | addExtras(image);
27 | });
28 |
29 | addExtras(gltf);
30 |
31 | return gltf;
32 | }
33 |
34 | function addExtras(object) {
35 | object.extras = defined(object.extras) ? object.extras : {};
36 | object.extras._pipeline = defined(object.extras._pipeline)
37 | ? object.extras._pipeline
38 | : {};
39 | }
40 |
--------------------------------------------------------------------------------
/lib/addToArray.js:
--------------------------------------------------------------------------------
1 | "use strict";
2 | const Cesium = require("cesium");
3 |
4 | const defaultValue = Cesium.defaultValue;
5 |
6 | module.exports = addToArray;
7 |
8 | /**
9 | * Adds an element to an array and returns the element's index.
10 | *
11 | * @param {Array} array The array to add to.
12 | * @param {object} element The element to add.
13 | * @param {boolean} [checkDuplicates=false] When true
, if a duplicate element is found its index is returned and element
is not added to the array.
14 | *
15 | * @private
16 | */
17 | function addToArray(array, element, checkDuplicates) {
18 | checkDuplicates = defaultValue(checkDuplicates, false);
19 | if (checkDuplicates) {
20 | const index = array.indexOf(element);
21 | if (index > -1) {
22 | return index;
23 | }
24 | }
25 |
26 | array.push(element);
27 | return array.length - 1;
28 | }
29 |
--------------------------------------------------------------------------------
/lib/dataUriToBuffer.js:
--------------------------------------------------------------------------------
1 | "use strict";
2 | module.exports = dataUriToBuffer;
3 |
4 | /**
5 | * Read a data uri string into a buffer.
6 | *
7 | * @param {string} dataUri The data uri.
8 | * @returns {Buffer}
9 | *
10 | * @private
11 | */
12 | function dataUriToBuffer(dataUri) {
13 | const data = dataUri.slice(dataUri.indexOf(",") + 1);
14 | if (dataUri.indexOf("base64") >= 0) {
15 | return Buffer.from(data, "base64");
16 | }
17 | return Buffer.from(data, "utf8");
18 | }
19 |
--------------------------------------------------------------------------------
/lib/findAccessorMinMax.js:
--------------------------------------------------------------------------------
1 | "use strict";
2 | const Cesium = require("cesium");
3 |
4 | const ComponentDatatype = Cesium.ComponentDatatype;
5 | const defined = Cesium.defined;
6 |
7 | const getAccessorByteStride = require("./getAccessorByteStride");
8 | const getComponentReader = require("./getComponentReader");
9 | const numberOfComponentsForType = require("./numberOfComponentsForType");
10 |
11 | module.exports = findAccessorMinMax;
12 |
13 | /**
14 | * Finds the min and max values of the accessor.
15 | *
16 | * @param {object} gltf A javascript object containing a glTF asset.
17 | * @param {object} accessor The accessor object from the glTF asset to read.
18 | * @returns {{min: Array, max: Array}} min holding the array of minimum values and max holding the array of maximum values.
19 | *
20 | * @private
21 | */
22 | function findAccessorMinMax(gltf, accessor) {
23 | const bufferViews = gltf.bufferViews;
24 | const buffers = gltf.buffers;
25 | const bufferViewId = accessor.bufferView;
26 | const numberOfComponents = numberOfComponentsForType(accessor.type);
27 |
28 | // According to the spec, when bufferView is not defined, accessor must be initialized with zeros
29 | if (!defined(accessor.bufferView)) {
30 | return {
31 | min: new Array(numberOfComponents).fill(0.0),
32 | max: new Array(numberOfComponents).fill(0.0),
33 | };
34 | }
35 |
36 | const min = new Array(numberOfComponents).fill(Number.POSITIVE_INFINITY);
37 | const max = new Array(numberOfComponents).fill(Number.NEGATIVE_INFINITY);
38 |
39 | const bufferView = bufferViews[bufferViewId];
40 | const bufferId = bufferView.buffer;
41 | const buffer = buffers[bufferId];
42 | const source = buffer.extras._pipeline.source;
43 |
44 | const count = accessor.count;
45 | const byteStride = getAccessorByteStride(gltf, accessor);
46 | let byteOffset =
47 | accessor.byteOffset + bufferView.byteOffset + source.byteOffset;
48 | const componentType = accessor.componentType;
49 | const componentTypeByteLength =
50 | ComponentDatatype.getSizeInBytes(componentType);
51 | const dataView = new DataView(source.buffer);
52 | const components = new Array(numberOfComponents);
53 | const componentReader = getComponentReader(componentType);
54 |
55 | for (let i = 0; i < count; i++) {
56 | componentReader(
57 | dataView,
58 | byteOffset,
59 | numberOfComponents,
60 | componentTypeByteLength,
61 | components,
62 | );
63 | for (let j = 0; j < numberOfComponents; j++) {
64 | const value = components[j];
65 | min[j] = Math.min(min[j], value);
66 | max[j] = Math.max(max[j], value);
67 | }
68 | byteOffset += byteStride;
69 | }
70 |
71 | return {
72 | min: min,
73 | max: max,
74 | };
75 | }
76 |
--------------------------------------------------------------------------------
/lib/forEachTextureInMaterial.js:
--------------------------------------------------------------------------------
1 | "use strict";
2 | const Cesium = require("cesium");
3 |
4 | const ForEach = require("./ForEach");
5 |
6 | const Check = Cesium.Check;
7 | const defined = Cesium.defined;
8 |
9 | module.exports = forEachTextureInMaterial;
10 |
11 | /**
12 | * Calls the provider handler function on each texture used by the material.
13 | * Mimics the behavior of functions in gltf-pipeline ForEach.
14 | * @param {object} material The glTF material.
15 | * @param {forEachTextureInMaterial~handler} handler Function that is called for each texture in the material.
16 | *
17 | * @private
18 | */
19 | function forEachTextureInMaterial(material, handler) {
20 | Check.typeOf.object("material", material);
21 | Check.defined("handler", handler);
22 |
23 | // Metallic roughness
24 | const pbrMetallicRoughness = material.pbrMetallicRoughness;
25 | if (defined(pbrMetallicRoughness)) {
26 | if (defined(pbrMetallicRoughness.baseColorTexture)) {
27 | const textureInfo = pbrMetallicRoughness.baseColorTexture;
28 | const value = handler(textureInfo.index, textureInfo);
29 | if (defined(value)) {
30 | return value;
31 | }
32 | }
33 | if (defined(pbrMetallicRoughness.metallicRoughnessTexture)) {
34 | const textureInfo = pbrMetallicRoughness.metallicRoughnessTexture;
35 | const value = handler(textureInfo.index, textureInfo);
36 | if (defined(value)) {
37 | return value;
38 | }
39 | }
40 | }
41 |
42 | if (defined(material.extensions)) {
43 | // Spec gloss extension
44 | const pbrSpecularGlossiness =
45 | material.extensions.KHR_materials_pbrSpecularGlossiness;
46 | if (defined(pbrSpecularGlossiness)) {
47 | if (defined(pbrSpecularGlossiness.diffuseTexture)) {
48 | const textureInfo = pbrSpecularGlossiness.diffuseTexture;
49 | const value = handler(textureInfo.index, textureInfo);
50 | if (defined(value)) {
51 | return value;
52 | }
53 | }
54 | if (defined(pbrSpecularGlossiness.specularGlossinessTexture)) {
55 | const textureInfo = pbrSpecularGlossiness.specularGlossinessTexture;
56 | const value = handler(textureInfo.index, textureInfo);
57 | if (defined(value)) {
58 | return value;
59 | }
60 | }
61 | }
62 |
63 | // Materials common extension (may be present in models converted from glTF 1.0)
64 | const materialsCommon = material.extensions.KHR_materials_common;
65 | if (defined(materialsCommon) && defined(materialsCommon.values)) {
66 | const diffuse = materialsCommon.values.diffuse;
67 | const ambient = materialsCommon.values.ambient;
68 | const emission = materialsCommon.values.emission;
69 | const specular = materialsCommon.values.specular;
70 | if (defined(diffuse) && defined(diffuse.index)) {
71 | const value = handler(diffuse.index, diffuse);
72 | if (defined(value)) {
73 | return value;
74 | }
75 | }
76 | if (defined(ambient) && defined(ambient.index)) {
77 | const value = handler(ambient.index, ambient);
78 | if (defined(value)) {
79 | return value;
80 | }
81 | }
82 | if (defined(emission) && defined(emission.index)) {
83 | const value = handler(emission.index, emission);
84 | if (defined(value)) {
85 | return value;
86 | }
87 | }
88 | if (defined(specular) && defined(specular.index)) {
89 | const value = handler(specular.index, specular);
90 | if (defined(value)) {
91 | return value;
92 | }
93 | }
94 | }
95 | }
96 |
97 | // KHR_techniques_webgl extension
98 | const value = ForEach.materialValue(material, function (materialValue) {
99 | if (defined(materialValue.index)) {
100 | const value = handler(materialValue.index, materialValue);
101 | if (defined(value)) {
102 | return value;
103 | }
104 | }
105 | });
106 | if (defined(value)) {
107 | return value;
108 | }
109 |
110 | // Top level textures
111 | if (defined(material.emissiveTexture)) {
112 | const textureInfo = material.emissiveTexture;
113 | const value = handler(textureInfo.index, textureInfo);
114 | if (defined(value)) {
115 | return value;
116 | }
117 | }
118 |
119 | if (defined(material.normalTexture)) {
120 | const textureInfo = material.normalTexture;
121 | const value = handler(textureInfo.index, textureInfo);
122 | if (defined(value)) {
123 | return value;
124 | }
125 | }
126 |
127 | if (defined(material.occlusionTexture)) {
128 | const textureInfo = material.occlusionTexture;
129 | const value = handler(textureInfo.index, textureInfo);
130 | if (defined(value)) {
131 | return value;
132 | }
133 | }
134 | }
135 |
136 | /**
137 | * Function that is called for each texture in the material. If this function returns a value the for each stops and returns that value.
138 | * @callback forEachTextureInMaterial~handler
139 | * @param {number} The texture index.
140 | * @param {object} The texture info object.
141 | *
142 | * @private
143 | */
144 |
--------------------------------------------------------------------------------
/lib/getAccessorByteStride.js:
--------------------------------------------------------------------------------
1 | "use strict";
2 | const Cesium = require("cesium");
3 | const numberOfComponentsForType = require("./numberOfComponentsForType");
4 |
5 | const ComponentDatatype = Cesium.ComponentDatatype;
6 | const defined = Cesium.defined;
7 |
8 | module.exports = getAccessorByteStride;
9 |
10 | /**
11 | * Returns the byte stride of the provided accessor.
12 | * If the byteStride is 0, it is calculated based on type and componentType
13 | *
14 | * @param {object} gltf A javascript object containing a glTF asset.
15 | * @param {object} accessor The accessor.
16 | * @returns {number} The byte stride of the accessor.
17 | *
18 | * @private
19 | */
20 | function getAccessorByteStride(gltf, accessor) {
21 | const bufferViewId = accessor.bufferView;
22 | if (defined(bufferViewId)) {
23 | const bufferView = gltf.bufferViews[bufferViewId];
24 | if (defined(bufferView.byteStride) && bufferView.byteStride > 0) {
25 | return bufferView.byteStride;
26 | }
27 | }
28 | return (
29 | ComponentDatatype.getSizeInBytes(accessor.componentType) *
30 | numberOfComponentsForType(accessor.type)
31 | );
32 | }
33 |
--------------------------------------------------------------------------------
/lib/getBufferPadded.js:
--------------------------------------------------------------------------------
1 | "use strict";
2 | module.exports = getBufferPadded;
3 |
4 | /**
5 | * Pad the buffer to the next 8-byte boundary to ensure proper alignment for the section that follows.
6 | * glTF only requires 4-byte alignment but some extensions like EXT_structural_metadata require 8-byte
7 | * alignment for some buffer views.
8 | *
9 | * @param {Buffer} buffer The buffer.
10 | * @returns {Buffer} The padded buffer.
11 | *
12 | * @private
13 | */
14 | function getBufferPadded(buffer) {
15 | const boundary = 8;
16 | const byteLength = buffer.length;
17 | const remainder = byteLength % boundary;
18 | if (remainder === 0) {
19 | return buffer;
20 | }
21 | const padding = remainder === 0 ? 0 : boundary - remainder;
22 | const emptyBuffer = Buffer.alloc(padding);
23 | return Buffer.concat([buffer, emptyBuffer]);
24 | }
25 |
--------------------------------------------------------------------------------
/lib/getComponentReader.js:
--------------------------------------------------------------------------------
1 | "use strict";
2 | const Cesium = require("cesium");
3 |
4 | const ComponentDatatype = Cesium.ComponentDatatype;
5 |
6 | module.exports = getComponentReader;
7 |
8 | /**
9 | * Returns a function to read and convert data from a DataView into an array.
10 | *
11 | * @param {number} componentType Type to convert the data to.
12 | * @returns {ComponentReader} Function that reads and converts data.
13 | *
14 | * @private
15 | */
16 | function getComponentReader(componentType) {
17 | switch (componentType) {
18 | case ComponentDatatype.BYTE:
19 | return function (
20 | dataView,
21 | byteOffset,
22 | numberOfComponents,
23 | componentTypeByteLength,
24 | result,
25 | ) {
26 | for (let i = 0; i < numberOfComponents; ++i) {
27 | result[i] = dataView.getInt8(
28 | byteOffset + i * componentTypeByteLength,
29 | );
30 | }
31 | };
32 | case ComponentDatatype.UNSIGNED_BYTE:
33 | return function (
34 | dataView,
35 | byteOffset,
36 | numberOfComponents,
37 | componentTypeByteLength,
38 | result,
39 | ) {
40 | for (let i = 0; i < numberOfComponents; ++i) {
41 | result[i] = dataView.getUint8(
42 | byteOffset + i * componentTypeByteLength,
43 | );
44 | }
45 | };
46 | case ComponentDatatype.SHORT:
47 | return function (
48 | dataView,
49 | byteOffset,
50 | numberOfComponents,
51 | componentTypeByteLength,
52 | result,
53 | ) {
54 | for (let i = 0; i < numberOfComponents; ++i) {
55 | result[i] = dataView.getInt16(
56 | byteOffset + i * componentTypeByteLength,
57 | true,
58 | );
59 | }
60 | };
61 | case ComponentDatatype.UNSIGNED_SHORT:
62 | return function (
63 | dataView,
64 | byteOffset,
65 | numberOfComponents,
66 | componentTypeByteLength,
67 | result,
68 | ) {
69 | for (let i = 0; i < numberOfComponents; ++i) {
70 | result[i] = dataView.getUint16(
71 | byteOffset + i * componentTypeByteLength,
72 | true,
73 | );
74 | }
75 | };
76 | case ComponentDatatype.INT:
77 | return function (
78 | dataView,
79 | byteOffset,
80 | numberOfComponents,
81 | componentTypeByteLength,
82 | result,
83 | ) {
84 | for (let i = 0; i < numberOfComponents; ++i) {
85 | result[i] = dataView.getInt32(
86 | byteOffset + i * componentTypeByteLength,
87 | true,
88 | );
89 | }
90 | };
91 | case ComponentDatatype.UNSIGNED_INT:
92 | return function (
93 | dataView,
94 | byteOffset,
95 | numberOfComponents,
96 | componentTypeByteLength,
97 | result,
98 | ) {
99 | for (let i = 0; i < numberOfComponents; ++i) {
100 | result[i] = dataView.getUint32(
101 | byteOffset + i * componentTypeByteLength,
102 | true,
103 | );
104 | }
105 | };
106 | case ComponentDatatype.FLOAT:
107 | return function (
108 | dataView,
109 | byteOffset,
110 | numberOfComponents,
111 | componentTypeByteLength,
112 | result,
113 | ) {
114 | for (let i = 0; i < numberOfComponents; ++i) {
115 | result[i] = dataView.getFloat32(
116 | byteOffset + i * componentTypeByteLength,
117 | true,
118 | );
119 | }
120 | };
121 | case ComponentDatatype.DOUBLE:
122 | return function (
123 | dataView,
124 | byteOffset,
125 | numberOfComponents,
126 | componentTypeByteLength,
127 | result,
128 | ) {
129 | for (let i = 0; i < numberOfComponents; ++i) {
130 | result[i] = dataView.getFloat64(
131 | byteOffset + i * componentTypeByteLength,
132 | true,
133 | );
134 | }
135 | };
136 | }
137 | }
138 |
139 | /**
140 | * A callback function that logs messages.
141 | * @callback ComponentReader
142 | *
143 | * @param {DataView} dataView The data view to read from.
144 | * @param {number} byteOffset The byte offset applied when reading from the data view.
145 | * @param {number} numberOfComponents The number of components to read.
146 | * @param {number} componentTypeByteLength The byte length of each component.
147 | * @param {number} result An array storing the components that are read.
148 | *
149 | * @private
150 | */
151 |
--------------------------------------------------------------------------------
/lib/getImageExtension.js:
--------------------------------------------------------------------------------
1 | "use strict";
2 | const Cesium = require("cesium");
3 |
4 | const RuntimeError = Cesium.RuntimeError;
5 |
6 | module.exports = getImageExtension;
7 |
8 | /**
9 | * Get the image extension from a Buffer containing image data.
10 | *
11 | * @param {Buffer} data The image data.
12 | * @returns {string} The image extension.
13 | *
14 | * @private
15 | */
16 | function getImageExtension(data) {
17 | const header = data.slice(0, 2);
18 | const webpHeaderRIFFChars = data.slice(0, 4);
19 | const webpHeaderWEBPChars = data.slice(8, 12);
20 |
21 | if (header.equals(Buffer.from([0x42, 0x4d]))) {
22 | return ".bmp";
23 | } else if (header.equals(Buffer.from([0x47, 0x49]))) {
24 | return ".gif";
25 | } else if (header.equals(Buffer.from([0xff, 0xd8]))) {
26 | return ".jpg";
27 | } else if (header.equals(Buffer.from([0x89, 0x50]))) {
28 | return ".png";
29 | } else if (header.equals(Buffer.from([0xab, 0x4b]))) {
30 | return ".ktx2";
31 | } else if (header.equals(Buffer.from([0x73, 0x42]))) {
32 | return ".basis";
33 | } else if (
34 | webpHeaderRIFFChars.equals(Buffer.from([0x52, 0x49, 0x46, 0x46])) &&
35 | webpHeaderWEBPChars.equals(Buffer.from([0x57, 0x45, 0x42, 0x50]))
36 | ) {
37 | // See https://developers.google.com/speed/webp/docs/riff_container#webp_file_header
38 | return ".webp";
39 | }
40 |
41 | throw new RuntimeError("Image data does not have valid header");
42 | }
43 |
--------------------------------------------------------------------------------
/lib/getJsonBufferPadded.js:
--------------------------------------------------------------------------------
1 | "use strict";
2 | const Cesium = require("cesium");
3 |
4 | const defaultValue = Cesium.defaultValue;
5 |
6 | module.exports = getJsonBufferPadded;
7 |
8 | /**
9 | * Convert the JSON object to a padded buffer.
10 | *
11 | * Pad the JSON with extra whitespace to fit the next 8-byte boundary. This ensures proper alignment
12 | * for the section that follows. glTF only requires 4-byte alignment but some extensions like
13 | * EXT_structural_metadata require 8-byte alignment for some buffer views.
14 | *
15 | * @param {object} json The JSON object.
16 | * @param {number} [byteOffset=0] The byte offset on which the buffer starts.
17 | * @returns {Buffer} The padded JSON buffer.
18 | *
19 | * @private
20 | */
21 | function getJsonBufferPadded(json, byteOffset) {
22 | byteOffset = defaultValue(byteOffset, 0);
23 | let string = JSON.stringify(json);
24 |
25 | const boundary = 8;
26 | const byteLength = Buffer.byteLength(string);
27 | const remainder = (byteOffset + byteLength) % boundary;
28 | const padding = remainder === 0 ? 0 : boundary - remainder;
29 | let whitespace = "";
30 | for (let i = 0; i < padding; ++i) {
31 | whitespace += " ";
32 | }
33 | string += whitespace;
34 |
35 | return Buffer.from(string);
36 | }
37 |
--------------------------------------------------------------------------------
/lib/getStatistics.js:
--------------------------------------------------------------------------------
1 | "use strict";
2 | const Cesium = require("cesium");
3 | const ForEach = require("./ForEach");
4 |
5 | const defined = Cesium.defined;
6 | const isDataUri = Cesium.isDataUri;
7 | const WebGLConstants = Cesium.WebGLConstants;
8 |
9 | module.exports = getStatistics;
10 |
11 | /**
12 | * Returns an object containing the statistics for the glTF asset.
13 | *
14 | * @param {object} gltf A javascript object containing a glTF asset.
15 | * @param {number} [nodeId] If defined, statistics will only process number of draw calls and rendered primitives for the specified node.
16 | * @returns {Statistics} Object containing the statistics of the glTF asset.
17 | *
18 | * @see Statistics
19 | */
20 | function getStatistics(gltf, nodeId) {
21 | const statistics = new Statistics();
22 |
23 | if (defined(nodeId)) {
24 | const nodeDrawStats = getDrawCallStatisticsForNode(gltf, nodeId);
25 | statistics.numberOfDrawCalls = nodeDrawStats.numberOfDrawCalls;
26 | statistics.numberOfRenderedPrimitives =
27 | nodeDrawStats.numberOfRenderedPrimitives;
28 | return statistics;
29 | }
30 |
31 | const drawStats = getDrawCallStatistics(gltf);
32 |
33 | statistics.buffersByteLength = getBuffersByteLength(gltf);
34 | statistics.numberOfImages = defined(gltf.images) ? gltf.images.length : 0;
35 | statistics.numberOfExternalRequests = getNumberOfExternalRequests(gltf);
36 | statistics.numberOfDrawCalls = drawStats.numberOfDrawCalls;
37 | statistics.numberOfRenderedPrimitives = drawStats.numberOfRenderedPrimitives;
38 | statistics.numberOfNodes = defined(gltf.nodes) ? gltf.nodes.length : 0;
39 | statistics.numberOfMeshes = defined(gltf.meshes) ? gltf.meshes.length : 0;
40 | statistics.numberOfMaterials = defined(gltf.materials)
41 | ? gltf.materials.length
42 | : 0;
43 | statistics.numberOfAnimations = defined(gltf.animations)
44 | ? gltf.animations.length
45 | : 0;
46 |
47 | return statistics;
48 | }
49 |
50 | function getBuffersByteLength(gltf) {
51 | let byteLength = 0;
52 | ForEach.buffer(gltf, function (buffer) {
53 | byteLength += buffer.byteLength;
54 | });
55 | return byteLength;
56 | }
57 |
58 | function getNumberOfExternalRequests(gltf) {
59 | let count = 0;
60 | ForEach.buffer(gltf, function (buffer) {
61 | if (defined(buffer.uri) && !isDataUri(buffer.uri)) {
62 | count++;
63 | }
64 | });
65 | ForEach.image(gltf, function (image) {
66 | if (defined(image.uri) && !isDataUri(image.uri)) {
67 | count++;
68 | }
69 | });
70 | ForEach.shader(gltf, function (shader) {
71 | if (defined(shader.uri) && !isDataUri(shader.uri)) {
72 | count++;
73 | }
74 | });
75 | return count;
76 | }
77 |
78 | function getNumberOfRenderedPrimitives(gltf, primitive) {
79 | let count = 0;
80 | if (defined(primitive.indices)) {
81 | count = gltf.accessors[primitive.indices].count;
82 | } else if (defined(primitive.attributes.POSITION)) {
83 | count = gltf.accessors[primitive.attributes.POSITION].count;
84 | }
85 | switch (primitive.mode) {
86 | case WebGLConstants.POINTS:
87 | return count;
88 | case WebGLConstants.LINES:
89 | return count / 2;
90 | case WebGLConstants.LINE_LOOP:
91 | return count;
92 | case WebGLConstants.LINE_STRIP:
93 | return Math.max(count - 1, 0);
94 | case WebGLConstants.TRIANGLES:
95 | return count / 3;
96 | case WebGLConstants.TRIANGLE_STRIP:
97 | case WebGLConstants.TRIANGLE_FAN:
98 | return Math.max(count - 2, 0);
99 | default:
100 | // TRIANGLES
101 | return count / 3;
102 | }
103 | }
104 |
105 | function getDrawCallStatisticsForNode(gltf, nodeId) {
106 | let numberOfDrawCalls = 0;
107 | let numberOfRenderedPrimitives = 0;
108 |
109 | ForEach.nodeInTree(gltf, [nodeId], function (node) {
110 | const mesh = gltf.meshes[node.mesh];
111 | if (defined(mesh)) {
112 | ForEach.meshPrimitive(mesh, function (primitive) {
113 | numberOfDrawCalls++;
114 | numberOfRenderedPrimitives += getNumberOfRenderedPrimitives(
115 | gltf,
116 | primitive,
117 | );
118 | });
119 | }
120 | });
121 |
122 | return {
123 | numberOfDrawCalls: numberOfDrawCalls,
124 | numberOfRenderedPrimitives: numberOfRenderedPrimitives,
125 | };
126 | }
127 |
128 | function getDrawCallStatistics(gltf) {
129 | let numberOfDrawCalls = 0;
130 | let numberOfRenderedPrimitives = 0;
131 |
132 | ForEach.mesh(gltf, function (mesh) {
133 | ForEach.meshPrimitive(mesh, function (primitive) {
134 | numberOfDrawCalls++;
135 | numberOfRenderedPrimitives += getNumberOfRenderedPrimitives(
136 | gltf,
137 | primitive,
138 | );
139 | });
140 | });
141 |
142 | return {
143 | numberOfDrawCalls: numberOfDrawCalls,
144 | numberOfRenderedPrimitives: numberOfRenderedPrimitives,
145 | };
146 | }
147 |
148 | /**
149 | * Contains statistics for a glTF asset.
150 | *
151 | * @property {number} buffersByteLength The total byte length of all buffers.
152 | * @property {number} numberOfImages The number of images in the asset.
153 | * @property {number} numberOfExternalRequests The number of external requests required to fetch the asset data.
154 | * @property {number} numberOfDrawCalls The number of draw calls required to render the asset.
155 | * @property {number} numberOfRenderedPrimitives The total number of rendered primitives in the asset (e.g. triangles).
156 | * @property {number} numberOfNodes The total number of nodes in the asset.
157 | * @property {number} numberOfMeshes The total number of meshes in the asset.
158 | * @property {number} numberOfMaterials The total number of materials in the asset.
159 | * @property {number} numberOfAnimations The total number of animations in the asset.
160 | *
161 | * @constructor
162 | *
163 | * @see getStatistics
164 | */
165 | function Statistics() {
166 | this.buffersByteLength = 0;
167 | this.numberOfImages = 0;
168 | this.numberOfExternalRequests = 0;
169 | this.numberOfDrawCalls = 0;
170 | this.numberOfRenderedPrimitives = 0;
171 | this.numberOfNodes = 0;
172 | this.numberOfMeshes = 0;
173 | this.numberOfMaterials = 0;
174 | this.numberOfAnimations = 0;
175 | }
176 |
177 | /**
178 | * Creates a string listing the statistics along with their descriptions.
179 | *
180 | * @returns {string} A string describing the statistics for the glTF asset.
181 | */
182 | Statistics.prototype.toString = function () {
183 | return (
184 | `Total byte length of all buffers: ${this.buffersByteLength} bytes` +
185 | `\nImages: ${this.numberOfImages}\nDraw calls: ${this.numberOfDrawCalls}\nRendered primitives (e.g., triangles): ${this.numberOfRenderedPrimitives}\nNodes: ${this.numberOfNodes}\nMeshes: ${this.numberOfMeshes}\nMaterials: ${this.numberOfMaterials}\nAnimations: ${this.numberOfAnimations}\nExternal requests (not data uris): ${this.numberOfExternalRequests}`
186 | );
187 | };
188 |
--------------------------------------------------------------------------------
/lib/glbToGltf.js:
--------------------------------------------------------------------------------
1 | "use strict";
2 | const parseGlb = require("./parseGlb");
3 | const processGltf = require("./processGltf");
4 |
5 | module.exports = glbToGltf;
6 |
7 | /**
8 | * Convert a glb to glTF.
9 | *
10 | * @param {Buffer} glb A buffer containing the glb contents.
11 | * @param {object} [options] The same options object as {@link processGltf}
12 | * @returns {Promise} A promise that resolves to an object containing the glTF and a dictionary containing separate resources.
13 | */
14 | function glbToGltf(glb, options) {
15 | const gltf = parseGlb(glb);
16 | return processGltf(gltf, options);
17 | }
18 |
--------------------------------------------------------------------------------
/lib/gltfToGlb.js:
--------------------------------------------------------------------------------
1 | "use strict";
2 | const Cesium = require("cesium");
3 | const getJsonBufferPadded = require("./getJsonBufferPadded");
4 | const processGltf = require("./processGltf");
5 |
6 | const clone = Cesium.clone;
7 | const defined = Cesium.defined;
8 |
9 | module.exports = gltfToGlb;
10 |
11 | /**
12 | * Convert a glTF to glb.
13 | *
14 | * @param {object} gltf A javascript object containing a glTF asset.
15 | * @param {object} [options] The same options object as {@link processGltf}
16 | * @returns {Promise} A promise that resolves to an object containing the glb and a dictionary containing separate resources.
17 | */
18 | function gltfToGlb(gltf, options) {
19 | options = defined(options) ? clone(options) : {};
20 | options.forceMergeBuffers = !options.separate;
21 | options.bufferStorage = {
22 | buffer: undefined,
23 | };
24 |
25 | return processGltf(gltf, options).then(function (results) {
26 | return {
27 | glb: getGlb(results.gltf, options.bufferStorage.buffer),
28 | separateResources: results.separateResources,
29 | };
30 | });
31 | }
32 |
33 | function getGlb(gltf, binaryBuffer) {
34 | const jsonBuffer = getJsonBufferPadded(gltf, 20);
35 |
36 | // Compute glb length: (Global header) + (JSON chunk header) + (JSON chunk) + [(Binary chunk header) + (Binary chunk)]
37 | let glbLength = 12 + 8 + jsonBuffer.length;
38 |
39 | if (defined(binaryBuffer)) {
40 | glbLength += 8 + binaryBuffer.length;
41 | }
42 |
43 | const glb = Buffer.alloc(glbLength);
44 |
45 | // Write binary glTF header (magic, version, length)
46 | let byteOffset = 0;
47 | glb.writeUInt32LE(0x46546c67, byteOffset);
48 | byteOffset += 4;
49 | glb.writeUInt32LE(2, byteOffset);
50 | byteOffset += 4;
51 | glb.writeUInt32LE(glbLength, byteOffset);
52 | byteOffset += 4;
53 |
54 | // Write JSON Chunk header (length, type)
55 | glb.writeUInt32LE(jsonBuffer.length, byteOffset);
56 | byteOffset += 4;
57 | glb.writeUInt32LE(0x4e4f534a, byteOffset); // JSON
58 | byteOffset += 4;
59 |
60 | // Write JSON Chunk
61 | jsonBuffer.copy(glb, byteOffset);
62 | byteOffset += jsonBuffer.length;
63 |
64 | if (defined(binaryBuffer)) {
65 | // Write Binary Chunk header (length, type)
66 | glb.writeUInt32LE(binaryBuffer.length, byteOffset);
67 | byteOffset += 4;
68 | glb.writeUInt32LE(0x004e4942, byteOffset); // BIN
69 | byteOffset += 4;
70 |
71 | // Write Binary Chunk
72 | binaryBuffer.copy(glb, byteOffset);
73 | }
74 |
75 | return glb;
76 | }
77 |
--------------------------------------------------------------------------------
/lib/mergeBuffers.js:
--------------------------------------------------------------------------------
1 | "use strict";
2 | const FS_WRITE_MAX_LENGTH = 2147479552; // See https://github.com/nodejs/node/issues/35605
3 | const BUFFER_MAX_LENGTH = require("buffer").constants.MAX_LENGTH;
4 | const BUFFER_MAX_BYTE_LENGTH = Math.min(FS_WRITE_MAX_LENGTH, BUFFER_MAX_LENGTH);
5 | const Cesium = require("cesium");
6 | const ForEach = require("./ForEach");
7 |
8 | const defaultValue = Cesium.defaultValue;
9 | const defined = Cesium.defined;
10 |
11 | module.exports = mergeBuffers;
12 |
13 | /**
14 | * Merge all buffers. Buffers with the same extras._pipeline.mergedBufferName will be merged together.
15 | *
16 | * @param {object} gltf A javascript object containing a glTF asset.
17 | * @param {string} [defaultName] The default name of the buffer data files.
18 | * @param {boolean} [force=false] Whether to force merging all buffers.
19 | * @returns {object} The glTF asset with its buffers merged.
20 | *
21 | * @private
22 | */
23 | function mergeBuffers(gltf, defaultName, force) {
24 | let baseBufferName = defaultName;
25 | if (!defined(baseBufferName)) {
26 | ForEach.buffer(gltf, function (buffer) {
27 | baseBufferName = defaultValue(baseBufferName, buffer.name);
28 | });
29 | baseBufferName = defaultValue(baseBufferName, "buffer");
30 | }
31 |
32 | let buffersByteLength = 0;
33 | const emptyBuffers = [];
34 | const emptyBufferIds = [];
35 |
36 | ForEach.buffer(gltf, function (buffer, bufferId) {
37 | const source = buffer.extras._pipeline.source;
38 | if (defined(source)) {
39 | buffersByteLength += source.length;
40 | } else {
41 | emptyBuffers.push(buffer);
42 | emptyBufferIds.push(bufferId);
43 | }
44 |
45 | const extensions = defaultValue(
46 | buffer.extensions,
47 | defaultValue.EMPTY_OBJECT,
48 | );
49 | const meshoptObject = extensions.EXT_meshopt_compression;
50 | if (defined(meshoptObject) && meshoptObject.fallback) {
51 | // Prevent empty meshopt buffer from being merged into main buffer
52 | buffer.extras._pipeline.mergedBufferName = `meshopt-fallback-${bufferId}`;
53 | }
54 | });
55 |
56 | // Don't merge buffers if the merged buffer will exceed the Node limit.
57 | const splitBuffers =
58 | buffersByteLength > mergeBuffers._getBufferMaxByteLength();
59 |
60 | const buffersToMerge = {};
61 | const mergedNameCount = {};
62 |
63 | forEachBufferViewLikeObject(gltf, function (bufferView) {
64 | const buffer = gltf.buffers[bufferView.buffer];
65 | const source = buffer.extras._pipeline.source;
66 | if (!defined(source)) {
67 | return;
68 | }
69 |
70 | let mergedName = buffer.extras._pipeline.mergedBufferName;
71 | mergedName = defined(mergedName)
72 | ? `${baseBufferName}-${mergedName}`
73 | : baseBufferName;
74 |
75 | if (splitBuffers) {
76 | if (!defined(mergedNameCount[mergedName])) {
77 | mergedNameCount[mergedName] = 0;
78 | }
79 | mergedName += `-${mergedNameCount[mergedName]++}`;
80 | }
81 |
82 | if (force) {
83 | mergedName = baseBufferName;
84 | }
85 |
86 | if (!defined(buffersToMerge[mergedName])) {
87 | buffersToMerge[mergedName] = {
88 | buffers: [],
89 | byteLength: 0,
90 | index: Object.keys(buffersToMerge).length,
91 | };
92 | }
93 | const buffers = buffersToMerge[mergedName].buffers;
94 | let byteLength = buffersToMerge[mergedName].byteLength;
95 | const index = buffersToMerge[mergedName].index;
96 |
97 | const sourceBufferViewData = Buffer.from(
98 | source.slice(
99 | bufferView.byteOffset,
100 | bufferView.byteOffset + bufferView.byteLength,
101 | ),
102 | );
103 | const bufferViewPadding = allocateBufferPadding(byteLength);
104 | if (defined(bufferViewPadding)) {
105 | buffers.push(bufferViewPadding);
106 | byteLength += bufferViewPadding.byteLength;
107 | }
108 |
109 | bufferView.byteOffset = byteLength;
110 | bufferView.buffer = index;
111 |
112 | buffers.push(sourceBufferViewData);
113 | byteLength += sourceBufferViewData.byteLength;
114 |
115 | buffersToMerge[mergedName].byteLength = byteLength;
116 | });
117 |
118 | const buffersLength = Object.keys(buffersToMerge).length;
119 | gltf.buffers = new Array(buffersLength);
120 |
121 | for (const mergedName in buffersToMerge) {
122 | if (Object.prototype.hasOwnProperty.call(buffersToMerge, mergedName)) {
123 | const buffers = buffersToMerge[mergedName].buffers;
124 | const byteLength = buffersToMerge[mergedName].byteLength;
125 | const index = buffersToMerge[mergedName].index;
126 | const bufferPadding = allocateBufferPadding(byteLength);
127 | if (defined(bufferPadding)) {
128 | buffers.push(bufferPadding);
129 | }
130 | const mergedSource =
131 | buffers.length > 1 ? Buffer.concat(buffers) : buffers[0];
132 | gltf.buffers[index] = {
133 | name: mergedName,
134 | byteLength: mergedSource.byteLength,
135 | extras: {
136 | _pipeline: {
137 | source: mergedSource,
138 | },
139 | },
140 | };
141 | }
142 | }
143 |
144 | const emptyBuffersLength = emptyBuffers.length;
145 | for (let i = 0; i < emptyBuffersLength; ++i) {
146 | const emptyBuffer = emptyBuffers[i];
147 | const emptyBufferId = emptyBufferIds[i];
148 | const newBufferId = gltf.buffers.length;
149 | forEachBufferViewLikeObject(gltf, function (bufferView) {
150 | if (bufferView.buffer === emptyBufferId) {
151 | bufferView.buffer = newBufferId;
152 | }
153 | });
154 | gltf.buffers.push(emptyBuffer);
155 | }
156 |
157 | return gltf;
158 | }
159 |
160 | function forEachBufferViewLikeObject(gltf, callback) {
161 | ForEach.bufferView(gltf, function (bufferView) {
162 | callback(bufferView);
163 |
164 | const extensions = defaultValue(
165 | bufferView.extensions,
166 | defaultValue.EMPTY_OBJECT,
167 | );
168 | const meshoptObject = extensions.EXT_meshopt_compression;
169 | if (defined(meshoptObject)) {
170 | // The EXT_meshopt_compression object has buffer, byteOffset, and byteLength properties like a bufferView
171 | callback(meshoptObject);
172 | }
173 | });
174 | }
175 |
176 | function allocateBufferPadding(byteLength) {
177 | const boundary = 8;
178 | const remainder = byteLength % boundary;
179 | const padding = remainder === 0 ? 0 : boundary - remainder;
180 | if (padding > 0) {
181 | return Buffer.alloc(padding);
182 | }
183 | return undefined;
184 | }
185 |
186 | // Exposed for testing
187 | mergeBuffers._getBufferMaxByteLength = function () {
188 | return BUFFER_MAX_BYTE_LENGTH;
189 | };
190 |
--------------------------------------------------------------------------------
/lib/moveTechniqueRenderStates.js:
--------------------------------------------------------------------------------
1 | "use strict";
2 |
3 | const Cesium = require("cesium");
4 | const addExtensionsUsed = require("./addExtensionsUsed");
5 | const ForEach = require("./ForEach");
6 |
7 | const defaultValue = Cesium.defaultValue;
8 | const defined = Cesium.defined;
9 | const WebGLConstants = Cesium.WebGLConstants;
10 |
11 | module.exports = moveTechniqueRenderStates;
12 |
13 | const defaultBlendEquation = [WebGLConstants.FUNC_ADD, WebGLConstants.FUNC_ADD];
14 |
15 | const defaultBlendFactors = [
16 | WebGLConstants.ONE,
17 | WebGLConstants.ZERO,
18 | WebGLConstants.ONE,
19 | WebGLConstants.ZERO,
20 | ];
21 |
22 | function isStateEnabled(renderStates, state) {
23 | const enabled = renderStates.enable;
24 | if (!defined(enabled)) {
25 | return false;
26 | }
27 |
28 | return enabled.indexOf(state) > -1;
29 | }
30 |
31 | const supportedBlendFactors = [
32 | WebGLConstants.ZERO,
33 | WebGLConstants.ONE,
34 | WebGLConstants.SRC_COLOR,
35 | WebGLConstants.ONE_MINUS_SRC_COLOR,
36 | WebGLConstants.SRC_ALPHA,
37 | WebGLConstants.ONE_MINUS_SRC_ALPHA,
38 | WebGLConstants.DST_ALPHA,
39 | WebGLConstants.ONE_MINUS_DST_ALPHA,
40 | WebGLConstants.DST_COLOR,
41 | WebGLConstants.ONE_MINUS_DST_COLOR,
42 | ];
43 |
44 | // If any of the blend factors are not supported, return the default
45 | function getSupportedBlendFactors(value, defaultValue) {
46 | if (!defined(value)) {
47 | return defaultValue;
48 | }
49 |
50 | for (let i = 0; i < 4; i++) {
51 | if (supportedBlendFactors.indexOf(value[i]) === -1) {
52 | return defaultValue;
53 | }
54 | }
55 |
56 | return value;
57 | }
58 |
59 | /**
60 | * Move glTF 1.0 technique render states to glTF 2.0 materials properties and KHR_blend extension.
61 | *
62 | * @param {object} gltf A javascript object containing a glTF asset.
63 | * @returns {object} The updated glTF asset.
64 | *
65 | * @private
66 | */
67 | function moveTechniqueRenderStates(gltf) {
68 | const blendingForTechnique = {};
69 | const materialPropertiesForTechnique = {};
70 | const techniquesLegacy = gltf.techniques;
71 | if (!defined(techniquesLegacy)) {
72 | return gltf;
73 | }
74 |
75 | ForEach.technique(gltf, function (techniqueLegacy, techniqueIndex) {
76 | const renderStates = techniqueLegacy.states;
77 | if (defined(renderStates)) {
78 | const materialProperties = (materialPropertiesForTechnique[
79 | techniqueIndex
80 | ] = {});
81 |
82 | // If BLEND is enabled, the material should have alpha mode BLEND
83 | if (isStateEnabled(renderStates, WebGLConstants.BLEND)) {
84 | materialProperties.alphaMode = "BLEND";
85 |
86 | const blendFunctions = renderStates.functions;
87 | if (
88 | defined(blendFunctions) &&
89 | (defined(blendFunctions.blendEquationSeparate) ||
90 | defined(blendFunctions.blendFuncSeparate))
91 | ) {
92 | blendingForTechnique[techniqueIndex] = {
93 | blendEquation: defaultValue(
94 | blendFunctions.blendEquationSeparate,
95 | defaultBlendEquation,
96 | ),
97 | blendFactors: getSupportedBlendFactors(
98 | blendFunctions.blendFuncSeparate,
99 | defaultBlendFactors,
100 | ),
101 | };
102 | }
103 | }
104 |
105 | // If CULL_FACE is not enabled, the material should be doubleSided
106 | if (!isStateEnabled(renderStates, WebGLConstants.CULL_FACE)) {
107 | materialProperties.doubleSided = true;
108 | }
109 |
110 | delete techniqueLegacy.states;
111 | }
112 | });
113 |
114 | if (Object.keys(blendingForTechnique).length > 0) {
115 | if (!defined(gltf.extensions)) {
116 | gltf.extensions = {};
117 | }
118 |
119 | addExtensionsUsed(gltf, "KHR_blend");
120 | }
121 |
122 | ForEach.material(gltf, function (material) {
123 | if (defined(material.technique)) {
124 | const materialProperties =
125 | materialPropertiesForTechnique[material.technique];
126 | ForEach.objectLegacy(materialProperties, function (value, property) {
127 | material[property] = value;
128 | });
129 |
130 | const blending = blendingForTechnique[material.technique];
131 | if (defined(blending)) {
132 | if (!defined(material.extensions)) {
133 | material.extensions = {};
134 | }
135 |
136 | material.extensions.KHR_blend = blending;
137 | }
138 | }
139 | });
140 |
141 | return gltf;
142 | }
143 |
--------------------------------------------------------------------------------
/lib/moveTechniquesToExtension.js:
--------------------------------------------------------------------------------
1 | "use strict";
2 |
3 | const Cesium = require("cesium");
4 | const addExtensionsUsed = require("./addExtensionsUsed");
5 | const addExtensionsRequired = require("./addExtensionsRequired");
6 | const addToArray = require("./addToArray");
7 | const ForEach = require("./ForEach");
8 |
9 | const defined = Cesium.defined;
10 |
11 | module.exports = moveTechniquesToExtension;
12 |
13 | /**
14 | * Move glTF 1.0 material techniques to glTF 2.0 KHR_techniques_webgl extension.
15 | *
16 | * @param {object} gltf A javascript object containing a glTF asset.
17 | * @returns {object} The updated glTF asset.
18 | *
19 | * @private
20 | */
21 | function moveTechniquesToExtension(gltf) {
22 | const techniquesLegacy = gltf.techniques;
23 | const mappedUniforms = {};
24 | const updatedTechniqueIndices = {};
25 | const seenPrograms = {};
26 | if (defined(techniquesLegacy)) {
27 | const extension = {
28 | programs: [],
29 | shaders: [],
30 | techniques: [],
31 | };
32 |
33 | // Some 1.1 models have a glExtensionsUsed property that can be transferred to program.glExtensions
34 | const glExtensions = gltf.glExtensionsUsed;
35 | delete gltf.glExtensionsUsed;
36 |
37 | ForEach.technique(gltf, function (techniqueLegacy, techniqueId) {
38 | const technique = {
39 | name: techniqueLegacy.name,
40 | program: undefined,
41 | attributes: {},
42 | uniforms: {},
43 | };
44 |
45 | let parameterLegacy;
46 | ForEach.techniqueAttribute(
47 | techniqueLegacy,
48 | function (parameterName, attributeName) {
49 | parameterLegacy = techniqueLegacy.parameters[parameterName];
50 | technique.attributes[attributeName] = {
51 | semantic: parameterLegacy.semantic,
52 | };
53 | },
54 | );
55 |
56 | ForEach.techniqueUniform(
57 | techniqueLegacy,
58 | function (parameterName, uniformName) {
59 | parameterLegacy = techniqueLegacy.parameters[parameterName];
60 | technique.uniforms[uniformName] = {
61 | count: parameterLegacy.count,
62 | node: parameterLegacy.node,
63 | type: parameterLegacy.type,
64 | semantic: parameterLegacy.semantic,
65 | value: parameterLegacy.value,
66 | };
67 |
68 | // Store the name of the uniform to update material values.
69 | if (!defined(mappedUniforms[techniqueId])) {
70 | mappedUniforms[techniqueId] = {};
71 | }
72 | mappedUniforms[techniqueId][parameterName] = uniformName;
73 | },
74 | );
75 |
76 | if (!defined(seenPrograms[techniqueLegacy.program])) {
77 | const programLegacy = gltf.programs[techniqueLegacy.program];
78 |
79 | const program = {
80 | name: programLegacy.name,
81 | fragmentShader: undefined,
82 | vertexShader: undefined,
83 | glExtensions: glExtensions,
84 | };
85 |
86 | const fs = gltf.shaders[programLegacy.fragmentShader];
87 | program.fragmentShader = addToArray(extension.shaders, fs, true);
88 |
89 | const vs = gltf.shaders[programLegacy.vertexShader];
90 | program.vertexShader = addToArray(extension.shaders, vs, true);
91 |
92 | technique.program = addToArray(extension.programs, program);
93 | seenPrograms[techniqueLegacy.program] = technique.program;
94 | } else {
95 | technique.program = seenPrograms[techniqueLegacy.program];
96 | }
97 |
98 | // Store the index of the new technique to reference instead.
99 | updatedTechniqueIndices[techniqueId] = addToArray(
100 | extension.techniques,
101 | technique,
102 | );
103 | });
104 |
105 | if (extension.techniques.length > 0) {
106 | if (!defined(gltf.extensions)) {
107 | gltf.extensions = {};
108 | }
109 |
110 | gltf.extensions.KHR_techniques_webgl = extension;
111 | addExtensionsUsed(gltf, "KHR_techniques_webgl");
112 | addExtensionsRequired(gltf, "KHR_techniques_webgl");
113 | }
114 | }
115 |
116 | ForEach.material(gltf, function (material) {
117 | if (defined(material.technique)) {
118 | const materialExtension = {
119 | technique: updatedTechniqueIndices[material.technique],
120 | };
121 |
122 | ForEach.objectLegacy(material.values, function (value, parameterName) {
123 | if (!defined(materialExtension.values)) {
124 | materialExtension.values = {};
125 | }
126 |
127 | const uniformName = mappedUniforms[material.technique][parameterName];
128 | if (defined(uniformName)) {
129 | materialExtension.values[uniformName] = value;
130 | }
131 | });
132 |
133 | if (!defined(material.extensions)) {
134 | material.extensions = {};
135 | }
136 |
137 | material.extensions.KHR_techniques_webgl = materialExtension;
138 | }
139 |
140 | delete material.technique;
141 | delete material.values;
142 | });
143 |
144 | delete gltf.techniques;
145 | delete gltf.programs;
146 | delete gltf.shaders;
147 |
148 | return gltf;
149 | }
150 |
--------------------------------------------------------------------------------
/lib/numberOfComponentsForType.js:
--------------------------------------------------------------------------------
1 | "use strict";
2 |
3 | module.exports = numberOfComponentsForType;
4 |
5 | /**
6 | * Utility function for retrieving the number of components in a given type.
7 | *
8 | * @param {string} type glTF type
9 | * @returns {number} The number of components in that type.
10 | *
11 | * @private
12 | */
13 | function numberOfComponentsForType(type) {
14 | switch (type) {
15 | case "SCALAR":
16 | return 1;
17 | case "VEC2":
18 | return 2;
19 | case "VEC3":
20 | return 3;
21 | case "VEC4":
22 | case "MAT2":
23 | return 4;
24 | case "MAT3":
25 | return 9;
26 | case "MAT4":
27 | return 16;
28 | }
29 | }
30 |
--------------------------------------------------------------------------------
/lib/parseGlb.js:
--------------------------------------------------------------------------------
1 | "use strict";
2 | const Cesium = require("cesium");
3 | const addPipelineExtras = require("./addPipelineExtras");
4 | const removeExtensionsUsed = require("./removeExtensionsUsed");
5 |
6 | const defaultValue = Cesium.defaultValue;
7 | const defined = Cesium.defined;
8 | const getMagic = Cesium.getMagic;
9 | const getStringFromTypedArray = Cesium.getStringFromTypedArray;
10 | const RuntimeError = Cesium.RuntimeError;
11 |
12 | const sizeOfUint32 = 4;
13 |
14 | module.exports = parseGlb;
15 |
16 | /**
17 | * Convert a binary glTF to glTF.
18 | *
19 | * The returned glTF has pipeline extras included. The embedded binary data is stored in gltf.buffers[0].extras._pipeline.source.
20 | *
21 | * @param {Buffer} glb The glb data to parse.
22 | * @returns {object} A javascript object containing a glTF asset with pipeline extras included.
23 | *
24 | * @private
25 | */
26 | function parseGlb(glb) {
27 | // Check that the magic string is present
28 | const magic = getMagic(glb);
29 | if (magic !== "glTF") {
30 | throw new RuntimeError("File is not valid binary glTF");
31 | }
32 |
33 | const header = readHeader(glb, 0, 5);
34 | const version = header[1];
35 | if (version !== 1 && version !== 2) {
36 | throw new RuntimeError("Binary glTF version is not 1 or 2");
37 | }
38 |
39 | if (version === 1) {
40 | return parseGlbVersion1(glb, header);
41 | }
42 |
43 | return parseGlbVersion2(glb, header);
44 | }
45 |
46 | function readHeader(glb, byteOffset, count) {
47 | const dataView = new DataView(glb.buffer);
48 | const header = new Array(count);
49 | for (let i = 0; i < count; ++i) {
50 | header[i] = dataView.getUint32(
51 | glb.byteOffset + byteOffset + i * sizeOfUint32,
52 | true,
53 | );
54 | }
55 | return header;
56 | }
57 |
58 | function parseGlbVersion1(glb, header) {
59 | const length = header[2];
60 | const contentLength = header[3];
61 | const contentFormat = header[4];
62 |
63 | // Check that the content format is 0, indicating that it is JSON
64 | if (contentFormat !== 0) {
65 | throw new RuntimeError("Binary glTF scene format is not JSON");
66 | }
67 |
68 | const jsonStart = 20;
69 | const binaryStart = jsonStart + contentLength;
70 |
71 | const contentString = getStringFromTypedArray(glb, jsonStart, contentLength);
72 | const gltf = JSON.parse(contentString);
73 | addPipelineExtras(gltf);
74 |
75 | const binaryBuffer = glb.subarray(binaryStart, length);
76 |
77 | const buffers = gltf.buffers;
78 | if (defined(buffers) && Object.keys(buffers).length > 0) {
79 | // In some older models, the binary glTF buffer is named KHR_binary_glTF
80 | const binaryGltfBuffer = defaultValue(
81 | buffers.binary_glTF,
82 | buffers.KHR_binary_glTF,
83 | );
84 | if (defined(binaryGltfBuffer)) {
85 | binaryGltfBuffer.extras._pipeline.source = binaryBuffer;
86 | delete binaryGltfBuffer.uri;
87 | }
88 | }
89 | // Remove the KHR_binary_glTF extension
90 | removeExtensionsUsed(gltf, "KHR_binary_glTF");
91 | return gltf;
92 | }
93 |
94 | function parseGlbVersion2(glb, header) {
95 | const length = header[2];
96 | let byteOffset = 12;
97 | let gltf;
98 | let binaryBuffer;
99 | while (byteOffset < length) {
100 | const chunkHeader = readHeader(glb, byteOffset, 2);
101 | const chunkLength = chunkHeader[0];
102 | const chunkType = chunkHeader[1];
103 | byteOffset += 8;
104 | const chunkBuffer = glb.subarray(byteOffset, byteOffset + chunkLength);
105 | byteOffset += chunkLength;
106 | // Load JSON chunk
107 | if (chunkType === 0x4e4f534a) {
108 | const jsonString = getStringFromTypedArray(chunkBuffer);
109 | gltf = JSON.parse(jsonString);
110 | addPipelineExtras(gltf);
111 | }
112 | // Load Binary chunk
113 | else if (chunkType === 0x004e4942) {
114 | binaryBuffer = chunkBuffer;
115 | }
116 | }
117 | if (defined(gltf) && defined(binaryBuffer)) {
118 | const buffers = gltf.buffers;
119 | if (defined(buffers) && buffers.length > 0) {
120 | const buffer = buffers[0];
121 | buffer.extras._pipeline.source = binaryBuffer;
122 | }
123 | }
124 | return gltf;
125 | }
126 |
--------------------------------------------------------------------------------
/lib/processGlb.js:
--------------------------------------------------------------------------------
1 | "use strict";
2 | const parseGlb = require("./parseGlb");
3 | const gltfToGlb = require("./gltfToGlb");
4 |
5 | module.exports = processGlb;
6 |
7 | /**
8 | * Run a glb through the gltf-pipeline.
9 | *
10 | * @param {Buffer} glb A buffer containing the glb contents.
11 | * @param {object} [options] The same options object as {@link processGltf}
12 | * @returns {Promise} A promise that resolves to an object containing the glb and a dictionary containing separate resources.
13 | */
14 | function processGlb(glb, options) {
15 | const gltf = parseGlb(glb);
16 | return gltfToGlb(gltf, options);
17 | }
18 |
--------------------------------------------------------------------------------
/lib/processGltf.js:
--------------------------------------------------------------------------------
1 | "use strict";
2 | const Cesium = require("cesium");
3 | const Promise = require("bluebird");
4 | const addDefaults = require("./addDefaults");
5 | const addPipelineExtras = require("./addPipelineExtras");
6 | const getStatistics = require("./getStatistics");
7 | const readResources = require("./readResources");
8 | const removeDefaults = require("./removeDefaults");
9 | const removePipelineExtras = require("./removePipelineExtras");
10 | const removeUnusedElements = require("./removeUnusedElements");
11 | const updateVersion = require("./updateVersion");
12 | const writeResources = require("./writeResources");
13 | const compressDracoMeshes = require("./compressDracoMeshes");
14 |
15 | const clone = Cesium.clone;
16 | const defaultValue = Cesium.defaultValue;
17 | const defined = Cesium.defined;
18 |
19 | module.exports = processGltf;
20 |
21 | /**
22 | * Run a glTF through the gltf-pipeline.
23 | *
24 | * @param {object} gltf A javascript object containing a glTF asset. The glTF is modified in place.
25 | * @param {object} [options] An object with the following properties:
26 | * @param {string} [options.resourceDirectory] The path for reading separate resources.
27 | * @param {string} [options.name] The name of the glTF asset, for writing separate resources.
28 | * @param {boolean} [options.separate = false] Write separate buffers, shaders, and textures instead of embedding them in the glTF.
29 | * @param {boolean} [options.separateTextures = false] Write out separate textures only.
30 | * @param {boolean} [options.stats = false] Print statistics to console for input and output glTF files.
31 | * @param {object} [options.dracoOptions] Options to pass to the compressDracoMeshes stage. If undefined, stage is not run.
32 | * @param {Stage[]} [options.customStages] Custom stages to run on the glTF asset.
33 | * @param {Logger} [options.logger] A callback function for handling logged messages. Defaults to console.log.
34 | * @param {string[]} [options.baseColorTextureNames] Names of uniforms that indicate base color textures.
35 | * @param {string[]} [options.baseColorFactorNames] Names of uniforms that indicate base color factors.
36 | *
37 | * @returns {Promise} A promise that resolves to an object containing the glTF and a dictionary containing separate resources.
38 | */
39 | function processGltf(gltf, options) {
40 | const defaults = processGltf.defaults;
41 | options = defined(options) ? clone(options) : {};
42 | options.separateBuffers = defaultValue(options.separate, defaults.separate);
43 | options.separateShaders = defaultValue(options.separate, defaults.separate);
44 | options.separateTextures =
45 | defaultValue(options.separateTextures, defaults.separateTextures) ||
46 | options.separate;
47 | options.stats = defaultValue(options.stats, defaults.stats);
48 | options.logger = defaultValue(options.logger, getDefaultLogger());
49 | options.separateResources = {};
50 | options.customStages = defaultValue(options.customStages, []);
51 |
52 | const preStages = [
53 | addPipelineExtras,
54 | readResources,
55 | updateVersion,
56 | addDefaults,
57 | ];
58 |
59 | const postStages = [writeResources, removePipelineExtras, removeDefaults];
60 |
61 | const pipelineStages = getStages(options);
62 | const stages = preStages.concat(
63 | options.customStages,
64 | pipelineStages,
65 | postStages,
66 | );
67 |
68 | return Promise.each(stages, function (stage) {
69 | return stage(gltf, options);
70 | }).then(function () {
71 | printStats(gltf, options, true);
72 | return {
73 | gltf: gltf,
74 | separateResources: options.separateResources,
75 | };
76 | });
77 | }
78 |
79 | function printStats(gltf, options, processed) {
80 | if (options.stats) {
81 | options.logger(processed ? "Statistics after:" : "Statistics before:");
82 | options.logger(getStatistics(gltf).toString());
83 | }
84 | }
85 |
86 | function getStages(options) {
87 | const stages = [];
88 | if (defined(options.dracoOptions)) {
89 | stages.push(compressDracoMeshes);
90 | }
91 | if (!options.keepUnusedElements) {
92 | stages.push(function (gltf, options) {
93 | removeUnusedElements(gltf);
94 | });
95 | }
96 | return stages;
97 | }
98 |
99 | function getDefaultLogger() {
100 | return function (message) {
101 | console.log(message);
102 | };
103 | }
104 |
105 | /**
106 | * Default values that will be used when calling processGltf(options) unless specified in the options object.
107 | */
108 | processGltf.defaults = {
109 | /**
110 | * Gets or sets whether to write out separate buffers, shaders, and textures instead of embedding them in the glTF
111 | * @type Boolean
112 | * @default false
113 | */
114 | separate: false,
115 | /**
116 | * Gets or sets whether to write out separate textures only.
117 | * @type Boolean
118 | * @default false
119 | */
120 | separateTextures: false,
121 | /**
122 | * Gets or sets whether to print statistics to console for input and output glTF files.
123 | * @type Boolean
124 | * @default false
125 | */
126 | stats: false,
127 | /**
128 | * Keep unused 'node', 'mesh' and 'material' elements.
129 | * @type Boolean
130 | * @default false
131 | */
132 | keepUnusedElements: false,
133 | /**
134 | * When false, materials with KHR_techniques_webgl, KHR_blend, or KHR_materials_common will be converted to PBR.
135 | * @type Boolean
136 | * @default false
137 | */
138 | keepLegacyExtensions: false,
139 | /**
140 | * Gets or sets whether to compress the meshes using Draco. Adds the KHR_draco_mesh_compression extension.
141 | * @type Boolean
142 | * @default false
143 | */
144 | compressDracoMeshes: false,
145 | };
146 |
147 | /**
148 | * A callback function that logs messages.
149 | * @callback Logger
150 | *
151 | * @param {string} message The message to log.
152 | */
153 |
154 | /**
155 | * A stage that processes a glTF asset.
156 | * @callback Stage
157 | *
158 | * @param {object} gltf The glTF asset.
159 | * @returns {Promise|Object} The glTF asset or a promise that resolves to the glTF asset.
160 | */
161 |
--------------------------------------------------------------------------------
/lib/readAccessorPacked.js:
--------------------------------------------------------------------------------
1 | "use strict";
2 | const Cesium = require("cesium");
3 | const getAccessorByteStride = require("./getAccessorByteStride");
4 | const getComponentReader = require("./getComponentReader");
5 | const numberOfComponentsForType = require("./numberOfComponentsForType");
6 |
7 | const ComponentDatatype = Cesium.ComponentDatatype;
8 | const defined = Cesium.defined;
9 |
10 | module.exports = readAccessorPacked;
11 |
12 | /**
13 | * Returns the accessor data in a contiguous array.
14 | *
15 | * @param {object} gltf A javascript object containing a glTF asset.
16 | * @param {object} accessor The accessor.
17 | * @returns {Array} The accessor values in a contiguous array.
18 | *
19 | * @private
20 | */
21 | function readAccessorPacked(gltf, accessor) {
22 | const byteStride = getAccessorByteStride(gltf, accessor);
23 | const componentTypeByteLength = ComponentDatatype.getSizeInBytes(
24 | accessor.componentType,
25 | );
26 | const numberOfComponents = numberOfComponentsForType(accessor.type);
27 | const count = accessor.count;
28 | const values = new Array(numberOfComponents * count);
29 |
30 | if (!defined(accessor.bufferView)) {
31 | return values.fill(0);
32 | }
33 |
34 | const bufferView = gltf.bufferViews[accessor.bufferView];
35 | const source = gltf.buffers[bufferView.buffer].extras._pipeline.source;
36 | let byteOffset =
37 | accessor.byteOffset + bufferView.byteOffset + source.byteOffset;
38 |
39 | const dataView = new DataView(source.buffer);
40 | const components = new Array(numberOfComponents);
41 | const componentReader = getComponentReader(accessor.componentType);
42 |
43 | for (let i = 0; i < count; ++i) {
44 | componentReader(
45 | dataView,
46 | byteOffset,
47 | numberOfComponents,
48 | componentTypeByteLength,
49 | components,
50 | );
51 | for (let j = 0; j < numberOfComponents; ++j) {
52 | values[i * numberOfComponents + j] = components[j];
53 | }
54 | byteOffset += byteStride;
55 | }
56 | return values;
57 | }
58 |
--------------------------------------------------------------------------------
/lib/readResources.js:
--------------------------------------------------------------------------------
1 | "use strict";
2 | const Cesium = require("cesium");
3 | const fsExtra = require("fs-extra");
4 | const path = require("path");
5 | const Promise = require("bluebird");
6 | const { URL } = require("url");
7 |
8 | const addPipelineExtras = require("./addPipelineExtras");
9 | const dataUriToBuffer = require("./dataUriToBuffer");
10 | const { fileURLToPath, pathToFileURL } = require("./FileUrl");
11 | const ForEach = require("./ForEach");
12 |
13 | const defined = Cesium.defined;
14 | const defaultValue = Cesium.defaultValue;
15 | const isDataUri = Cesium.isDataUri;
16 | const RuntimeError = Cesium.RuntimeError;
17 |
18 | module.exports = readResources;
19 |
20 | /**
21 | * Read data uris, buffer views, or files referenced by the glTF into buffers.
22 | * The buffer data is placed into extras._pipeline.source for the corresponding object.
23 | * This stage runs before updateVersion and handles both glTF 1.0 and glTF 2.0 assets.
24 | *
25 | * @param {object} gltf A javascript object containing a glTF asset.
26 | * @param {object} [options] Object with the following properties:
27 | * @param {string} [options.resourceDirectory] The path to look in when reading separate files.
28 | * @returns {Promise} A promise that resolves to the glTF asset when all resources are read.
29 | *
30 | * @private
31 | */
32 | function readResources(gltf, options) {
33 | addPipelineExtras(gltf);
34 | options = defaultValue(options, {});
35 |
36 | // Make sure its an absolute path with a trailing separator
37 | options.resourceDirectory = defined(options.resourceDirectory)
38 | ? path.resolve(options.resourceDirectory) + path.sep
39 | : undefined;
40 |
41 | const bufferPromises = [];
42 | const resourcePromises = [];
43 |
44 | ForEach.buffer(gltf, function (buffer) {
45 | bufferPromises.push(readBuffer(gltf, buffer, options));
46 | });
47 |
48 | // Buffers need to be read first because images and shader may resolve to bufferViews
49 | return Promise.all(bufferPromises)
50 | .then(function () {
51 | ForEach.shader(gltf, function (shader) {
52 | resourcePromises.push(readShader(gltf, shader, options));
53 | });
54 | ForEach.image(gltf, function (image) {
55 | resourcePromises.push(readImage(gltf, image, options));
56 | });
57 | return Promise.all(resourcePromises);
58 | })
59 | .then(function () {
60 | return gltf;
61 | });
62 | }
63 |
64 | function readBuffer(gltf, buffer, options) {
65 | return readResource(gltf, buffer, false, options).then(function (data) {
66 | if (defined(data)) {
67 | buffer.extras._pipeline.source = data;
68 | }
69 | });
70 | }
71 |
72 | function readImage(gltf, image, options) {
73 | return readResource(gltf, image, true, options).then(function (data) {
74 | image.extras._pipeline.source = data;
75 | });
76 | }
77 |
78 | function readShader(gltf, shader, options) {
79 | return readResource(gltf, shader, true, options).then(function (data) {
80 | shader.extras._pipeline.source = data.toString();
81 | });
82 | }
83 |
84 | function readResource(gltf, object, saveResourceId, options) {
85 | const uri = object.uri;
86 | delete object.uri; // Don't hold onto the uri, its contents will be stored in extras._pipeline.source
87 |
88 | // Source already exists if the gltf was converted from a glb
89 | const source = object.extras._pipeline.source;
90 | if (defined(source)) {
91 | return Promise.resolve(Buffer.from(source));
92 | }
93 | // Handle reading buffer view from 1.0 glb model
94 | const extensions = object.extensions;
95 | if (defined(extensions)) {
96 | const khrBinaryGltf = extensions.KHR_binary_glTF;
97 | if (defined(khrBinaryGltf)) {
98 | return Promise.resolve(
99 | readBufferView(gltf, khrBinaryGltf.bufferView, object, saveResourceId),
100 | );
101 | }
102 | }
103 | if (defined(object.bufferView)) {
104 | return Promise.resolve(
105 | readBufferView(gltf, object.bufferView, object, saveResourceId),
106 | );
107 | }
108 | if (!defined(uri)) {
109 | return Promise.resolve(undefined);
110 | }
111 | if (isDataUri(uri)) {
112 | return Promise.resolve(dataUriToBuffer(uri));
113 | }
114 | return readFile(object, uri, saveResourceId, options);
115 | }
116 |
117 | function readBufferView(gltf, bufferViewId, object, saveResourceId) {
118 | if (saveResourceId) {
119 | object.extras._pipeline.resourceId = bufferViewId;
120 | }
121 | const bufferView = gltf.bufferViews[bufferViewId];
122 | const buffer = gltf.buffers[bufferView.buffer];
123 | const source = buffer.extras._pipeline.source;
124 | const byteOffset = defaultValue(bufferView.byteOffset, 0);
125 | return source.slice(byteOffset, byteOffset + bufferView.byteLength);
126 | }
127 |
128 | function readFile(object, uri, saveResourceId, options) {
129 | const resourceDirectory = options.resourceDirectory;
130 | const hasResourceDirectory = defined(resourceDirectory);
131 |
132 | // Resolve the URL
133 | let absoluteUrl;
134 | try {
135 | absoluteUrl = new URL(
136 | uri,
137 | hasResourceDirectory ? pathToFileURL(resourceDirectory) : undefined,
138 | );
139 | } catch (error) {
140 | return Promise.reject(
141 | new RuntimeError(
142 | "glTF model references separate files but no resourceDirectory is supplied",
143 | ),
144 | );
145 | }
146 |
147 | // Generate file paths for the resource
148 | const absolutePath = fileURLToPath(absoluteUrl);
149 | const relativePath = hasResourceDirectory
150 | ? path.relative(resourceDirectory, absolutePath)
151 | : path.basename(absolutePath);
152 |
153 | if (!defined(object.name)) {
154 | const extension = path.extname(relativePath);
155 | object.name = path.basename(relativePath, extension);
156 | }
157 |
158 | if (saveResourceId) {
159 | object.extras._pipeline.resourceId = absolutePath;
160 | }
161 |
162 | object.extras._pipeline.absolutePath = absolutePath;
163 | object.extras._pipeline.relativePath = relativePath;
164 | return fsExtra.readFile(absolutePath);
165 | }
166 |
--------------------------------------------------------------------------------
/lib/removeDefaults.js:
--------------------------------------------------------------------------------
1 | "use strict";
2 | const Cesium = require("cesium");
3 | const ForEach = require("./ForEach");
4 |
5 | const defined = Cesium.defined;
6 | const Matrix4 = Cesium.Matrix4;
7 |
8 | module.exports = removeDefaults;
9 |
10 | /**
11 | * Remove default values from the glTF. Not exhaustive.
12 | *
13 | * @param {object} gltf A javascript object containing a glTF asset.
14 | * @returns {object} glTF with default values removed.
15 | *
16 | * @private
17 | */
18 | function removeDefaults(gltf) {
19 | ForEach.node(gltf, function (node) {
20 | if (
21 | defined(node.matrix) &&
22 | Matrix4.equals(Matrix4.fromArray(node.matrix), Matrix4.IDENTITY)
23 | ) {
24 | delete node.matrix;
25 | }
26 | });
27 | ForEach.accessor(gltf, function (accessor) {
28 | if (accessor.normalized === false) {
29 | delete accessor.normalized;
30 | }
31 | });
32 | return gltf;
33 | }
34 |
--------------------------------------------------------------------------------
/lib/removeExtension.js:
--------------------------------------------------------------------------------
1 | "use strict";
2 | const Cesium = require("cesium");
3 | const ForEach = require("./ForEach");
4 | const removeExtensionsUsed = require("./removeExtensionsUsed");
5 |
6 | const defined = Cesium.defined;
7 |
8 | module.exports = removeExtension;
9 |
10 | /**
11 | * Removes an extension from gltf.extensions, gltf.extensionsUsed, gltf.extensionsRequired, and any other objects in the glTF if it is present.
12 | *
13 | * @param {object} gltf A javascript object containing a glTF asset.
14 | * @param {string} extension The extension to remove.
15 | *
16 | * @returns {*} The extension data removed from gltf.extensions.
17 | */
18 | function removeExtension(gltf, extension) {
19 | removeExtensionsUsed(gltf, extension); // Also removes from extensionsRequired
20 |
21 | if (extension === "CESIUM_RTC") {
22 | removeCesiumRTC(gltf);
23 | }
24 |
25 | return removeExtensionAndTraverse(gltf, extension);
26 | }
27 |
28 | function removeCesiumRTC(gltf) {
29 | ForEach.technique(gltf, function (technique) {
30 | ForEach.techniqueUniform(technique, function (uniform) {
31 | if (uniform.semantic === "CESIUM_RTC_MODELVIEW") {
32 | uniform.semantic = "MODELVIEW";
33 | }
34 | });
35 | });
36 | }
37 |
38 | function removeExtensionAndTraverse(object, extension) {
39 | if (Array.isArray(object)) {
40 | const length = object.length;
41 | for (let i = 0; i < length; ++i) {
42 | removeExtensionAndTraverse(object[i], extension);
43 | }
44 | } else if (
45 | object !== null &&
46 | typeof object === "object" &&
47 | object.constructor === Object
48 | ) {
49 | const extensions = object.extensions;
50 | let extensionData;
51 | if (defined(extensions)) {
52 | extensionData = extensions[extension];
53 | if (defined(extensionData)) {
54 | delete extensions[extension];
55 | if (Object.keys(extensions).length === 0) {
56 | delete object.extensions;
57 | }
58 | }
59 | }
60 | for (const key in object) {
61 | if (Object.prototype.hasOwnProperty.call(object, key)) {
62 | removeExtensionAndTraverse(object[key], extension);
63 | }
64 | }
65 | return extensionData;
66 | }
67 | }
68 |
--------------------------------------------------------------------------------
/lib/removeExtensionsRequired.js:
--------------------------------------------------------------------------------
1 | "use strict";
2 | const Cesium = require("cesium");
3 |
4 | const defined = Cesium.defined;
5 |
6 | module.exports = removeExtensionsRequired;
7 |
8 | /**
9 | * Removes an extension from gltf.extensionsRequired if it is present.
10 | *
11 | * @param {object} gltf A javascript object containing a glTF asset.
12 | * @param {string} extension The extension to remove.
13 | *
14 | * @private
15 | */
16 | function removeExtensionsRequired(gltf, extension) {
17 | const extensionsRequired = gltf.extensionsRequired;
18 | if (defined(extensionsRequired)) {
19 | const index = extensionsRequired.indexOf(extension);
20 | if (index >= 0) {
21 | extensionsRequired.splice(index, 1);
22 | }
23 | if (extensionsRequired.length === 0) {
24 | delete gltf.extensionsRequired;
25 | }
26 | }
27 | }
28 |
--------------------------------------------------------------------------------
/lib/removeExtensionsUsed.js:
--------------------------------------------------------------------------------
1 | "use strict";
2 | const Cesium = require("cesium");
3 | const removeExtensionsRequired = require("./removeExtensionsRequired");
4 |
5 | const defined = Cesium.defined;
6 |
7 | module.exports = removeExtensionsUsed;
8 |
9 | /**
10 | * Removes an extension from gltf.extensionsUsed and gltf.extensionsRequired if it is present.
11 | *
12 | * @param {object} gltf A javascript object containing a glTF asset.
13 | * @param {string} extension The extension to remove.
14 | *
15 | * @private
16 | */
17 | function removeExtensionsUsed(gltf, extension) {
18 | const extensionsUsed = gltf.extensionsUsed;
19 | if (defined(extensionsUsed)) {
20 | const index = extensionsUsed.indexOf(extension);
21 | if (index >= 0) {
22 | extensionsUsed.splice(index, 1);
23 | }
24 | removeExtensionsRequired(gltf, extension);
25 | if (extensionsUsed.length === 0) {
26 | delete gltf.extensionsUsed;
27 | }
28 | }
29 | }
30 |
--------------------------------------------------------------------------------
/lib/removePipelineExtras.js:
--------------------------------------------------------------------------------
1 | "use strict";
2 | const Cesium = require("cesium");
3 | const ForEach = require("./ForEach");
4 |
5 | const defined = Cesium.defined;
6 |
7 | module.exports = removePipelineExtras;
8 |
9 | /**
10 | * Iterate through the objects within the glTF and delete their pipeline extras object.
11 | *
12 | * @param {object} gltf A javascript object containing a glTF asset.
13 | * @returns {object} glTF with no pipeline extras.
14 | *
15 | * @private
16 | */
17 | function removePipelineExtras(gltf) {
18 | ForEach.shader(gltf, function (shader) {
19 | removeExtras(shader);
20 | });
21 | ForEach.buffer(gltf, function (buffer) {
22 | removeExtras(buffer);
23 | });
24 | ForEach.image(gltf, function (image) {
25 | removeExtras(image);
26 | });
27 |
28 | removeExtras(gltf);
29 |
30 | return gltf;
31 | }
32 |
33 | function removeExtras(object) {
34 | if (!defined(object.extras)) {
35 | return;
36 | }
37 |
38 | if (defined(object.extras._pipeline)) {
39 | delete object.extras._pipeline;
40 | }
41 |
42 | if (Object.keys(object.extras).length === 0) {
43 | delete object.extras;
44 | }
45 | }
46 |
--------------------------------------------------------------------------------
/lib/updateAccessorComponentTypes.js:
--------------------------------------------------------------------------------
1 | "use strict";
2 | const Cesium = require("cesium");
3 | const addBuffer = require("./addBuffer");
4 | const ForEach = require("./ForEach");
5 | const readAccessorPacked = require("./readAccessorPacked");
6 |
7 | const ComponentDatatype = Cesium.ComponentDatatype;
8 | const WebGLConstants = Cesium.WebGLConstants;
9 |
10 | module.exports = updateAccessorComponentTypes;
11 |
12 | /**
13 | * Update accessors referenced by JOINTS_0 and WEIGHTS_0 attributes to use correct component types.
14 | *
15 | * @param {object} gltf A javascript object containing a glTF asset.
16 | * @returns {object} The glTF asset with compressed meshes.
17 | *
18 | * @private
19 | */
20 | function updateAccessorComponentTypes(gltf) {
21 | let componentType;
22 | ForEach.accessorWithSemantic(gltf, "JOINTS_0", function (accessorId) {
23 | const accessor = gltf.accessors[accessorId];
24 | componentType = accessor.componentType;
25 | if (componentType === WebGLConstants.BYTE) {
26 | convertType(gltf, accessor, ComponentDatatype.UNSIGNED_BYTE);
27 | } else if (
28 | componentType !== WebGLConstants.UNSIGNED_BYTE &&
29 | componentType !== WebGLConstants.UNSIGNED_SHORT
30 | ) {
31 | convertType(gltf, accessor, ComponentDatatype.UNSIGNED_SHORT);
32 | }
33 | });
34 | ForEach.accessorWithSemantic(gltf, "WEIGHTS_0", function (accessorId) {
35 | const accessor = gltf.accessors[accessorId];
36 | componentType = accessor.componentType;
37 | if (componentType === WebGLConstants.BYTE) {
38 | convertType(gltf, accessor, ComponentDatatype.UNSIGNED_BYTE);
39 | } else if (componentType === WebGLConstants.SHORT) {
40 | convertType(gltf, accessor, ComponentDatatype.UNSIGNED_SHORT);
41 | }
42 | });
43 |
44 | return gltf;
45 | }
46 |
47 | function convertType(gltf, accessor, updatedComponentType) {
48 | const typedArray = ComponentDatatype.createTypedArray(
49 | updatedComponentType,
50 | readAccessorPacked(gltf, accessor),
51 | );
52 | const newBuffer = new Uint8Array(typedArray.buffer);
53 | accessor.bufferView = addBuffer(gltf, newBuffer);
54 | accessor.componentType = updatedComponentType;
55 | accessor.byteOffset = 0;
56 | }
57 |
--------------------------------------------------------------------------------
/lib/usesExtension.js:
--------------------------------------------------------------------------------
1 | "use strict";
2 | const Cesium = require("cesium");
3 |
4 | const defined = Cesium.defined;
5 |
6 | module.exports = usesExtension;
7 |
8 | /**
9 | * Checks whether the glTF uses the given extension.
10 | *
11 | * @param {object} gltf A javascript object containing a glTF asset.
12 | * @param {string} extension The name of the extension.
13 | * @returns {boolean} Whether the glTF uses the given extension.
14 | *
15 | * @private
16 | */
17 | function usesExtension(gltf, extension) {
18 | return (
19 | defined(gltf.extensionsUsed) && gltf.extensionsUsed.indexOf(extension) >= 0
20 | );
21 | }
22 |
--------------------------------------------------------------------------------
/package.json:
--------------------------------------------------------------------------------
1 | {
2 | "name": "gltf-pipeline",
3 | "version": "4.1.0",
4 | "description": "Content pipeline tools for optimizing glTF assets.",
5 | "license": "Apache-2.0",
6 | "contributors": [
7 | {
8 | "name": "Richard Lee, Cesium GS, Inc., and Contributors",
9 | "url": "https://github.com/CesiumGS/gltf-pipeline/graphs/contributors"
10 | }
11 | ],
12 | "keywords": [
13 | "glTF",
14 | "WebGL"
15 | ],
16 | "homepage": "https://github.com/CesiumGS/gltf-pipeline",
17 | "repository": {
18 | "type": "git",
19 | "url": "git@github.com:CesiumGS/gltf-pipeline.git"
20 | },
21 | "bugs": {
22 | "url": "https://github.com/CesiumGS/gltf-pipeline/issues"
23 | },
24 | "main": "index.js",
25 | "engines": {
26 | "node": ">=16.0.0"
27 | },
28 | "dependencies": {
29 | "bluebird": "^3.7.2",
30 | "cesium": "^1.86.1",
31 | "draco3d": "^1.4.3",
32 | "fs-extra": "^11.0.0",
33 | "mime": "^3.0.0",
34 | "object-hash": "^3.0.0",
35 | "yargs": "^17.2.1"
36 | },
37 | "devDependencies": {
38 | "cloc": "^2.8.0",
39 | "dependency-tree": "^10.0.9",
40 | "eslint": "^8.0.1",
41 | "eslint-config-cesium": "^10.0.1",
42 | "eslint-config-prettier": "^9.0.0",
43 | "eslint-plugin-n": "^16.1.0",
44 | "gulp": "^4.0.2",
45 | "husky": "^8.0.3",
46 | "jasmine": "^5.0.0",
47 | "jasmine-spec-reporter": "^7.0.0",
48 | "jsdoc": "^4.0.0",
49 | "nyc": "^15.1.0",
50 | "prettier": "3.1.1",
51 | "lint-staged": "^15.0.2"
52 | },
53 | "lint-staged": {
54 | "*.(js|ts)": [
55 | "eslint --cache --quiet --fix",
56 | "prettier --write"
57 | ],
58 | "*.!(js|ts)": "prettier --write"
59 | },
60 | "scripts": {
61 | "prepare": "husky install",
62 | "pre-commit": "lint-staged",
63 | "jsdoc": "jsdoc ./lib -R ./README.md -d doc",
64 | "eslint-fix": "eslint \"./**/*.js\" --fix",
65 | "eslint": "eslint \"./**/*.js\" --cache --quiet",
66 | "eslint-watch": "gulp eslint-watch",
67 | "test": "gulp test",
68 | "test-watch": "gulp test-watch",
69 | "coverage": "gulp coverage",
70 | "cloc": "gulp cloc",
71 | "prettier": "prettier --write \"**/*\"",
72 | "prettier-check": "prettier --check \"**/*\"",
73 | "build-cesium": "gulp build-cesium",
74 | "generate-third-party": "gulp generate-third-party"
75 | },
76 | "bin": {
77 | "gltf-pipeline": "./bin/gltf-pipeline.js"
78 | }
79 | }
--------------------------------------------------------------------------------
/specs/.eslintrc.json:
--------------------------------------------------------------------------------
1 | {
2 | "extends": "../.eslintrc.json",
3 | "env": {
4 | "jasmine": true
5 | },
6 | "rules": {
7 | "no-restricted-globals": ["error", "fdescribe", "fit"]
8 | }
9 | }
10 |
--------------------------------------------------------------------------------
/specs/data/1.0/box-materials-common/box-materials-common.bin:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/CesiumGS/gltf-pipeline/cbbff2f212b5aa6163c7152d46ffcaf3a2ff3ac3/specs/data/1.0/box-materials-common/box-materials-common.bin
--------------------------------------------------------------------------------
/specs/data/1.0/box-materials-common/box-materials-common.gltf:
--------------------------------------------------------------------------------
1 | {
2 | "accessors": {
3 | "accessor_21": {
4 | "bufferView": "bufferView_29",
5 | "byteOffset": 0,
6 | "byteStride": 0,
7 | "componentType": 5123,
8 | "count": 36,
9 | "type": "SCALAR"
10 | },
11 | "accessor_23": {
12 | "bufferView": "bufferView_30",
13 | "byteOffset": 0,
14 | "byteStride": 12,
15 | "componentType": 5126,
16 | "count": 24,
17 | "max": [
18 | 0.5,
19 | 0.5,
20 | 0.5
21 | ],
22 | "min": [
23 | -0.5,
24 | -0.5,
25 | -0.5
26 | ],
27 | "type": "VEC3"
28 | },
29 | "accessor_25": {
30 | "bufferView": "bufferView_30",
31 | "byteOffset": 288,
32 | "byteStride": 12,
33 | "componentType": 5126,
34 | "count": 24,
35 | "max": [
36 | 1,
37 | 1,
38 | 1
39 | ],
40 | "min": [
41 | -1,
42 | -1,
43 | -1
44 | ],
45 | "type": "VEC3"
46 | }
47 | },
48 | "animations": {},
49 | "asset": {
50 | "generator": "collada2gltf@027f74366341d569dea42e9a68b7104cc3892054",
51 | "premultipliedAlpha": true,
52 | "profile": {
53 | "api": "WebGL",
54 | "version": "1.0.2"
55 | },
56 | "version": "1.0"
57 | },
58 | "bufferViews": {
59 | "bufferView_29": {
60 | "buffer": "Box",
61 | "byteLength": 72,
62 | "byteOffset": 0,
63 | "target": 34963
64 | },
65 | "bufferView_30": {
66 | "buffer": "Box",
67 | "byteLength": 576,
68 | "byteOffset": 72,
69 | "target": 34962
70 | }
71 | },
72 | "buffers": {
73 | "Box": {
74 | "byteLength": 648,
75 | "type": "arraybuffer",
76 | "uri": "box-materials-common.bin"
77 | }
78 | },
79 | "extensionsUsed": [
80 | "KHR_materials_common"
81 | ],
82 | "materials": {
83 | "Effect-Red": {
84 | "extensions": {
85 | "KHR_materials_common": {
86 | "doubleSided": false,
87 | "jointCount": 0,
88 | "technique": "PHONG",
89 | "transparent": false,
90 | "values": {
91 | "diffuse": [
92 | 0.8,
93 | 0,
94 | 0,
95 | 1
96 | ],
97 | "shininess": 256,
98 | "specular": [
99 | 0.2,
100 | 0.2,
101 | 0.2,
102 | 1
103 | ]
104 | }
105 | }
106 | },
107 | "name": "Red"
108 | }
109 | },
110 | "meshes": {
111 | "Geometry-mesh002": {
112 | "name": "Mesh",
113 | "primitives": [
114 | {
115 | "attributes": {
116 | "NORMAL": "accessor_25",
117 | "POSITION": "accessor_23"
118 | },
119 | "indices": "accessor_21",
120 | "material": "Effect-Red",
121 | "mode": 4
122 | }
123 | ]
124 | }
125 | },
126 | "nodes": {
127 | "Geometry-mesh002Node": {
128 | "children": [],
129 | "matrix": [
130 | 1,
131 | 0,
132 | 0,
133 | 0,
134 | 0,
135 | 1,
136 | 0,
137 | 0,
138 | 0,
139 | 0,
140 | 1,
141 | 0,
142 | 0,
143 | 0,
144 | 0,
145 | 1
146 | ],
147 | "meshes": [
148 | "Geometry-mesh002"
149 | ],
150 | "name": "Mesh"
151 | },
152 | "node_1": {
153 | "children": [
154 | "Geometry-mesh002Node"
155 | ],
156 | "matrix": [
157 | 1,
158 | 0,
159 | 0,
160 | 0,
161 | 0,
162 | 0,
163 | -1,
164 | 0,
165 | 0,
166 | 1,
167 | 0,
168 | 0,
169 | 0,
170 | 0,
171 | 0,
172 | 1
173 | ],
174 | "name": "Y_UP_Transform"
175 | }
176 | },
177 | "scene": "defaultScene",
178 | "scenes": {
179 | "defaultScene": {
180 | "nodes": [
181 | "node_1"
182 | ]
183 | }
184 | },
185 | "skins": {}
186 | }
--------------------------------------------------------------------------------
/specs/data/1.0/box-textured-binary-separate/box-textured-binary-separate-fs.glsl:
--------------------------------------------------------------------------------
1 | precision highp float;
2 | varying vec3 v_normal;
3 | varying vec2 v_texcoord0;
4 | uniform sampler2D u_diffuse;
5 | uniform vec4 u_specular;
6 | uniform float u_shininess;
7 | void main(void) {
8 | vec3 normal = normalize(v_normal);
9 | vec4 color = vec4(0., 0., 0., 0.);
10 | vec4 diffuse = vec4(0., 0., 0., 1.);
11 | vec4 specular;
12 | diffuse = texture2D(u_diffuse, v_texcoord0);
13 | specular = u_specular;
14 | diffuse.xyz *= max(dot(normal,vec3(0.,0.,1.)), 0.);
15 | color.xyz += diffuse.xyz;
16 | color = vec4(color.rgb * diffuse.a, diffuse.a);
17 | gl_FragColor = color;
18 | }
19 |
--------------------------------------------------------------------------------
/specs/data/1.0/box-textured-binary-separate/box-textured-binary-separate-vs.glsl:
--------------------------------------------------------------------------------
1 | precision highp float;
2 | attribute vec3 a_position;
3 | attribute vec3 a_normal;
4 | varying vec3 v_normal;
5 | uniform mat3 u_normalMatrix;
6 | uniform mat4 u_modelViewMatrix;
7 | uniform mat4 u_projectionMatrix;
8 | attribute vec2 a_texcoord0;
9 | varying vec2 v_texcoord0;
10 | void main(void) {
11 | vec4 pos = u_modelViewMatrix * vec4(a_position,1.0);
12 | v_normal = u_normalMatrix * a_normal;
13 | v_texcoord0 = a_texcoord0;
14 | gl_Position = u_projectionMatrix * pos;
15 | }
16 |
--------------------------------------------------------------------------------
/specs/data/1.0/box-textured-binary-separate/box-textured-binary-separate.glb:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/CesiumGS/gltf-pipeline/cbbff2f212b5aa6163c7152d46ffcaf3a2ff3ac3/specs/data/1.0/box-textured-binary-separate/box-textured-binary-separate.glb
--------------------------------------------------------------------------------
/specs/data/1.0/box-textured-binary-separate/cesium.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/CesiumGS/gltf-pipeline/cbbff2f212b5aa6163c7152d46ffcaf3a2ff3ac3/specs/data/1.0/box-textured-binary-separate/cesium.png
--------------------------------------------------------------------------------
/specs/data/1.0/box-textured-binary/box-textured-binary.glb:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/CesiumGS/gltf-pipeline/cbbff2f212b5aa6163c7152d46ffcaf3a2ff3ac3/specs/data/1.0/box-textured-binary/box-textured-binary.glb
--------------------------------------------------------------------------------
/specs/data/1.0/box-textured-materials-common/CesiumLogoFlat.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/CesiumGS/gltf-pipeline/cbbff2f212b5aa6163c7152d46ffcaf3a2ff3ac3/specs/data/1.0/box-textured-materials-common/CesiumLogoFlat.png
--------------------------------------------------------------------------------
/specs/data/1.0/box-textured-materials-common/box-textured-materials-common.bin:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/CesiumGS/gltf-pipeline/cbbff2f212b5aa6163c7152d46ffcaf3a2ff3ac3/specs/data/1.0/box-textured-materials-common/box-textured-materials-common.bin
--------------------------------------------------------------------------------
/specs/data/1.0/box-textured-materials-common/box-textured-materials-common.gltf:
--------------------------------------------------------------------------------
1 | {
2 | "accessors": {
3 | "accessor_21": {
4 | "bufferView": "bufferView_29",
5 | "byteOffset": 0,
6 | "byteStride": 0,
7 | "componentType": 5123,
8 | "count": 36,
9 | "type": "SCALAR"
10 | },
11 | "accessor_23": {
12 | "bufferView": "bufferView_30",
13 | "byteOffset": 0,
14 | "byteStride": 12,
15 | "componentType": 5126,
16 | "count": 24,
17 | "max": [
18 | 0.5,
19 | 0.5,
20 | 0.5
21 | ],
22 | "min": [
23 | -0.5,
24 | -0.5,
25 | -0.5
26 | ],
27 | "type": "VEC3"
28 | },
29 | "accessor_25": {
30 | "bufferView": "bufferView_30",
31 | "byteOffset": 288,
32 | "byteStride": 12,
33 | "componentType": 5126,
34 | "count": 24,
35 | "max": [
36 | 1,
37 | 1,
38 | 1
39 | ],
40 | "min": [
41 | -1,
42 | -1,
43 | -1
44 | ],
45 | "type": "VEC3"
46 | },
47 | "accessor_27": {
48 | "bufferView": "bufferView_30",
49 | "byteOffset": 576,
50 | "byteStride": 8,
51 | "componentType": 5126,
52 | "count": 24,
53 | "max": [
54 | 6,
55 | 1
56 | ],
57 | "min": [
58 | 0,
59 | 0
60 | ],
61 | "type": "VEC2"
62 | }
63 | },
64 | "animations": {},
65 | "asset": {
66 | "generator": "collada2gltf@027f74366341d569dea42e9a68b7104cc3892054",
67 | "premultipliedAlpha": true,
68 | "profile": {
69 | "api": "WebGL",
70 | "version": "1.0.2"
71 | },
72 | "version": "1.0"
73 | },
74 | "bufferViews": {
75 | "bufferView_29": {
76 | "buffer": "BoxTextured",
77 | "byteLength": 72,
78 | "byteOffset": 0,
79 | "target": 34963
80 | },
81 | "bufferView_30": {
82 | "buffer": "BoxTextured",
83 | "byteLength": 768,
84 | "byteOffset": 72,
85 | "target": 34962
86 | }
87 | },
88 | "buffers": {
89 | "BoxTextured": {
90 | "byteLength": 840,
91 | "type": "arraybuffer",
92 | "uri": "box-textured-materials-common.bin"
93 | }
94 | },
95 | "extensionsUsed": [
96 | "KHR_materials_common"
97 | ],
98 | "images": {
99 | "Image0001": {
100 | "name": "Image0001",
101 | "uri": "CesiumLogoFlat.png"
102 | }
103 | },
104 | "materials": {
105 | "Effect-Texture": {
106 | "extensions": {
107 | "KHR_materials_common": {
108 | "doubleSided": false,
109 | "jointCount": 0,
110 | "technique": "PHONG",
111 | "transparent": false,
112 | "values": {
113 | "diffuse": "texture_Image0001",
114 | "shininess": 256,
115 | "specular": [
116 | 0.2,
117 | 0.2,
118 | 0.2,
119 | 1
120 | ]
121 | }
122 | }
123 | },
124 | "name": "Texture"
125 | }
126 | },
127 | "meshes": {
128 | "Geometry-mesh002": {
129 | "name": "Mesh",
130 | "primitives": [
131 | {
132 | "attributes": {
133 | "NORMAL": "accessor_25",
134 | "POSITION": "accessor_23",
135 | "TEXCOORD_0": "accessor_27"
136 | },
137 | "indices": "accessor_21",
138 | "material": "Effect-Texture",
139 | "mode": 4
140 | }
141 | ]
142 | }
143 | },
144 | "nodes": {
145 | "Geometry-mesh002Node": {
146 | "children": [],
147 | "matrix": [
148 | 1,
149 | 0,
150 | 0,
151 | 0,
152 | 0,
153 | 1,
154 | 0,
155 | 0,
156 | 0,
157 | 0,
158 | 1,
159 | 0,
160 | 0,
161 | 0,
162 | 0,
163 | 1
164 | ],
165 | "meshes": [
166 | "Geometry-mesh002"
167 | ],
168 | "name": "Mesh"
169 | },
170 | "groupLocator030Node": {
171 | "children": [
172 | "txtrLocator026Node"
173 | ],
174 | "matrix": [
175 | 1,
176 | 0,
177 | 0,
178 | 0,
179 | 0,
180 | 1,
181 | 0,
182 | 0,
183 | 0,
184 | 0,
185 | 1,
186 | 0,
187 | 0,
188 | 0,
189 | 0,
190 | 1
191 | ],
192 | "name": "Texture_Group"
193 | },
194 | "node_3": {
195 | "children": [
196 | "Geometry-mesh002Node",
197 | "groupLocator030Node"
198 | ],
199 | "matrix": [
200 | 1,
201 | 0,
202 | 0,
203 | 0,
204 | 0,
205 | 0,
206 | -1,
207 | 0,
208 | 0,
209 | 1,
210 | 0,
211 | 0,
212 | 0,
213 | 0,
214 | 0,
215 | 1
216 | ],
217 | "name": "Y_UP_Transform"
218 | },
219 | "txtrLocator026Node": {
220 | "children": [],
221 | "matrix": [
222 | 1,
223 | 0,
224 | 0,
225 | 0,
226 | 0,
227 | 1,
228 | 0,
229 | 0,
230 | 0,
231 | 0,
232 | 1,
233 | 0,
234 | 0,
235 | 0,
236 | 0,
237 | 1
238 | ],
239 | "name": "Cesium_Logo_Flat__Image___Texture_"
240 | }
241 | },
242 | "samplers": {
243 | "sampler_0": {
244 | "magFilter": 9729,
245 | "minFilter": 9987,
246 | "wrapS": 10497,
247 | "wrapT": 10497
248 | }
249 | },
250 | "scene": "defaultScene",
251 | "scenes": {
252 | "defaultScene": {
253 | "nodes": [
254 | "node_3"
255 | ]
256 | }
257 | },
258 | "skins": {},
259 | "textures": {
260 | "texture_Image0001": {
261 | "format": 6408,
262 | "internalFormat": 6408,
263 | "sampler": "sampler_0",
264 | "source": "Image0001",
265 | "target": 3553,
266 | "type": 5121
267 | }
268 | }
269 | }
--------------------------------------------------------------------------------
/specs/data/1.0/box-textured-separate/box-textured-separate-fs.glsl:
--------------------------------------------------------------------------------
1 | precision highp float;
2 | varying vec3 v_normal;
3 | varying vec2 v_texcoord0;
4 | uniform sampler2D u_diffuse;
5 | uniform vec4 u_specular;
6 | uniform float u_shininess;
7 | void main(void) {
8 | vec3 normal = normalize(v_normal);
9 | vec4 color = vec4(0., 0., 0., 0.);
10 | vec4 diffuse = vec4(0., 0., 0., 1.);
11 | vec4 specular;
12 | diffuse = texture2D(u_diffuse, v_texcoord0);
13 | specular = u_specular;
14 | diffuse.xyz *= max(dot(normal,vec3(0.,0.,1.)), 0.);
15 | color.xyz += diffuse.xyz;
16 | color = vec4(color.rgb * diffuse.a, diffuse.a);
17 | gl_FragColor = color;
18 | }
19 |
--------------------------------------------------------------------------------
/specs/data/1.0/box-textured-separate/box-textured-separate-vs.glsl:
--------------------------------------------------------------------------------
1 | precision highp float;
2 | attribute vec3 a_position;
3 | attribute vec3 a_normal;
4 | varying vec3 v_normal;
5 | uniform mat3 u_normalMatrix;
6 | uniform mat4 u_modelViewMatrix;
7 | uniform mat4 u_projectionMatrix;
8 | attribute vec2 a_texcoord0;
9 | varying vec2 v_texcoord0;
10 | void main(void) {
11 | vec4 pos = u_modelViewMatrix * vec4(a_position,1.0);
12 | v_normal = u_normalMatrix * a_normal;
13 | v_texcoord0 = a_texcoord0;
14 | gl_Position = u_projectionMatrix * pos;
15 | }
16 |
--------------------------------------------------------------------------------
/specs/data/1.0/box-textured-separate/box-textured-separate.bin:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/CesiumGS/gltf-pipeline/cbbff2f212b5aa6163c7152d46ffcaf3a2ff3ac3/specs/data/1.0/box-textured-separate/box-textured-separate.bin
--------------------------------------------------------------------------------
/specs/data/1.0/box-textured-separate/cesium.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/CesiumGS/gltf-pipeline/cbbff2f212b5aa6163c7152d46ffcaf3a2ff3ac3/specs/data/1.0/box-textured-separate/cesium.png
--------------------------------------------------------------------------------
/specs/data/2.0/box-shared-image-references-separate/CesiumTexturedBoxTest0FS.glsl:
--------------------------------------------------------------------------------
1 | precision highp float;
2 | varying vec3 v_normal;
3 | varying vec2 v_texcoord0;
4 | uniform sampler2D u_diffuse;
5 | uniform vec4 u_specular;
6 | uniform float u_shininess;
7 | void main(void) {
8 | vec3 normal = normalize(v_normal);
9 | vec4 color = vec4(0., 0., 0., 0.);
10 | vec4 diffuse = vec4(0., 0., 0., 1.);
11 | vec4 specular;
12 | diffuse = texture2D(u_diffuse, v_texcoord0);
13 | specular = u_specular;
14 | diffuse.xyz *= max(dot(normal,vec3(0.,0.,1.)), 0.);
15 | color.xyz += diffuse.xyz;
16 | color = vec4(color.rgb * diffuse.a, diffuse.a);
17 | gl_FragColor = color;
18 | }
19 |
--------------------------------------------------------------------------------
/specs/data/2.0/box-shared-image-references-separate/CesiumTexturedBoxTest0VS.glsl:
--------------------------------------------------------------------------------
1 | precision highp float;
2 | attribute vec3 a_position;
3 | attribute vec3 a_normal;
4 | varying vec3 v_normal;
5 | uniform mat3 u_normalMatrix;
6 | uniform mat4 u_modelViewMatrix;
7 | uniform mat4 u_projectionMatrix;
8 | attribute vec2 a_texcoord0;
9 | varying vec2 v_texcoord0;
10 | void main(void) {
11 | vec4 pos = u_modelViewMatrix * vec4(a_position,1.0);
12 | v_normal = u_normalMatrix * a_normal;
13 | v_texcoord0 = a_texcoord0;
14 | gl_Position = u_projectionMatrix * pos;
15 | }
16 |
--------------------------------------------------------------------------------
/specs/data/2.0/box-shared-image-references-separate/Image.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/CesiumGS/gltf-pipeline/cbbff2f212b5aa6163c7152d46ffcaf3a2ff3ac3/specs/data/2.0/box-shared-image-references-separate/Image.png
--------------------------------------------------------------------------------
/specs/data/2.0/box-shared-image-references-separate/box-shared-image-references-separate.bin:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/CesiumGS/gltf-pipeline/cbbff2f212b5aa6163c7152d46ffcaf3a2ff3ac3/specs/data/2.0/box-shared-image-references-separate/box-shared-image-references-separate.bin
--------------------------------------------------------------------------------
/specs/data/2.0/box-shared-image-references-separate/box-shared-image-references-separate.gltf:
--------------------------------------------------------------------------------
1 | {
2 | "accessors": [
3 | {
4 | "bufferView": 2,
5 | "byteOffset": 0,
6 | "componentType": 5123,
7 | "count": 36,
8 | "type": "SCALAR",
9 | "name": "accessor_indices"
10 | },
11 | {
12 | "bufferView": 0,
13 | "byteOffset": 0,
14 | "componentType": 5126,
15 | "count": 24,
16 | "max": [
17 | 0.5,
18 | 0.5,
19 | 0.5
20 | ],
21 | "min": [
22 | -0.5,
23 | -0.5,
24 | -0.5
25 | ],
26 | "type": "VEC3",
27 | "name": "accessor_positions"
28 | },
29 | {
30 | "bufferView": 0,
31 | "byteOffset": 288,
32 | "componentType": 5126,
33 | "count": 24,
34 | "max": [
35 | 1,
36 | 1,
37 | 1
38 | ],
39 | "min": [
40 | -1,
41 | -1,
42 | -1
43 | ],
44 | "type": "VEC3",
45 | "name": "accessor_normals"
46 | },
47 | {
48 | "bufferView": 1,
49 | "byteOffset": 0,
50 | "componentType": 5126,
51 | "count": 24,
52 | "max": [
53 | 6,
54 | 1
55 | ],
56 | "min": [
57 | 0,
58 | 0
59 | ],
60 | "type": "VEC2",
61 | "name": "accessor_texcoord"
62 | }
63 | ],
64 | "asset": {
65 | "generator": "collada2gltf@ceec062e3d5793f2f249f53cbd843aee382ad40b",
66 | "version": "2.0"
67 | },
68 | "bufferViews": [
69 | {
70 | "buffer": 0,
71 | "byteLength": 576,
72 | "byteOffset": 0,
73 | "target": 34962,
74 | "name": "bufferView_0",
75 | "byteStride": 12
76 | },
77 | {
78 | "buffer": 0,
79 | "byteLength": 192,
80 | "byteOffset": 576,
81 | "target": 34962,
82 | "name": "bufferView_0",
83 | "byteStride": 8
84 | },
85 | {
86 | "buffer": 0,
87 | "byteLength": 72,
88 | "byteOffset": 768,
89 | "target": 34963,
90 | "name": "bufferView_1"
91 | }
92 | ],
93 | "buffers": [
94 | {
95 | "name": "binary_glTF",
96 | "byteLength": 840,
97 | "uri": "box-shared-image-references-separate.bin"
98 | }
99 | ],
100 | "images": [
101 | {
102 | "mimeType": "image/png",
103 | "uri": "Image.png"
104 | },
105 | {
106 | "mimeType": "image/png",
107 | "uri": "Image.png"
108 | }
109 | ],
110 | "materials": [
111 | {
112 | "name": "Texture",
113 | "extensions": {
114 | "KHR_techniques_webgl": {
115 | "technique": 0,
116 | "values": {
117 | "u_diffuse": {
118 | "index": 0,
119 | "texCoord": 0
120 | },
121 | "u_shininess": 256,
122 | "u_specular": [
123 | 0.2,
124 | 0.2,
125 | 0.2,
126 | 1
127 | ]
128 | }
129 | }
130 | },
131 | "emissiveFactor": [
132 | 0,
133 | 0,
134 | 0
135 | ],
136 | "alphaMode": "OPAQUE",
137 | "doubleSided": false
138 | }
139 | ],
140 | "meshes": [
141 | {
142 | "name": "Mesh",
143 | "primitives": [
144 | {
145 | "attributes": {
146 | "NORMAL": 2,
147 | "POSITION": 1,
148 | "TEXCOORD_0": 3
149 | },
150 | "indices": 0,
151 | "material": 0,
152 | "mode": 4
153 | }
154 | ]
155 | }
156 | ],
157 | "nodes": [
158 | {
159 | "name": "rootNode",
160 | "mesh": 0
161 | }
162 | ],
163 | "samplers": [
164 | {
165 | "magFilter": 9729,
166 | "minFilter": 9987,
167 | "wrapS": 10497,
168 | "wrapT": 10497,
169 | "name": "sampler_0"
170 | }
171 | ],
172 | "scene": 0,
173 | "scenes": [
174 | {
175 | "nodes": [
176 | 0
177 | ],
178 | "name": "defaultScene"
179 | }
180 | ],
181 | "textures": [
182 | {
183 | "sampler": 0,
184 | "source": 0,
185 | "name": "texture_Image0001"
186 | },
187 | {
188 | "sampler": 0,
189 | "source": 1,
190 | "name": "texture_Image0002"
191 | }
192 | ],
193 | "extensionsRequired": [
194 | "KHR_techniques_webgl"
195 | ],
196 | "extensions": {
197 | "KHR_techniques_webgl": {
198 | "programs": [
199 | {
200 | "name": "program_0",
201 | "fragmentShader": 0,
202 | "vertexShader": 1
203 | }
204 | ],
205 | "shaders": [
206 | {
207 | "type": 35632,
208 | "name": "CesiumTexturedBoxTest0FS",
209 | "uri": "CesiumTexturedBoxTest0FS.glsl"
210 | },
211 | {
212 | "type": 35633,
213 | "name": "CesiumTexturedBoxTest0VS",
214 | "uri": "CesiumTexturedBoxTest0VS.glsl"
215 | }
216 | ],
217 | "techniques": [
218 | {
219 | "name": "technique0",
220 | "program": 0,
221 | "attributes": {
222 | "a_normal": {
223 | "semantic": "NORMAL"
224 | },
225 | "a_position": {
226 | "semantic": "POSITION"
227 | },
228 | "a_texcoord0": {
229 | "semantic": "TEXCOORD_0"
230 | }
231 | },
232 | "uniforms": {
233 | "u_diffuse": {
234 | "type": 35678
235 | },
236 | "u_modelViewMatrix": {
237 | "type": 35676,
238 | "semantic": "MODELVIEW"
239 | },
240 | "u_normalMatrix": {
241 | "type": 35675,
242 | "semantic": "MODELVIEWINVERSETRANSPOSE"
243 | },
244 | "u_projectionMatrix": {
245 | "type": 35676,
246 | "semantic": "PROJECTION"
247 | },
248 | "u_shininess": {
249 | "type": 5126
250 | },
251 | "u_specular": {
252 | "type": 35666
253 | }
254 | }
255 | }
256 | ]
257 | }
258 | },
259 | "extensionsUsed": [
260 | "KHR_techniques_webgl"
261 | ]
262 | }
263 |
--------------------------------------------------------------------------------
/specs/data/2.0/box-techniques-separate/CesiumTexturedBoxTest.bin:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/CesiumGS/gltf-pipeline/cbbff2f212b5aa6163c7152d46ffcaf3a2ff3ac3/specs/data/2.0/box-techniques-separate/CesiumTexturedBoxTest.bin
--------------------------------------------------------------------------------
/specs/data/2.0/box-techniques-separate/CesiumTexturedBoxTest0FS.glsl:
--------------------------------------------------------------------------------
1 | precision highp float;
2 | varying vec3 v_normal;
3 | varying vec2 v_texcoord0;
4 | uniform sampler2D u_diffuse;
5 | uniform vec4 u_specular;
6 | uniform float u_shininess;
7 | void main(void) {
8 | vec3 normal = normalize(v_normal);
9 | vec4 color = vec4(0., 0., 0., 0.);
10 | vec4 diffuse = vec4(0., 0., 0., 1.);
11 | vec4 specular;
12 | diffuse = texture2D(u_diffuse, v_texcoord0);
13 | specular = u_specular;
14 | diffuse.xyz *= max(dot(normal,vec3(0.,0.,1.)), 0.);
15 | color.xyz += diffuse.xyz;
16 | color = vec4(color.rgb * diffuse.a, diffuse.a);
17 | gl_FragColor = color;
18 | }
19 |
--------------------------------------------------------------------------------
/specs/data/2.0/box-techniques-separate/CesiumTexturedBoxTest0VS.glsl:
--------------------------------------------------------------------------------
1 | precision highp float;
2 | attribute vec3 a_position;
3 | attribute vec3 a_normal;
4 | varying vec3 v_normal;
5 | uniform mat3 u_normalMatrix;
6 | uniform mat4 u_modelViewMatrix;
7 | uniform mat4 u_projectionMatrix;
8 | attribute vec2 a_texcoord0;
9 | varying vec2 v_texcoord0;
10 | void main(void) {
11 | vec4 pos = u_modelViewMatrix * vec4(a_position,1.0);
12 | v_normal = u_normalMatrix * a_normal;
13 | v_texcoord0 = a_texcoord0;
14 | gl_Position = u_projectionMatrix * pos;
15 | }
16 |
--------------------------------------------------------------------------------
/specs/data/2.0/box-techniques-separate/Image0001.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/CesiumGS/gltf-pipeline/cbbff2f212b5aa6163c7152d46ffcaf3a2ff3ac3/specs/data/2.0/box-techniques-separate/Image0001.png
--------------------------------------------------------------------------------
/specs/data/2.0/box-techniques-separate/box-techniques-separate.gltf:
--------------------------------------------------------------------------------
1 | {
2 | "accessors": [
3 | {
4 | "bufferView": 2,
5 | "byteOffset": 0,
6 | "componentType": 5123,
7 | "count": 36,
8 | "type": "SCALAR",
9 | "name": "accessor_indices",
10 | "min": [
11 | 0
12 | ],
13 | "max": [
14 | 23
15 | ]
16 | },
17 | {
18 | "bufferView": 0,
19 | "byteOffset": 0,
20 | "componentType": 5126,
21 | "count": 24,
22 | "max": [
23 | 0.5,
24 | 0.5,
25 | 0.5
26 | ],
27 | "min": [
28 | -0.5,
29 | -0.5,
30 | -0.5
31 | ],
32 | "type": "VEC3",
33 | "name": "accessor_positions"
34 | },
35 | {
36 | "bufferView": 0,
37 | "byteOffset": 288,
38 | "componentType": 5126,
39 | "count": 24,
40 | "max": [
41 | 1,
42 | 1,
43 | 1
44 | ],
45 | "min": [
46 | -1,
47 | -1,
48 | -1
49 | ],
50 | "type": "VEC3",
51 | "name": "accessor_normals"
52 | },
53 | {
54 | "bufferView": 1,
55 | "byteOffset": 0,
56 | "componentType": 5126,
57 | "count": 24,
58 | "max": [
59 | 6,
60 | 1
61 | ],
62 | "min": [
63 | 0,
64 | 0
65 | ],
66 | "type": "VEC2",
67 | "name": "accessor_texcoord"
68 | }
69 | ],
70 | "asset": {
71 | "generator": "collada2gltf@ceec062e3d5793f2f249f53cbd843aee382ad40b",
72 | "version": "2.0"
73 | },
74 | "bufferViews": [
75 | {
76 | "buffer": 0,
77 | "byteLength": 576,
78 | "byteOffset": 0,
79 | "target": 34962,
80 | "name": "bufferView_0",
81 | "byteStride": 12
82 | },
83 | {
84 | "buffer": 0,
85 | "byteLength": 192,
86 | "byteOffset": 576,
87 | "target": 34962,
88 | "name": "bufferView_0",
89 | "byteStride": 8
90 | },
91 | {
92 | "buffer": 0,
93 | "byteLength": 72,
94 | "byteOffset": 768,
95 | "target": 34963,
96 | "name": "bufferView_1"
97 | }
98 | ],
99 | "buffers": [
100 | {
101 | "name": "CesiumTexturedBoxTest",
102 | "byteLength": 840,
103 | "uri": "CesiumTexturedBoxTest.bin"
104 | }
105 | ],
106 | "images": [
107 | {
108 | "name": "Image0001",
109 | "uri": "Image0001.png"
110 | }
111 | ],
112 | "materials": [
113 | {
114 | "name": "Texture",
115 | "extensions": {
116 | "KHR_techniques_webgl": {
117 | "technique": 0,
118 | "values": {
119 | "u_diffuse": {
120 | "index": 0,
121 | "texCoord": 0
122 | },
123 | "u_shininess": 256,
124 | "u_specular": [
125 | 0.2,
126 | 0.2,
127 | 0.2,
128 | 1
129 | ]
130 | }
131 | }
132 | },
133 | "emissiveFactor": [
134 | 0,
135 | 0,
136 | 0
137 | ],
138 | "alphaMode": "OPAQUE",
139 | "doubleSided": false
140 | }
141 | ],
142 | "meshes": [
143 | {
144 | "name": "Mesh",
145 | "primitives": [
146 | {
147 | "attributes": {
148 | "NORMAL": 2,
149 | "POSITION": 1,
150 | "TEXCOORD_0": 3
151 | },
152 | "indices": 0,
153 | "material": 0,
154 | "mode": 4
155 | }
156 | ]
157 | }
158 | ],
159 | "nodes": [
160 | {
161 | "name": "rootNode",
162 | "mesh": 0
163 | }
164 | ],
165 | "samplers": [
166 | {
167 | "magFilter": 9729,
168 | "minFilter": 9987,
169 | "wrapS": 10497,
170 | "wrapT": 10497,
171 | "name": "sampler_0"
172 | }
173 | ],
174 | "scene": 0,
175 | "scenes": [
176 | {
177 | "nodes": [
178 | 0
179 | ],
180 | "name": "defaultScene"
181 | }
182 | ],
183 | "textures": [
184 | {
185 | "sampler": 0,
186 | "source": 0,
187 | "name": "texture_Image0001"
188 | }
189 | ],
190 | "extensionsUsed": [
191 | "KHR_techniques_webgl"
192 | ],
193 | "extensionsRequired": [
194 | "KHR_techniques_webgl"
195 | ],
196 | "extensions": {
197 | "KHR_techniques_webgl": {
198 | "programs": [
199 | {
200 | "name": "program_0",
201 | "fragmentShader": 0,
202 | "vertexShader": 1
203 | }
204 | ],
205 | "shaders": [
206 | {
207 | "type": 35632,
208 | "name": "CesiumTexturedBoxTest0FS",
209 | "uri": "CesiumTexturedBoxTest0FS.glsl"
210 | },
211 | {
212 | "type": 35633,
213 | "name": "CesiumTexturedBoxTest0VS",
214 | "uri": "CesiumTexturedBoxTest0VS.glsl"
215 | }
216 | ],
217 | "techniques": [
218 | {
219 | "name": "technique0",
220 | "program": 0,
221 | "attributes": {
222 | "a_normal": {
223 | "semantic": "NORMAL"
224 | },
225 | "a_position": {
226 | "semantic": "POSITION"
227 | },
228 | "a_texcoord0": {
229 | "semantic": "TEXCOORD_0"
230 | }
231 | },
232 | "uniforms": {
233 | "u_diffuse": {
234 | "type": 35678
235 | },
236 | "u_modelViewMatrix": {
237 | "type": 35676,
238 | "semantic": "MODELVIEW"
239 | },
240 | "u_normalMatrix": {
241 | "type": 35675,
242 | "semantic": "MODELVIEWINVERSETRANSPOSE"
243 | },
244 | "u_projectionMatrix": {
245 | "type": 35676,
246 | "semantic": "PROJECTION"
247 | },
248 | "u_shininess": {
249 | "type": 5126
250 | },
251 | "u_specular": {
252 | "type": 35666
253 | }
254 | }
255 | }
256 | ]
257 | }
258 | }
259 | }
260 |
--------------------------------------------------------------------------------
/specs/data/2.0/box-textured-binary-separate/box-textured-binary-separate.glb:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/CesiumGS/gltf-pipeline/cbbff2f212b5aa6163c7152d46ffcaf3a2ff3ac3/specs/data/2.0/box-textured-binary-separate/box-textured-binary-separate.glb
--------------------------------------------------------------------------------
/specs/data/2.0/box-textured-binary-separate/cesium.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/CesiumGS/gltf-pipeline/cbbff2f212b5aa6163c7152d46ffcaf3a2ff3ac3/specs/data/2.0/box-textured-binary-separate/cesium.png
--------------------------------------------------------------------------------
/specs/data/2.0/box-textured-binary/box-textured-binary.glb:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/CesiumGS/gltf-pipeline/cbbff2f212b5aa6163c7152d46ffcaf3a2ff3ac3/specs/data/2.0/box-textured-binary/box-textured-binary.glb
--------------------------------------------------------------------------------
/specs/data/2.0/box-textured-separate/box-textured-separate.bin:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/CesiumGS/gltf-pipeline/cbbff2f212b5aa6163c7152d46ffcaf3a2ff3ac3/specs/data/2.0/box-textured-separate/box-textured-separate.bin
--------------------------------------------------------------------------------
/specs/data/2.0/box-textured-separate/box-textured-separate.gltf:
--------------------------------------------------------------------------------
1 | {
2 | "asset": {
3 | "generator": "COLLADA2GLTF",
4 | "version": "2.0"
5 | },
6 | "scene": 0,
7 | "scenes": [
8 | {
9 | "nodes": [
10 | 0
11 | ]
12 | }
13 | ],
14 | "nodes": [
15 | {
16 | "children": [
17 | 1
18 | ],
19 | "matrix": [
20 | 1,
21 | 0,
22 | 0,
23 | 0,
24 | 0,
25 | 0,
26 | -1,
27 | 0,
28 | 0,
29 | 1,
30 | 0,
31 | 0,
32 | 0,
33 | 0,
34 | 0,
35 | 1
36 | ]
37 | },
38 | {
39 | "mesh": 0
40 | }
41 | ],
42 | "meshes": [
43 | {
44 | "primitives": [
45 | {
46 | "attributes": {
47 | "NORMAL": 1,
48 | "POSITION": 2,
49 | "TEXCOORD_0": 3
50 | },
51 | "indices": 0,
52 | "mode": 4,
53 | "material": 0
54 | }
55 | ],
56 | "name": "Mesh"
57 | }
58 | ],
59 | "accessors": [
60 | {
61 | "bufferView": 0,
62 | "byteOffset": 0,
63 | "componentType": 5123,
64 | "count": 36,
65 | "max": [
66 | 23
67 | ],
68 | "min": [
69 | 0
70 | ],
71 | "type": "SCALAR"
72 | },
73 | {
74 | "bufferView": 1,
75 | "byteOffset": 0,
76 | "componentType": 5126,
77 | "count": 24,
78 | "max": [
79 | 1,
80 | 1,
81 | 1
82 | ],
83 | "min": [
84 | -1,
85 | -1,
86 | -1
87 | ],
88 | "type": "VEC3"
89 | },
90 | {
91 | "bufferView": 1,
92 | "byteOffset": 288,
93 | "componentType": 5126,
94 | "count": 24,
95 | "max": [
96 | 0.5,
97 | 0.5,
98 | 0.5
99 | ],
100 | "min": [
101 | -0.5,
102 | -0.5,
103 | -0.5
104 | ],
105 | "type": "VEC3"
106 | },
107 | {
108 | "bufferView": 2,
109 | "byteOffset": 0,
110 | "componentType": 5126,
111 | "count": 24,
112 | "max": [
113 | 6,
114 | 1
115 | ],
116 | "min": [
117 | 0,
118 | 0
119 | ],
120 | "type": "VEC2"
121 | }
122 | ],
123 | "materials": [
124 | {
125 | "pbrMetallicRoughness": {
126 | "baseColorTexture": {
127 | "index": 0,
128 | "texCoord": 0
129 | },
130 | "metallicFactor": 0,
131 | "baseColorFactor": [
132 | 1,
133 | 1,
134 | 1,
135 | 1
136 | ],
137 | "roughnessFactor": 1
138 | },
139 | "name": "Texture",
140 | "emissiveFactor": [
141 | 0,
142 | 0,
143 | 0
144 | ],
145 | "alphaMode": "OPAQUE",
146 | "doubleSided": false
147 | }
148 | ],
149 | "textures": [
150 | {
151 | "sampler": 0,
152 | "source": 0
153 | }
154 | ],
155 | "images": [
156 | {
157 | "uri": "cesium%20logo.png"
158 | }
159 | ],
160 | "samplers": [
161 | {
162 | "magFilter": 9729,
163 | "minFilter": 9986,
164 | "wrapS": 10497,
165 | "wrapT": 10497
166 | }
167 | ],
168 | "bufferViews": [
169 | {
170 | "buffer": 0,
171 | "byteOffset": 768,
172 | "byteLength": 72,
173 | "target": 34963
174 | },
175 | {
176 | "buffer": 0,
177 | "byteOffset": 0,
178 | "byteLength": 576,
179 | "byteStride": 12,
180 | "target": 34962
181 | },
182 | {
183 | "buffer": 0,
184 | "byteOffset": 576,
185 | "byteLength": 192,
186 | "byteStride": 8,
187 | "target": 34962
188 | }
189 | ],
190 | "buffers": [
191 | {
192 | "byteLength": 840,
193 | "uri": "box-textured-separate.bin"
194 | }
195 | ]
196 | }
197 |
--------------------------------------------------------------------------------
/specs/data/2.0/box-textured-separate/cesium logo.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/CesiumGS/gltf-pipeline/cbbff2f212b5aa6163c7152d46ffcaf3a2ff3ac3/specs/data/2.0/box-textured-separate/cesium logo.png
--------------------------------------------------------------------------------
/specs/data/2.0/extensions/EXT_meshopt_compression/meshopt-fallback/meshopt-fallback.bin:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/CesiumGS/gltf-pipeline/cbbff2f212b5aa6163c7152d46ffcaf3a2ff3ac3/specs/data/2.0/extensions/EXT_meshopt_compression/meshopt-fallback/meshopt-fallback.bin
--------------------------------------------------------------------------------
/specs/data/2.0/extensions/EXT_meshopt_compression/meshopt-fallback/meshopt-fallback.fallback.bin:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/CesiumGS/gltf-pipeline/cbbff2f212b5aa6163c7152d46ffcaf3a2ff3ac3/specs/data/2.0/extensions/EXT_meshopt_compression/meshopt-fallback/meshopt-fallback.fallback.bin
--------------------------------------------------------------------------------
/specs/data/2.0/extensions/EXT_meshopt_compression/meshopt-fallback/meshopt-fallback.gltf:
--------------------------------------------------------------------------------
1 | {
2 | "buffers": [
3 | {
4 | "uri": "meshopt-fallback.bin",
5 | "byteLength": 468
6 | },
7 | {
8 | "uri": "meshopt-fallback.fallback.bin",
9 | "byteLength": 960,
10 | "extensions": {
11 | "EXT_meshopt_compression": {
12 | "fallback": true
13 | }
14 | }
15 | }
16 | ],
17 | "asset": {
18 | "version": "2.0",
19 | "generator": "gltfpack 0.16"
20 | },
21 | "extensionsUsed": [
22 | "KHR_mesh_quantization",
23 | "EXT_meshopt_compression"
24 | ],
25 | "extensionsRequired": [
26 | "KHR_mesh_quantization"
27 | ],
28 | "bufferViews": [
29 | {
30 | "buffer": 1,
31 | "byteOffset": 0,
32 | "byteLength": 400,
33 | "byteStride": 8,
34 | "target": 34962,
35 | "extensions": {
36 | "EXT_meshopt_compression": {
37 | "buffer": 0,
38 | "byteOffset": 0,
39 | "byteLength": 265,
40 | "byteStride": 8,
41 | "mode": "ATTRIBUTES",
42 | "count": 50
43 | }
44 | }
45 | },
46 | {
47 | "buffer": 1,
48 | "byteOffset": 400,
49 | "byteLength": 200,
50 | "byteStride": 4,
51 | "target": 34962,
52 | "extensions": {
53 | "EXT_meshopt_compression": {
54 | "buffer": 0,
55 | "byteOffset": 268,
56 | "byteLength": 117,
57 | "byteStride": 4,
58 | "mode": "ATTRIBUTES",
59 | "filter": "OCTAHEDRAL",
60 | "count": 50
61 | }
62 | }
63 | },
64 | {
65 | "buffer": 1,
66 | "byteOffset": 600,
67 | "byteLength": 360,
68 | "target": 34963,
69 | "extensions": {
70 | "EXT_meshopt_compression": {
71 | "buffer": 0,
72 | "byteOffset": 388,
73 | "byteLength": 78,
74 | "byteStride": 2,
75 | "mode": "TRIANGLES",
76 | "count": 180
77 | }
78 | }
79 | }
80 | ],
81 | "accessors": [
82 | {
83 | "bufferView": 0,
84 | "byteOffset": 0,
85 | "componentType": 5123,
86 | "count": 24,
87 | "type": "VEC3",
88 | "min": [
89 | 0,
90 | 0,
91 | 0
92 | ],
93 | "max": [
94 | 16383,
95 | 16383,
96 | 16383
97 | ]
98 | },
99 | {
100 | "bufferView": 1,
101 | "byteOffset": 0,
102 | "componentType": 5120,
103 | "count": 24,
104 | "type": "VEC3",
105 | "normalized": true
106 | },
107 | {
108 | "bufferView": 2,
109 | "byteOffset": 0,
110 | "componentType": 5123,
111 | "count": 36,
112 | "type": "SCALAR"
113 | },
114 | {
115 | "bufferView": 0,
116 | "byteOffset": 192,
117 | "componentType": 5123,
118 | "count": 26,
119 | "type": "VEC3",
120 | "min": [
121 | 1315,
122 | 1315,
123 | 1315
124 | ],
125 | "max": [
126 | 15068,
127 | 15068,
128 | 15068
129 | ]
130 | },
131 | {
132 | "bufferView": 1,
133 | "byteOffset": 96,
134 | "componentType": 5120,
135 | "count": 26,
136 | "type": "VEC3",
137 | "normalized": true
138 | },
139 | {
140 | "bufferView": 2,
141 | "byteOffset": 72,
142 | "componentType": 5123,
143 | "count": 144,
144 | "type": "SCALAR"
145 | }
146 | ],
147 | "materials": [
148 | {
149 | "name": "Material.002",
150 | "pbrMetallicRoughness": {
151 | "baseColorFactor": [
152 | 0.800000072,
153 | 0.00122899958,
154 | 0,
155 | 1
156 | ],
157 | "metallicFactor": 0,
158 | "roughnessFactor": 0.5
159 | },
160 | "doubleSided": true
161 | },
162 | {
163 | "name": "Material.001",
164 | "pbrMetallicRoughness": {
165 | "baseColorFactor": [
166 | 0.800000012,
167 | 0.800000012,
168 | 0.800000012,
169 | 1
170 | ],
171 | "metallicFactor": 0,
172 | "roughnessFactor": 0.5
173 | },
174 | "doubleSided": true
175 | }
176 | ],
177 | "meshes": [
178 | {
179 | "primitives": [
180 | {
181 | "attributes": {
182 | "POSITION": 0,
183 | "NORMAL": 1
184 | },
185 | "mode": 4,
186 | "indices": 2,
187 | "material": 0
188 | }
189 | ]
190 | },
191 | {
192 | "primitives": [
193 | {
194 | "attributes": {
195 | "POSITION": 3,
196 | "NORMAL": 4
197 | },
198 | "mode": 4,
199 | "indices": 5,
200 | "material": 1
201 | }
202 | ]
203 | }
204 | ],
205 | "nodes": [
206 | {
207 | "mesh": 0,
208 | "translation": [
209 | -1,
210 | -1,
211 | -1
212 | ],
213 | "scale": [
214 | 0.000122077763,
215 | 0.000122077763,
216 | 0.000122077763
217 | ]
218 | },
219 | {
220 | "mesh": 1,
221 | "translation": [
222 | -1,
223 | -1,
224 | -1
225 | ],
226 | "scale": [
227 | 0.000122077763,
228 | 0.000122077763,
229 | 0.000122077763
230 | ]
231 | },
232 | {
233 | "name": "Cube",
234 | "children": [
235 | 0
236 | ]
237 | },
238 | {
239 | "name": "Cube.001",
240 | "translation": [
241 | 0,
242 | 2.87217259,
243 | 0
244 | ],
245 | "children": [
246 | 1
247 | ]
248 | }
249 | ],
250 | "scenes": [
251 | {
252 | "name": "Scene",
253 | "nodes": [
254 | 2,
255 | 3
256 | ]
257 | }
258 | ],
259 | "scene": 0
260 | }
261 |
--------------------------------------------------------------------------------
/specs/data/2.0/extensions/EXT_meshopt_compression/meshopt-no-fallback/meshopt-no-fallback.bin:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/CesiumGS/gltf-pipeline/cbbff2f212b5aa6163c7152d46ffcaf3a2ff3ac3/specs/data/2.0/extensions/EXT_meshopt_compression/meshopt-no-fallback/meshopt-no-fallback.bin
--------------------------------------------------------------------------------
/specs/data/2.0/extensions/EXT_meshopt_compression/meshopt-no-fallback/meshopt-no-fallback.gltf:
--------------------------------------------------------------------------------
1 | {
2 | "buffers": [
3 | {
4 | "uri": "meshopt-no-fallback.bin",
5 | "byteLength": 468
6 | },
7 | {
8 | "byteLength": 960
9 | }
10 | ],
11 | "asset": {
12 | "version": "2.0",
13 | "generator": "gltfpack 0.16"
14 | },
15 | "extensionsUsed": [
16 | "KHR_mesh_quantization",
17 | "EXT_meshopt_compression"
18 | ],
19 | "extensionsRequired": [
20 | "KHR_mesh_quantization",
21 | "EXT_meshopt_compression"
22 | ],
23 | "bufferViews": [
24 | {
25 | "buffer": 1,
26 | "byteOffset": 0,
27 | "byteLength": 400,
28 | "byteStride": 8,
29 | "target": 34962,
30 | "extensions": {
31 | "EXT_meshopt_compression": {
32 | "buffer": 0,
33 | "byteOffset": 0,
34 | "byteLength": 265,
35 | "byteStride": 8,
36 | "mode": "ATTRIBUTES",
37 | "count": 50
38 | }
39 | }
40 | },
41 | {
42 | "buffer": 1,
43 | "byteOffset": 400,
44 | "byteLength": 200,
45 | "byteStride": 4,
46 | "target": 34962,
47 | "extensions": {
48 | "EXT_meshopt_compression": {
49 | "buffer": 0,
50 | "byteOffset": 268,
51 | "byteLength": 117,
52 | "byteStride": 4,
53 | "mode": "ATTRIBUTES",
54 | "filter": "OCTAHEDRAL",
55 | "count": 50
56 | }
57 | }
58 | },
59 | {
60 | "buffer": 1,
61 | "byteOffset": 600,
62 | "byteLength": 360,
63 | "target": 34963,
64 | "extensions": {
65 | "EXT_meshopt_compression": {
66 | "buffer": 0,
67 | "byteOffset": 388,
68 | "byteLength": 78,
69 | "byteStride": 2,
70 | "mode": "TRIANGLES",
71 | "count": 180
72 | }
73 | }
74 | }
75 | ],
76 | "accessors": [
77 | {
78 | "bufferView": 0,
79 | "byteOffset": 0,
80 | "componentType": 5123,
81 | "count": 24,
82 | "type": "VEC3",
83 | "min": [
84 | 0,
85 | 0,
86 | 0
87 | ],
88 | "max": [
89 | 16383,
90 | 16383,
91 | 16383
92 | ]
93 | },
94 | {
95 | "bufferView": 1,
96 | "byteOffset": 0,
97 | "componentType": 5120,
98 | "count": 24,
99 | "type": "VEC3",
100 | "normalized": true
101 | },
102 | {
103 | "bufferView": 2,
104 | "byteOffset": 0,
105 | "componentType": 5123,
106 | "count": 36,
107 | "type": "SCALAR"
108 | },
109 | {
110 | "bufferView": 0,
111 | "byteOffset": 192,
112 | "componentType": 5123,
113 | "count": 26,
114 | "type": "VEC3",
115 | "min": [
116 | 1315,
117 | 1315,
118 | 1315
119 | ],
120 | "max": [
121 | 15068,
122 | 15068,
123 | 15068
124 | ]
125 | },
126 | {
127 | "bufferView": 1,
128 | "byteOffset": 96,
129 | "componentType": 5120,
130 | "count": 26,
131 | "type": "VEC3",
132 | "normalized": true
133 | },
134 | {
135 | "bufferView": 2,
136 | "byteOffset": 72,
137 | "componentType": 5123,
138 | "count": 144,
139 | "type": "SCALAR"
140 | }
141 | ],
142 | "materials": [
143 | {
144 | "name": "Material.002",
145 | "pbrMetallicRoughness": {
146 | "baseColorFactor": [
147 | 0.800000072,
148 | 0.00122899958,
149 | 0,
150 | 1
151 | ],
152 | "metallicFactor": 0,
153 | "roughnessFactor": 0.5
154 | },
155 | "doubleSided": true
156 | },
157 | {
158 | "name": "Material.001",
159 | "pbrMetallicRoughness": {
160 | "baseColorFactor": [
161 | 0.800000012,
162 | 0.800000012,
163 | 0.800000012,
164 | 1
165 | ],
166 | "metallicFactor": 0,
167 | "roughnessFactor": 0.5
168 | },
169 | "doubleSided": true
170 | }
171 | ],
172 | "meshes": [
173 | {
174 | "primitives": [
175 | {
176 | "attributes": {
177 | "POSITION": 0,
178 | "NORMAL": 1
179 | },
180 | "mode": 4,
181 | "indices": 2,
182 | "material": 0
183 | }
184 | ]
185 | },
186 | {
187 | "primitives": [
188 | {
189 | "attributes": {
190 | "POSITION": 3,
191 | "NORMAL": 4
192 | },
193 | "mode": 4,
194 | "indices": 5,
195 | "material": 1
196 | }
197 | ]
198 | }
199 | ],
200 | "nodes": [
201 | {
202 | "mesh": 0,
203 | "translation": [
204 | -1,
205 | -1,
206 | -1
207 | ],
208 | "scale": [
209 | 0.000122077763,
210 | 0.000122077763,
211 | 0.000122077763
212 | ]
213 | },
214 | {
215 | "mesh": 1,
216 | "translation": [
217 | -1,
218 | -1,
219 | -1
220 | ],
221 | "scale": [
222 | 0.000122077763,
223 | 0.000122077763,
224 | 0.000122077763
225 | ]
226 | },
227 | {
228 | "name": "Cube",
229 | "children": [
230 | 0
231 | ]
232 | },
233 | {
234 | "name": "Cube.001",
235 | "translation": [
236 | 0,
237 | 2.87217259,
238 | 0
239 | ],
240 | "children": [
241 | 1
242 | ]
243 | }
244 | ],
245 | "scenes": [
246 | {
247 | "name": "Scene",
248 | "nodes": [
249 | 2,
250 | 3
251 | ]
252 | }
253 | ],
254 | "scene": 0
255 | }
256 |
--------------------------------------------------------------------------------
/specs/data/2.0/extensions/EXT_texture_webp/box-textured-separate/box-textured-separate.bin:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/CesiumGS/gltf-pipeline/cbbff2f212b5aa6163c7152d46ffcaf3a2ff3ac3/specs/data/2.0/extensions/EXT_texture_webp/box-textured-separate/box-textured-separate.bin
--------------------------------------------------------------------------------
/specs/data/2.0/extensions/EXT_texture_webp/box-textured-separate/box-textured-separate.gltf:
--------------------------------------------------------------------------------
1 | {
2 | "asset": {
3 | "generator": "COLLADA2GLTF",
4 | "version": "2.0"
5 | },
6 | "extensionsUsed": [
7 | "EXT_texture_webp"
8 | ],
9 | "extensionsRequired": [
10 | "EXT_texture_webp"
11 | ],
12 | "scene": 0,
13 | "scenes": [
14 | {
15 | "nodes": [
16 | 0
17 | ]
18 | }
19 | ],
20 | "nodes": [
21 | {
22 | "children": [
23 | 1
24 | ],
25 | "matrix": [
26 | 1,
27 | 0,
28 | 0,
29 | 0,
30 | 0,
31 | 0,
32 | -1,
33 | 0,
34 | 0,
35 | 1,
36 | 0,
37 | 0,
38 | 0,
39 | 0,
40 | 0,
41 | 1
42 | ]
43 | },
44 | {
45 | "mesh": 0
46 | }
47 | ],
48 | "meshes": [
49 | {
50 | "primitives": [
51 | {
52 | "attributes": {
53 | "NORMAL": 1,
54 | "POSITION": 2,
55 | "TEXCOORD_0": 3
56 | },
57 | "indices": 0,
58 | "mode": 4,
59 | "material": 0
60 | }
61 | ],
62 | "name": "Mesh"
63 | }
64 | ],
65 | "accessors": [
66 | {
67 | "bufferView": 0,
68 | "byteOffset": 0,
69 | "componentType": 5123,
70 | "count": 36,
71 | "max": [
72 | 23
73 | ],
74 | "min": [
75 | 0
76 | ],
77 | "type": "SCALAR"
78 | },
79 | {
80 | "bufferView": 1,
81 | "byteOffset": 0,
82 | "componentType": 5126,
83 | "count": 24,
84 | "max": [
85 | 1,
86 | 1,
87 | 1
88 | ],
89 | "min": [
90 | -1,
91 | -1,
92 | -1
93 | ],
94 | "type": "VEC3"
95 | },
96 | {
97 | "bufferView": 1,
98 | "byteOffset": 288,
99 | "componentType": 5126,
100 | "count": 24,
101 | "max": [
102 | 0.5,
103 | 0.5,
104 | 0.5
105 | ],
106 | "min": [
107 | -0.5,
108 | -0.5,
109 | -0.5
110 | ],
111 | "type": "VEC3"
112 | },
113 | {
114 | "bufferView": 2,
115 | "byteOffset": 0,
116 | "componentType": 5126,
117 | "count": 24,
118 | "max": [
119 | 6,
120 | 1
121 | ],
122 | "min": [
123 | 0,
124 | 0
125 | ],
126 | "type": "VEC2"
127 | }
128 | ],
129 | "materials": [
130 | {
131 | "pbrMetallicRoughness": {
132 | "baseColorTexture": {
133 | "index": 0,
134 | "texCoord": 0
135 | },
136 | "metallicFactor": 0,
137 | "baseColorFactor": [
138 | 1,
139 | 1,
140 | 1,
141 | 1
142 | ],
143 | "roughnessFactor": 1
144 | },
145 | "name": "Texture",
146 | "emissiveFactor": [
147 | 0,
148 | 0,
149 | 0
150 | ],
151 | "alphaMode": "OPAQUE",
152 | "doubleSided": false
153 | }
154 | ],
155 | "textures": [
156 | {
157 | "sampler": 0,
158 | "extensions": {
159 | "EXT_texture_webp": {
160 | "source": 0
161 | }
162 | }
163 | }
164 | ],
165 | "images": [
166 | {
167 | "uri": "cesium%20logo.webp"
168 | }
169 | ],
170 | "samplers": [
171 | {
172 | "magFilter": 9729,
173 | "minFilter": 9986,
174 | "wrapS": 10497,
175 | "wrapT": 10497
176 | }
177 | ],
178 | "bufferViews": [
179 | {
180 | "buffer": 0,
181 | "byteOffset": 768,
182 | "byteLength": 72,
183 | "target": 34963
184 | },
185 | {
186 | "buffer": 0,
187 | "byteOffset": 0,
188 | "byteLength": 576,
189 | "byteStride": 12,
190 | "target": 34962
191 | },
192 | {
193 | "buffer": 0,
194 | "byteOffset": 576,
195 | "byteLength": 192,
196 | "byteStride": 8,
197 | "target": 34962
198 | }
199 | ],
200 | "buffers": [
201 | {
202 | "byteLength": 840,
203 | "uri": "box-textured-separate.bin"
204 | }
205 | ]
206 | }
207 |
--------------------------------------------------------------------------------
/specs/data/2.0/extensions/EXT_texture_webp/box-textured-separate/box-textured-with-fallback.gltf:
--------------------------------------------------------------------------------
1 | {
2 | "asset": {
3 | "generator": "COLLADA2GLTF",
4 | "version": "2.0"
5 | },
6 | "extensionsUsed": [
7 | "EXT_texture_webp"
8 | ],
9 | "scene": 0,
10 | "scenes": [
11 | {
12 | "nodes": [
13 | 0
14 | ]
15 | }
16 | ],
17 | "nodes": [
18 | {
19 | "children": [
20 | 1
21 | ],
22 | "matrix": [
23 | 1,
24 | 0,
25 | 0,
26 | 0,
27 | 0,
28 | 0,
29 | -1,
30 | 0,
31 | 0,
32 | 1,
33 | 0,
34 | 0,
35 | 0,
36 | 0,
37 | 0,
38 | 1
39 | ]
40 | },
41 | {
42 | "mesh": 0
43 | }
44 | ],
45 | "meshes": [
46 | {
47 | "primitives": [
48 | {
49 | "attributes": {
50 | "NORMAL": 1,
51 | "POSITION": 2,
52 | "TEXCOORD_0": 3
53 | },
54 | "indices": 0,
55 | "mode": 4,
56 | "material": 0
57 | }
58 | ],
59 | "name": "Mesh"
60 | }
61 | ],
62 | "accessors": [
63 | {
64 | "bufferView": 0,
65 | "byteOffset": 0,
66 | "componentType": 5123,
67 | "count": 36,
68 | "max": [
69 | 23
70 | ],
71 | "min": [
72 | 0
73 | ],
74 | "type": "SCALAR"
75 | },
76 | {
77 | "bufferView": 1,
78 | "byteOffset": 0,
79 | "componentType": 5126,
80 | "count": 24,
81 | "max": [
82 | 1,
83 | 1,
84 | 1
85 | ],
86 | "min": [
87 | -1,
88 | -1,
89 | -1
90 | ],
91 | "type": "VEC3"
92 | },
93 | {
94 | "bufferView": 1,
95 | "byteOffset": 288,
96 | "componentType": 5126,
97 | "count": 24,
98 | "max": [
99 | 0.5,
100 | 0.5,
101 | 0.5
102 | ],
103 | "min": [
104 | -0.5,
105 | -0.5,
106 | -0.5
107 | ],
108 | "type": "VEC3"
109 | },
110 | {
111 | "bufferView": 2,
112 | "byteOffset": 0,
113 | "componentType": 5126,
114 | "count": 24,
115 | "max": [
116 | 6,
117 | 1
118 | ],
119 | "min": [
120 | 0,
121 | 0
122 | ],
123 | "type": "VEC2"
124 | }
125 | ],
126 | "materials": [
127 | {
128 | "pbrMetallicRoughness": {
129 | "baseColorTexture": {
130 | "index": 0,
131 | "texCoord": 0
132 | },
133 | "metallicFactor": 0,
134 | "baseColorFactor": [
135 | 1,
136 | 1,
137 | 1,
138 | 1
139 | ],
140 | "roughnessFactor": 1
141 | },
142 | "name": "Texture",
143 | "emissiveFactor": [
144 | 0,
145 | 0,
146 | 0
147 | ],
148 | "alphaMode": "OPAQUE",
149 | "doubleSided": false
150 | }
151 | ],
152 | "textures": [
153 | {
154 | "sampler": 0,
155 | "source": 1,
156 | "extensions": {
157 | "EXT_texture_webp": {
158 | "source": 0
159 | }
160 | }
161 | }
162 | ],
163 | "images": [
164 | {
165 | "uri": "cesium%20logo.webp"
166 | },
167 | {
168 | "uri": "cesium%20logo.png"
169 | }
170 | ],
171 | "samplers": [
172 | {
173 | "magFilter": 9729,
174 | "minFilter": 9986,
175 | "wrapS": 10497,
176 | "wrapT": 10497
177 | }
178 | ],
179 | "bufferViews": [
180 | {
181 | "buffer": 0,
182 | "byteOffset": 768,
183 | "byteLength": 72,
184 | "target": 34963
185 | },
186 | {
187 | "buffer": 0,
188 | "byteOffset": 0,
189 | "byteLength": 576,
190 | "byteStride": 12,
191 | "target": 34962
192 | },
193 | {
194 | "buffer": 0,
195 | "byteOffset": 576,
196 | "byteLength": 192,
197 | "byteStride": 8,
198 | "target": 34962
199 | }
200 | ],
201 | "buffers": [
202 | {
203 | "byteLength": 840,
204 | "uri": "box-textured-separate.bin"
205 | }
206 | ]
207 | }
208 |
--------------------------------------------------------------------------------
/specs/data/2.0/extensions/EXT_texture_webp/box-textured-separate/cesium logo.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/CesiumGS/gltf-pipeline/cbbff2f212b5aa6163c7152d46ffcaf3a2ff3ac3/specs/data/2.0/extensions/EXT_texture_webp/box-textured-separate/cesium logo.png
--------------------------------------------------------------------------------
/specs/data/2.0/extensions/EXT_texture_webp/box-textured-separate/cesium logo.webp:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/CesiumGS/gltf-pipeline/cbbff2f212b5aa6163c7152d46ffcaf3a2ff3ac3/specs/data/2.0/extensions/EXT_texture_webp/box-textured-separate/cesium logo.webp
--------------------------------------------------------------------------------
/specs/data/2.0/triangle-without-indices/triangle-without-indices.gltf:
--------------------------------------------------------------------------------
1 | {
2 | "scenes" : [
3 | {
4 | "nodes" : [ 0 ]
5 | }
6 | ],
7 |
8 | "nodes" : [
9 | {
10 | "mesh" : 0
11 | }
12 | ],
13 |
14 | "meshes" : [
15 | {
16 | "primitives" : [ {
17 | "attributes" : {
18 | "POSITION" : 0
19 | }
20 | } ]
21 | }
22 | ],
23 |
24 | "buffers" : [
25 | {
26 | "uri" : "data:application/octet-stream;base64,AAAAAAAAAAAAAAAAAACAPwAAAAAAAAAAAAAAAAAAgD8AAAAA",
27 | "byteLength" : 36
28 | }
29 | ],
30 | "bufferViews" : [
31 | {
32 | "buffer" : 0,
33 | "byteOffset" : 0,
34 | "byteLength" : 36,
35 | "target" : 34962
36 | }
37 | ],
38 | "accessors" : [
39 | {
40 | "bufferView" : 0,
41 | "byteOffset" : 0,
42 | "componentType" : 5126,
43 | "count" : 3,
44 | "type" : "VEC3",
45 | "max" : [ 1.0, 1.0, 0.0 ],
46 | "min" : [ 0.0, 0.0, 0.0 ]
47 | }
48 | ],
49 |
50 | "asset" : {
51 | "version" : "2.0"
52 | }
53 | }
54 |
--------------------------------------------------------------------------------
/specs/jasmine.json:
--------------------------------------------------------------------------------
1 | {
2 | "spec_dir": "specs",
3 | "spec_files": [
4 | "**/*Spec.js"
5 | ]
6 | }
7 |
--------------------------------------------------------------------------------
/specs/lib/addBufferSpec.js:
--------------------------------------------------------------------------------
1 | "use strict";
2 | const addBuffer = require("../../lib/addBuffer");
3 |
4 | describe("addBuffer", () => {
5 | it("adds buffer to gltf and returns its bufferView id", () => {
6 | const gltf = {
7 | buffers: [],
8 | bufferViews: [],
9 | };
10 | const buffer0 = Buffer.alloc(100);
11 | const buffer1 = Buffer.alloc(200);
12 |
13 | expect(addBuffer(gltf, buffer0)).toBe(0);
14 | expect(addBuffer(gltf, buffer1)).toBe(1);
15 | expect(gltf.buffers.length).toBe(2);
16 | expect(gltf.bufferViews.length).toBe(2);
17 | expect(gltf.buffers[0].extras._pipeline.source).toEqual(buffer0);
18 | expect(gltf.buffers[1].extras._pipeline.source).toEqual(buffer1);
19 | expect(gltf.bufferViews[0].buffer).toBe(0);
20 | expect(gltf.bufferViews[1].buffer).toBe(1);
21 | });
22 | });
23 |
--------------------------------------------------------------------------------
/specs/lib/addExtensionsRequiredSpec.js:
--------------------------------------------------------------------------------
1 | "use strict";
2 | const addExtensionsRequired = require("../../lib/addExtensionsRequired");
3 |
4 | describe("addExtensionsRequired", () => {
5 | it("adds an extension to extensionsRequired", () => {
6 | const gltf = {};
7 | addExtensionsRequired(gltf, "KHR_materials_pbrSpecularGlossiness");
8 | addExtensionsRequired(gltf, "KHR_draco_mesh_compression");
9 | addExtensionsRequired(gltf, "KHR_draco_mesh_compression"); // Test adding duplicate
10 | expect(gltf.extensionsRequired).toEqual([
11 | "KHR_materials_pbrSpecularGlossiness",
12 | "KHR_draco_mesh_compression",
13 | ]);
14 | expect(gltf.extensionsUsed).toEqual([
15 | "KHR_materials_pbrSpecularGlossiness",
16 | "KHR_draco_mesh_compression",
17 | ]);
18 | });
19 | });
20 |
--------------------------------------------------------------------------------
/specs/lib/addExtensionsUsedSpec.js:
--------------------------------------------------------------------------------
1 | "use strict";
2 | const addExtensionsUsed = require("../../lib/addExtensionsUsed");
3 |
4 | describe("addExtensionsUsed", () => {
5 | it("adds an extension to extensionsUsed", () => {
6 | const gltf = {};
7 | addExtensionsUsed(gltf, "KHR_materials_pbrSpecularGlossiness");
8 | addExtensionsUsed(gltf, "KHR_draco_mesh_compression");
9 | addExtensionsUsed(gltf, "KHR_draco_mesh_compression"); // Test adding duplicate
10 | expect(gltf.extensionsUsed).toEqual([
11 | "KHR_materials_pbrSpecularGlossiness",
12 | "KHR_draco_mesh_compression",
13 | ]);
14 | expect(gltf.extensionsRequired).toBeUndefined();
15 | });
16 | });
17 |
--------------------------------------------------------------------------------
/specs/lib/addPipelineExtrasSpec.js:
--------------------------------------------------------------------------------
1 | "use strict";
2 | const Cesium = require("cesium");
3 | const addPipelineExtras = require("../../lib/addPipelineExtras");
4 |
5 | const WebGLConstants = Cesium.WebGLConstants;
6 |
7 | describe("addPipelineExtras", () => {
8 | it("adds pipeline extras to glTF 1.0 assets", () => {
9 | const gltf = {
10 | buffers: {
11 | sampleBuffer0: {
12 | byteLength: 100,
13 | },
14 | },
15 | shaders: {
16 | sample0VS: {
17 | type: WebGLConstants.VERTEX_SHADER,
18 | uri: "data:,",
19 | },
20 | },
21 | };
22 | const gltfWithExtras = addPipelineExtras(gltf);
23 | expect(
24 | gltfWithExtras.buffers["sampleBuffer0"].extras._pipeline,
25 | ).toBeDefined();
26 | expect(gltfWithExtras.shaders["sample0VS"].extras._pipeline).toBeDefined();
27 | });
28 |
29 | it("adds pipeline extras to glTF 2.0 assets", () => {
30 | const gltf = {
31 | buffers: [
32 | {
33 | byteLength: 100,
34 | },
35 | ],
36 | extensions: {
37 | KHR_techniques_webgl: {
38 | shaders: [
39 | {
40 | type: WebGLConstants.VERTEX_SHADER,
41 | uri: "data:,",
42 | },
43 | ],
44 | },
45 | },
46 | extensionsRequired: ["KHR_techniques_webgl"],
47 | extensionsUsed: ["KHR_techniques_webgl"],
48 | };
49 | const gltfWithExtras = addPipelineExtras(gltf);
50 | expect(gltfWithExtras.buffers[0].extras._pipeline).toBeDefined();
51 | expect(
52 | gltfWithExtras.extensions.KHR_techniques_webgl.shaders[0].extras
53 | ._pipeline,
54 | ).toBeDefined();
55 | });
56 | });
57 |
--------------------------------------------------------------------------------
/specs/lib/addToArraySpec.js:
--------------------------------------------------------------------------------
1 | "use strict";
2 | const addToArray = require("../../lib/addToArray");
3 |
4 | describe("addToArray", () => {
5 | it("adds item to array and returns its index", () => {
6 | const gltf = {
7 | buffers: [],
8 | };
9 | const buffer0 = {
10 | byteLength: 100,
11 | };
12 | const buffer1 = {
13 | byteLength: 200,
14 | };
15 | expect(addToArray(gltf.buffers, buffer0)).toBe(0);
16 | expect(addToArray(gltf.buffers, buffer1)).toBe(1);
17 | expect(gltf.buffers).toEqual([buffer0, buffer1]);
18 | });
19 |
20 | it("returns index of duplicate element when checkDuplicates is true", () => {
21 | const gltf = {
22 | buffers: [],
23 | };
24 | const buffer0 = {
25 | byteLength: 100,
26 | };
27 | const buffer1 = {
28 | byteLength: 200,
29 | };
30 | expect(addToArray(gltf.buffers, buffer0, true)).toBe(0);
31 | expect(addToArray(gltf.buffers, buffer1, true)).toBe(1);
32 | expect(addToArray(gltf.buffers, buffer0, true)).toBe(0);
33 | expect(gltf.buffers.length).toBe(2);
34 | expect(addToArray(gltf.buffers, buffer0)).toBe(2);
35 | expect(gltf.buffers.length).toBe(3);
36 | });
37 | });
38 |
--------------------------------------------------------------------------------
/specs/lib/dataUriToBufferSpec.js:
--------------------------------------------------------------------------------
1 | "use strict";
2 | const dataUriToBuffer = require("../../lib/dataUriToBuffer");
3 |
4 | describe("dataUriToBuffer", () => {
5 | it("converts base64 data uri to buffer", () => {
6 | const buffer = Buffer.from([103, 108, 84, 70]);
7 | const dataUri = `data:application/octet-stream;base64,${buffer.toString(
8 | "base64",
9 | )}`;
10 | expect(dataUriToBuffer(dataUri)).toEqual(buffer);
11 | });
12 |
13 | it("converts utf8 data uri to buffer", () => {
14 | const buffer = Buffer.from([103, 108, 84, 70]);
15 | const dataUri = `data:text/plain;charset=utf-8,${buffer.toString("utf8")}`;
16 | expect(dataUriToBuffer(dataUri)).toEqual(buffer);
17 | });
18 | });
19 |
--------------------------------------------------------------------------------
/specs/lib/findAccessorMinMaxSpec.js:
--------------------------------------------------------------------------------
1 | "use strict";
2 | const findAccessorMinMax = require("../../lib/findAccessorMinMax");
3 | const readResources = require("../../lib/readResources");
4 |
5 | const contiguousData = [
6 | -1.0, -2.0, -3.0, 3.0, 2.0, 1.0, 0.0, 0.0, 0.0, 0.5, -0.5, 0.5,
7 | ];
8 |
9 | const nan = Number.NaN;
10 | const nonContiguousData = [
11 | -1.0,
12 | 1.0,
13 | -1.0,
14 | nan,
15 | nan,
16 | nan,
17 | 0.0,
18 | 0.0,
19 | 0.0,
20 | nan,
21 | nan,
22 | nan,
23 | 3.0,
24 | 2.0,
25 | 1.0,
26 | nan,
27 | nan,
28 | nan,
29 | -1.0,
30 | -2.0,
31 | -3.0,
32 | nan,
33 | nan,
34 | nan,
35 | ];
36 |
37 | function createGltf(elements, byteStride) {
38 | const buffer = Buffer.from(new Float32Array(elements).buffer);
39 | const byteLength = buffer.length;
40 | const dataUri = `data:application/octet-stream;base64,${buffer.toString(
41 | "base64",
42 | )}`;
43 | const gltf = {
44 | asset: {
45 | version: "2.0",
46 | },
47 | accessors: [
48 | {
49 | bufferView: 0,
50 | byteOffset: 0,
51 | componentType: 5126,
52 | count: 4,
53 | type: "VEC3",
54 | },
55 | ],
56 | bufferViews: [
57 | {
58 | buffer: 0,
59 | byteOffset: 0,
60 | byteLength: byteLength,
61 | byteStride: byteStride,
62 | },
63 | ],
64 | buffers: [
65 | {
66 | uri: dataUri,
67 | byteLength: byteLength,
68 | },
69 | ],
70 | };
71 | return readResources(gltf);
72 | }
73 |
74 | describe("findAccessorMinMax", () => {
75 | it("finds the min and max of an accessor", async () => {
76 | const gltf = await createGltf(contiguousData, 12);
77 | const expectedMin = [-1.0, -2.0, -3.0];
78 | const expectedMax = [3.0, 2.0, 1.0];
79 | const minMax = findAccessorMinMax(gltf, gltf.accessors[0]);
80 | expect(minMax.min).toEqual(expectedMin);
81 | expect(minMax.max).toEqual(expectedMax);
82 | });
83 |
84 | it("finds the min and max in a non-contiguous accessor", async () => {
85 | const gltf = await createGltf(nonContiguousData, 24);
86 | const expectedMin = [-1.0, -2.0, -3.0];
87 | const expectedMax = [3.0, 2.0, 1.0];
88 | const minMax = findAccessorMinMax(gltf, gltf.accessors[0]);
89 | expect(minMax.min).toEqual(expectedMin);
90 | expect(minMax.max).toEqual(expectedMax);
91 | });
92 | });
93 |
--------------------------------------------------------------------------------
/specs/lib/getAccessorByteStrideSpec.js:
--------------------------------------------------------------------------------
1 | "use strict";
2 | const Cesium = require("cesium");
3 | const getAccessorByteStride = require("../../lib/getAccessorByteStride");
4 |
5 | const WebGLConstants = Cesium.WebGLConstants;
6 |
7 | describe("getAccessorByteStride", () => {
8 | it("gets accessor byte stride", () => {
9 | const gltf = {
10 | accessors: [
11 | {
12 | componentType: WebGLConstants.FLOAT,
13 | count: 24,
14 | type: "VEC3",
15 | min: [-1.0, -1.0, -1.0],
16 | max: [1.0, 1.0, 1.0],
17 | },
18 | {
19 | bufferView: 0,
20 | componentType: WebGLConstants.FLOAT,
21 | count: 24,
22 | type: "VEC3",
23 | min: [-1.0, -1.0, -1.0],
24 | max: [1.0, 1.0, 1.0],
25 | },
26 | {
27 | bufferView: 1,
28 | componentType: WebGLConstants.FLOAT,
29 | count: 24,
30 | type: "VEC3",
31 | min: [-1.0, -1.0, -1.0],
32 | max: [1.0, 1.0, 1.0],
33 | },
34 | {
35 | componentType: WebGLConstants.FLOAT,
36 | count: 24,
37 | type: "VEC2",
38 | min: [0.0, 0.0],
39 | max: [1.0, 1.0],
40 | },
41 | {
42 | componentType: WebGLConstants.INT,
43 | count: 36,
44 | type: "SCALAR",
45 | min: [0],
46 | max: [24],
47 | },
48 | ],
49 | bufferViews: [
50 | {
51 | buffer: 0,
52 | byteLength: 288,
53 | byteOffset: 0,
54 | },
55 | {
56 | buffer: 0,
57 | byteLength: 288,
58 | byteOffset: 288,
59 | byteStride: 32,
60 | },
61 | ],
62 | };
63 |
64 | expect(getAccessorByteStride(gltf, gltf.accessors[0])).toBe(12);
65 | expect(getAccessorByteStride(gltf, gltf.accessors[1])).toBe(12);
66 | expect(getAccessorByteStride(gltf, gltf.accessors[2])).toBe(32);
67 | expect(getAccessorByteStride(gltf, gltf.accessors[3])).toBe(8);
68 | expect(getAccessorByteStride(gltf, gltf.accessors[4])).toBe(4);
69 | });
70 | });
71 |
--------------------------------------------------------------------------------
/specs/lib/getBufferPaddedSpec.js:
--------------------------------------------------------------------------------
1 | "use strict";
2 | const getBufferPadded = require("../../lib/getBufferPadded");
3 |
4 | describe("getBufferPadded", () => {
5 | it("gets buffer padded to 8 bytes", () => {
6 | let buffer = Buffer.alloc(0);
7 | let bufferPadded = getBufferPadded(buffer);
8 | expect(bufferPadded.length).toBe(0);
9 |
10 | buffer = Buffer.from([1]);
11 | bufferPadded = getBufferPadded(buffer);
12 | expect(bufferPadded.length).toBe(8);
13 | expect(bufferPadded.readUInt8(0)).toBe(1);
14 | expect(bufferPadded.readUInt8(1)).toBe(0);
15 | expect(bufferPadded.readUInt8(2)).toBe(0);
16 | expect(bufferPadded.readUInt8(3)).toBe(0);
17 | expect(bufferPadded.readUInt8(4)).toBe(0);
18 | expect(bufferPadded.readUInt8(5)).toBe(0);
19 | expect(bufferPadded.readUInt8(6)).toBe(0);
20 | expect(bufferPadded.readUInt8(7)).toBe(0);
21 |
22 | // Does not allocate a new buffer when buffer length is already aligned to 8 bytes
23 | buffer = Buffer.alloc(8);
24 | bufferPadded = getBufferPadded(buffer);
25 | expect(bufferPadded).toBe(buffer);
26 |
27 | buffer = Buffer.alloc(70);
28 | bufferPadded = getBufferPadded(buffer);
29 | expect(bufferPadded.length).toBe(72);
30 | });
31 | });
32 |
--------------------------------------------------------------------------------
/specs/lib/getComponentReaderSpec.js:
--------------------------------------------------------------------------------
1 | "use strict";
2 | const Cesium = require("cesium");
3 | const getComponentReader = require("../../lib/getComponentReader");
4 |
5 | const ComponentDatatype = Cesium.ComponentDatatype;
6 |
7 | function testComponentReader(componentType) {
8 | const typedArray = ComponentDatatype.createTypedArray(
9 | componentType,
10 | [0, 1, 2],
11 | );
12 | const dataView = new DataView(typedArray.buffer);
13 | const componentTypeByteLength =
14 | ComponentDatatype.getSizeInBytes(componentType);
15 | const componentReader = getComponentReader(componentType);
16 | const byteOffset = componentTypeByteLength;
17 | const numberOfComponents = 2;
18 | const result = new Array(numberOfComponents);
19 | componentReader(
20 | dataView,
21 | byteOffset,
22 | numberOfComponents,
23 | componentTypeByteLength,
24 | result,
25 | );
26 | expect(result).toEqual([1, 2]);
27 | }
28 |
29 | describe("getComponentReader", () => {
30 | it("reads values", () => {
31 | testComponentReader(ComponentDatatype.BYTE);
32 | testComponentReader(ComponentDatatype.UNSIGNED_BYTE);
33 | testComponentReader(ComponentDatatype.SHORT);
34 | testComponentReader(ComponentDatatype.UNSIGNED_SHORT);
35 | testComponentReader(ComponentDatatype.INT);
36 | testComponentReader(ComponentDatatype.UNSIGNED_INT);
37 | testComponentReader(ComponentDatatype.FLOAT);
38 | testComponentReader(ComponentDatatype.DOUBLE);
39 | });
40 | });
41 |
--------------------------------------------------------------------------------
/specs/lib/getImageExtensionSpec.js:
--------------------------------------------------------------------------------
1 | "use strict";
2 |
3 | const { RuntimeError } = require("cesium");
4 |
5 | const dataUriToBuffer = require("../../lib/dataUriToBuffer");
6 | const getImageExtension = require("../../lib/getImageExtension");
7 |
8 | const pngData = dataUriToBuffer(
9 | "",
10 | );
11 | const gifData = dataUriToBuffer(
12 | "",
13 | );
14 | const jpgData = dataUriToBuffer(
15 | "",
16 | );
17 | const bmpData = dataUriToBuffer(
18 | "",
19 | );
20 | const ktx2Data = dataUriToBuffer(
21 | "data:image/ktx2:base64,q0tUWCAyMLsNChoKAAAAAAEAAABAAAAAQAAAAAAAAAAAAAAAAQAAAAEAAAABAAAAaAAAACwAAACUAAAAdAAAAAgBAAAAAAAA7QEAAAAAAAD1AgAAAAAAAPgAAAAAAAAAAAAAAAAAAAAsAAAAAAAAAAIAKACjAQIAAwMAAAAAAAAAAAAAAAA/AAAAAAAAAAAA/////xIAAABLVFhvcmllbnRhdGlvbgByZAAAAFUAAABLVFh3cml0ZXIAdG9rdHggdjQuMC5iZXRhMi41My5nZjgzZDI4NDUuZGlydHkgLyBsaWJrdHggdjQuMC5iZXRhMi41MS5nM2EzMWQ0ZGEuZGlydHkAAAAAHQBfAEcAAAAcAQAAYgAAAAAAAAAAAAAAAAAAAPgAAAAAAAAAAAAAACDAGwABAIBgclixPD5qPIA3EAEAAAJpL1qO+wKACQAAAAAAAAQCEAEAAABACMSHxYxC5qRfZDX/tDYozqTrsZwcLKFPOj8HAAjmamgUQQAA4G9md89jvYMNJJDEE0erNdUVV1pMAlpY6II/hRiwULCxywNMUggi6NVWCjb+FBYpBRcLawsfwLyBDyAEEpo8f13mE8N3+MzWN8aG0uInpReLBNfESNpLUhJjn6b0UYqK6fRJOijLPA7LtfXfpawtHOM5wslUOelTK5qmmCyvsvvBs9vv7W4xDJxWwcfD2u9chVzP234bZjCbzdBt9+rlcHEnF9Uqax51Mg7aopqvbDx89+Bm8yd0dfjyNdzXnWaTS7dwHQMKYiRu4hcUupGEncw9fArUMQahsqgFgqPLjReLeCFOmjpQB/VCqMhnZzzd2+xeHe83j+sRN/Cff4yZ8Q0Cxnpj/0d75e6dw3y0evn6/C0AwdsAARAGgie2geTzWDtGyh9KgpX8WJb+BoILSDZI+sB3AA8ABEAQhjVfWhkigV6/gQJxMCXBAADD9G9HsB/tCHQCO40d4H1eHXIIwiGiqaHGIEjX+AaACQAAAAAAEMRAAPxNflpzjZbtuGAsfCOdQtK6+1SN1+ekY6j7t75MEIfKzChq81nw7nJCXzrVkyZprvXxESlhruXkPYHimL3u9o3k57aC4TXqLc9ZhbxM8VrYjcZ4NtT2+55ZPICCeHUISHGmLmU+/LNbHx0jlpkCmnpHFAZSaHnCJ49gGUlMs7+K4PcBQ9KG618GBGDlxTULGQlp8NdIpU8tdzAFkCSPVU20irfukO/YSpScsCHeCglbH2ljLGR+7McSAoyLd9j0k06gjHZdYiPGpi0n6fX7PCd22juv/0qvP5YIoHnrGZEXMiFaw9chnA2C8POpNi0EQu/lwCZoILII",
22 | );
23 | const basisData = dataUriToBuffer(
24 | "",
25 | );
26 | const textData = dataUriToBuffer("data:text/plain;charset=utf-8,randomtext");
27 |
28 | describe("getImageExtension", () => {
29 | it("gets image extension from buffer", () => {
30 | expect(getImageExtension(pngData)).toBe(".png");
31 | expect(getImageExtension(gifData)).toBe(".gif");
32 | expect(getImageExtension(jpgData)).toBe(".jpg");
33 | expect(getImageExtension(bmpData)).toBe(".bmp");
34 | expect(getImageExtension(ktx2Data)).toBe(".ktx2");
35 | expect(getImageExtension(basisData)).toBe(".basis");
36 | });
37 |
38 | it("throws error if buffer does not contain image data", () => {
39 | let thrownError;
40 | try {
41 | getImageExtension(textData);
42 | } catch (e) {
43 | thrownError = e;
44 | }
45 | expect(thrownError).toEqual(
46 | new RuntimeError("Image data does not have valid header"),
47 | );
48 | });
49 | });
50 |
--------------------------------------------------------------------------------
/specs/lib/getJsonBufferPaddedSpec.js:
--------------------------------------------------------------------------------
1 | "use strict";
2 | const getJsonBufferPadded = require("../../lib/getJsonBufferPadded");
3 |
4 | describe("getJsonBufferPadded", () => {
5 | it("get json buffer padded to 8 bytes", () => {
6 | const gltf = {
7 | asset: {
8 | version: "2.0",
9 | },
10 | };
11 | const string = JSON.stringify(gltf);
12 | expect(string.length).toBe(27);
13 | const bufferPadded = getJsonBufferPadded(gltf);
14 | expect(bufferPadded.length).toBe(32);
15 | expect(bufferPadded.readUInt8(27)).toBe(32); // Space
16 | expect(bufferPadded.readUInt8(28)).toBe(32);
17 | expect(bufferPadded.readUInt8(29)).toBe(32);
18 | expect(bufferPadded.readUInt8(30)).toBe(32);
19 | expect(bufferPadded.readUInt8(31)).toBe(32);
20 | });
21 |
22 | it("get json buffer padded to 8 bytes relative to byte offset", () => {
23 | const gltf = {
24 | asset: {
25 | version: "2.0",
26 | },
27 | };
28 | const string = JSON.stringify(gltf);
29 | expect(string.length).toBe(27);
30 | const bufferPadded = getJsonBufferPadded(gltf, 20);
31 | expect(bufferPadded.length).toBe(28);
32 | expect(bufferPadded.readUInt8(27)).toBe(32); // Space
33 | });
34 | });
35 |
--------------------------------------------------------------------------------
/specs/lib/getStatisticsSpec.js:
--------------------------------------------------------------------------------
1 | "use strict";
2 | const getStatistics = require("../../lib/getStatistics");
3 |
4 | describe("getStatistics", () => {
5 | const gltf = {
6 | accessors: [
7 | {
8 | componentType: 5123,
9 | type: "SCALAR",
10 | count: 1,
11 | },
12 | {
13 | componentType: 5123,
14 | type: "SCALAR",
15 | count: 2,
16 | },
17 | {
18 | componentType: 5123,
19 | type: "SCALAR",
20 | count: 6,
21 | },
22 | {
23 | componentType: 5126,
24 | type: "VEC3",
25 | count: 6,
26 | },
27 | ],
28 | buffers: [
29 | {
30 | byteLength: 140,
31 | },
32 | {
33 | byteLength: 120,
34 | uri: "buffer.bin",
35 | },
36 | ],
37 | images: [
38 | {
39 | uri: "image.png",
40 | },
41 | {
42 | uri: "data:image/png;",
43 | },
44 | {
45 | uri: "image2.png",
46 | },
47 | ],
48 | meshes: [
49 | {
50 | primitives: [
51 | {
52 | indices: 0,
53 | mode: 0, // POINTS
54 | },
55 | {
56 | indices: 1,
57 | mode: 1, // LINES
58 | },
59 | {
60 | attributes: {
61 | POSITION: 3,
62 | },
63 | },
64 | ],
65 | },
66 | {
67 | primitives: [
68 | {
69 | indices: 2,
70 | mode: 4, // TRIANGLES
71 | },
72 | ],
73 | },
74 | ],
75 | materials: [{}, {}],
76 | animations: [{}, {}, {}],
77 | nodes: [
78 | {
79 | name: "rootNode",
80 | mesh: 0,
81 | },
82 | ],
83 | };
84 |
85 | it("returns statistics for a gltf", () => {
86 | const statistics = getStatistics(gltf);
87 | expect(statistics.buffersByteLength).toEqual(260);
88 | expect(statistics.numberOfImages).toEqual(3);
89 | expect(statistics.numberOfExternalRequests).toEqual(3);
90 | expect(statistics.numberOfDrawCalls).toEqual(4);
91 | expect(statistics.numberOfRenderedPrimitives).toEqual(6);
92 | expect(statistics.numberOfNodes).toEqual(1);
93 | expect(statistics.numberOfMeshes).toEqual(2);
94 | expect(statistics.numberOfMaterials).toEqual(2);
95 | expect(statistics.numberOfAnimations).toEqual(3);
96 | });
97 |
98 | it("returns draw call statistics for a gltf node", () => {
99 | const statistics = getStatistics(gltf, 0);
100 | expect(statistics.numberOfDrawCalls).toEqual(3);
101 | expect(statistics.numberOfRenderedPrimitives).toEqual(4);
102 | });
103 | });
104 |
--------------------------------------------------------------------------------
/specs/lib/glbToGltfSpec.js:
--------------------------------------------------------------------------------
1 | "use strict";
2 | const fsExtra = require("fs-extra");
3 | const glbToGltf = require("../../lib/glbToGltf");
4 |
5 | const glbPath = "specs/data/2.0/box-textured-binary/box-textured-binary.glb";
6 |
7 | describe("glbToGltf", () => {
8 | it("glbToGltf", async () => {
9 | spyOn(console, "log");
10 | const glb = fsExtra.readFileSync(glbPath);
11 | const options = {
12 | separate: true,
13 | stats: true,
14 | };
15 | const results = await glbToGltf(glb, options);
16 | expect(results.gltf).toBeDefined();
17 | expect(results.separateResources).toBeDefined();
18 | expect(results.gltf.buffers.length).toBe(1);
19 | expect(console.log).toHaveBeenCalled();
20 | });
21 | });
22 |
--------------------------------------------------------------------------------
/specs/lib/gltfToGlbSpec.js:
--------------------------------------------------------------------------------
1 | "use strict";
2 | const fsExtra = require("fs-extra");
3 | const path = require("path");
4 |
5 | const gltfToGlb = require("../../lib/gltfToGlb");
6 | const parseGlb = require("../../lib/parseGlb");
7 |
8 | const gltfPath =
9 | "specs/data/2.0/box-textured-embedded/box-textured-embedded.gltf";
10 | const gltfMeshoptFallbackPath =
11 | "specs/data/2.0/extensions/EXT_meshopt_compression/meshopt-fallback/meshopt-fallback.gltf";
12 |
13 | describe("gltfToGlb", () => {
14 | it("gltfToGlb", async () => {
15 | spyOn(console, "log");
16 | const gltf = fsExtra.readJsonSync(gltfPath);
17 | const options = {
18 | separateTextures: true,
19 | stats: true,
20 | };
21 | const results = await gltfToGlb(gltf, options);
22 | const glb = results.glb;
23 | const separateResources = results.separateResources;
24 | expect(Buffer.isBuffer(glb)).toBe(true);
25 | expect(Object.keys(separateResources).length).toBe(1);
26 | expect(console.log).toHaveBeenCalled();
27 |
28 | // Header + JSON header + JSON content + binary header + binary content
29 | const glbLength = glb.readUInt32LE(8);
30 | const jsonChunkLength = glb.readUInt32LE(12);
31 | const binaryChunkLength = glb.readUInt32LE(12 + 8 + jsonChunkLength);
32 | const expectedLength = 12 + 8 + jsonChunkLength + 8 + binaryChunkLength;
33 | expect(glbLength).toBe(expectedLength);
34 | expect(glb.length).toBe(expectedLength);
35 | });
36 |
37 | it("gltfToGlb with separate resources", async () => {
38 | spyOn(console, "log");
39 | const gltf = fsExtra.readJsonSync(gltfPath);
40 | const options = {
41 | separate: true,
42 | stats: true,
43 | };
44 | const results = await gltfToGlb(gltf, options);
45 | const glb = results.glb;
46 | const separateResources = results.separateResources;
47 | expect(Buffer.isBuffer(glb)).toBe(true);
48 | expect(Object.keys(separateResources).length).toBe(2);
49 | expect(console.log).toHaveBeenCalled();
50 |
51 | // Header + JSON header + JSON content. No binary header or content.
52 | const glbLength = glb.readUInt32LE(8);
53 | const jsonChunkLength = glb.readUInt32LE(12);
54 | const expectedLength = 12 + 8 + jsonChunkLength;
55 | expect(glbLength).toBe(expectedLength);
56 | expect(glb.length).toBe(expectedLength);
57 | });
58 |
59 | it("gltfToGlb processes glTF with EXT_meshopt_compression extension.", async () => {
60 | const gltf = fsExtra.readJsonSync(gltfMeshoptFallbackPath);
61 | const options = {
62 | resourceDirectory: path.dirname(gltfMeshoptFallbackPath),
63 | };
64 | const results = await gltfToGlb(gltf, options);
65 | expect(results.glb).toBeDefined();
66 |
67 | const processedGltf = parseGlb(results.glb);
68 |
69 | expect(processedGltf).toBeDefined();
70 | expect(processedGltf.buffers.length).toBe(1);
71 |
72 | const buffer0 = processedGltf.buffers[0];
73 |
74 | const bufferView0 = processedGltf.bufferViews[0];
75 | const bufferView1 = processedGltf.bufferViews[1];
76 | const bufferView2 = processedGltf.bufferViews[2];
77 |
78 | const meshoptObject0 = bufferView0.extensions.EXT_meshopt_compression;
79 | const meshoptObject1 = bufferView1.extensions.EXT_meshopt_compression;
80 | const meshoptObject2 = bufferView2.extensions.EXT_meshopt_compression;
81 |
82 | expect(buffer0.byteLength).toBe(1432);
83 | expect(buffer0.uri).not.toBeDefined();
84 |
85 | expect(bufferView0.buffer).toBe(0);
86 | expect(bufferView0.byteOffset).toBe(0);
87 | expect(bufferView0.byteLength).toBe(400);
88 | expect(meshoptObject0.buffer).toBe(0);
89 | expect(meshoptObject0.byteOffset).toBe(400);
90 | expect(meshoptObject0.byteLength).toBe(265);
91 |
92 | expect(bufferView1.buffer).toBe(0);
93 | expect(bufferView1.byteOffset).toBe(672);
94 | expect(bufferView1.byteLength).toBe(200);
95 | expect(meshoptObject1.buffer).toBe(0);
96 | expect(meshoptObject1.byteOffset).toBe(872);
97 | expect(meshoptObject1.byteLength).toBe(117);
98 |
99 | expect(bufferView2.buffer).toBe(0);
100 | expect(bufferView2.byteOffset).toBe(992);
101 | expect(bufferView2.byteLength).toBe(360);
102 | expect(meshoptObject2.buffer).toBe(0);
103 | expect(meshoptObject2.byteOffset).toBe(1352);
104 | expect(meshoptObject2.byteLength).toBe(78);
105 | });
106 |
107 | it("gltfToGlb processes glTF with EXT_meshopt_compression extension with separate resources.", async () => {
108 | const gltf = fsExtra.readJsonSync(gltfMeshoptFallbackPath);
109 | const options = {
110 | separate: true,
111 | resourceDirectory: path.dirname(gltfMeshoptFallbackPath),
112 | };
113 | const results = await gltfToGlb(gltf, options);
114 | expect(results.glb).toBeDefined();
115 |
116 | const processedGltf = parseGlb(results.glb);
117 |
118 | expect(processedGltf).toBeDefined();
119 | expect(processedGltf.buffers.length).toBe(2);
120 |
121 | const buffer0 = processedGltf.buffers[0];
122 | const buffer1 = processedGltf.buffers[1];
123 |
124 | const bufferView0 = gltf.bufferViews[0];
125 | const bufferView1 = gltf.bufferViews[1];
126 | const bufferView2 = gltf.bufferViews[2];
127 |
128 | const meshoptObject0 = bufferView0.extensions.EXT_meshopt_compression;
129 | const meshoptObject1 = bufferView1.extensions.EXT_meshopt_compression;
130 | const meshoptObject2 = bufferView2.extensions.EXT_meshopt_compression;
131 |
132 | expect(buffer0.byteLength).toBe(960);
133 | expect(buffer0.uri).toBe("meshopt-fallback-meshopt-fallback-1.bin");
134 | expect(buffer1.byteLength).toBe(472);
135 | expect(buffer1.uri).toBe("meshopt-fallback.bin");
136 |
137 | expect(bufferView0.buffer).toBe(0);
138 | expect(bufferView0.byteOffset).toBe(0);
139 | expect(bufferView0.byteLength).toBe(400);
140 | expect(meshoptObject0.buffer).toBe(1);
141 | expect(meshoptObject0.byteOffset).toBe(0);
142 | expect(meshoptObject0.byteLength).toBe(265);
143 |
144 | expect(bufferView1.buffer).toBe(0);
145 | expect(bufferView1.byteOffset).toBe(400);
146 | expect(bufferView1.byteLength).toBe(200);
147 | expect(meshoptObject1.buffer).toBe(1);
148 | expect(meshoptObject1.byteOffset).toBe(272);
149 | expect(meshoptObject1.byteLength).toBe(117);
150 |
151 | expect(bufferView2.buffer).toBe(0);
152 | expect(bufferView2.byteOffset).toBe(600);
153 | expect(bufferView2.byteLength).toBe(360);
154 | expect(meshoptObject2.buffer).toBe(1);
155 | expect(meshoptObject2.byteOffset).toBe(392);
156 | expect(meshoptObject2.byteLength).toBe(78);
157 | });
158 | });
159 |
--------------------------------------------------------------------------------
/specs/lib/moveTechniquesToExtensionSpec.js:
--------------------------------------------------------------------------------
1 | "use strict";
2 | const Cesium = require("cesium");
3 | const moveTechniquesToExtension = require("../../lib/moveTechniquesToExtension");
4 |
5 | const WebGLConstants = Cesium.WebGLConstants;
6 | describe("moveTechniquesToExtension", () => {
7 | it("moves techniques, shaders, and programs to extension", () => {
8 | const gltf = {
9 | programs: {
10 | program_0: {
11 | attributes: ["a_normal", "a_position", "a_texcoord0"],
12 | fragmentShader: "BoxTextured0FS",
13 | vertexShader: "BoxTextured0VS",
14 | },
15 | program_1: {
16 | attributes: ["a_normal", "a_position", "a_texcoord0"],
17 | fragmentShader: "BoxTextured1FS",
18 | vertexShader: "BoxTextured0VS",
19 | },
20 | },
21 | shaders: {
22 | BoxTextured0FS: {
23 | type: WebGLConstants.FRAGMENT_SHADER,
24 | uri: "BoxTextured0FS.glsl",
25 | },
26 | BoxTextured0VS: {
27 | type: WebGLConstants.VERTEX_SHADER,
28 | uri: "BoxTextured0VS.glsl",
29 | },
30 | BoxTextured1FS: {
31 | type: WebGLConstants.FRAGMENT_SHADER,
32 | uri: "BoxTextured1FS.glsl",
33 | },
34 | },
35 | techniques: {
36 | technique0: {
37 | attributes: {
38 | a_normal: "normal",
39 | a_position: "position",
40 | a_texcoord0: "texcoord0",
41 | },
42 | parameters: {
43 | diffuse: {
44 | type: WebGLConstants.SAMPLER_2D,
45 | },
46 | modelViewMatrix: {
47 | semantic: "MODELVIEW",
48 | type: WebGLConstants.FLOAT_MAT4,
49 | },
50 | normal: {
51 | semantic: "NORMAL",
52 | type: WebGLConstants.FLOAT_VEC3,
53 | },
54 | normalMatrix: {
55 | semantic: "MODELVIEWINVERSETRANSPOSE",
56 | type: WebGLConstants.FLOAT_MAT3,
57 | },
58 | position: {
59 | semantic: "POSITION",
60 | type: WebGLConstants.FLOAT_VEC3,
61 | },
62 | projectionMatrix: {
63 | semantic: "PROJECTION",
64 | type: WebGLConstants.FLOAT_MAT4,
65 | },
66 | shininess: {
67 | type: WebGLConstants.FLOAT,
68 | },
69 | specular: {
70 | type: WebGLConstants.FLOAT_VEC4,
71 | },
72 | texcoord0: {
73 | semantic: "TEXCOORD_0",
74 | type: WebGLConstants.FLOAT_VEC2,
75 | },
76 | },
77 | program: "program_0",
78 | states: {
79 | enable: [WebGLConstants.DEPTH_TEST, WebGLConstants.CULL_FACE],
80 | },
81 | uniforms: {
82 | u_diffuse: "diffuse",
83 | u_modelViewMatrix: "modelViewMatrix",
84 | u_normalMatrix: "normalMatrix",
85 | u_projectionMatrix: "projectionMatrix",
86 | u_shininess: "shininess",
87 | u_specular: "specular",
88 | },
89 | },
90 | technique1: {
91 | program: "program_1",
92 | },
93 | technique2: {
94 | parameters: {
95 | diffuse: {
96 | type: WebGLConstants.FLOAT_VEC4,
97 | },
98 | },
99 | program: "program_0",
100 | uniforms: {
101 | u_diffuse: "diffuse",
102 | },
103 | },
104 | },
105 | materials: [
106 | {
107 | name: "Texture",
108 | technique: "technique0",
109 | values: {
110 | diffuse: "texture_Image0001",
111 | shininess: 256,
112 | specular: [0.2, 0.2, 0.2, 1],
113 | },
114 | },
115 | {
116 | name: "Color",
117 | technique: "technique2",
118 | values: {
119 | diffuse: [0.2, 0.2, 0.2, 1],
120 | },
121 | },
122 | ],
123 | };
124 |
125 | const gltfWithTechniquesWebgl = moveTechniquesToExtension(gltf);
126 | expect(gltfWithTechniquesWebgl.extensions).toBeDefined();
127 | const techniques = gltfWithTechniquesWebgl.extensions.KHR_techniques_webgl;
128 | expect(techniques).toBeDefined();
129 | expect(techniques.techniques.length).toBe(3);
130 |
131 | const technique = techniques.techniques[0];
132 | const attributes = technique.attributes;
133 | expect(attributes).toBeDefined();
134 | expect(attributes.a_position.semantic).toBe("POSITION");
135 |
136 | const uniforms = technique.uniforms;
137 | expect(uniforms).toBeDefined();
138 | expect(uniforms.u_modelViewMatrix.semantic).toBe("MODELVIEW");
139 | expect(uniforms.u_modelViewMatrix.type).toBe(WebGLConstants.FLOAT_MAT4);
140 |
141 | expect(technique.program).toBe(0);
142 | expect(technique.parameters).toBeUndefined();
143 | expect(technique.states).toBeUndefined();
144 |
145 | expect(techniques.programs.length).toBe(2);
146 | const program = techniques.programs[technique.program];
147 | expect(program).toBeDefined();
148 |
149 | expect(techniques.shaders.length).toBe(3);
150 | expect(techniques.shaders[program.fragmentShader].type).toBe(
151 | WebGLConstants.FRAGMENT_SHADER,
152 | );
153 | expect(techniques.shaders[program.vertexShader].type).toBe(
154 | WebGLConstants.VERTEX_SHADER,
155 | );
156 |
157 | expect(gltfWithTechniquesWebgl.techniques).toBeUndefined();
158 | expect(gltfWithTechniquesWebgl.programs).toBeUndefined();
159 | expect(gltfWithTechniquesWebgl.shaders).toBeUndefined();
160 |
161 | const material = gltf.materials[0];
162 | expect(material.extensions).toBeDefined();
163 | const materialTechniques = material.extensions.KHR_techniques_webgl;
164 | expect(materialTechniques).toBeDefined();
165 | expect(materialTechniques.technique).toBe(0);
166 | expect(materialTechniques.values.u_shininess).toBe(256);
167 | expect(materialTechniques.values.u_diffuse).toBe("texture_Image0001");
168 |
169 | expect(material.technique).toBeUndefined();
170 | expect(material.values).toBeUndefined();
171 |
172 | const technique2 = techniques.techniques[1];
173 | const program2 = techniques.programs[technique2.program];
174 | expect(program2.vertexShader).toBe(program.vertexShader);
175 | expect(program2.fragmentShader).not.toBe(program.fragmentShader);
176 |
177 | const technique3 = techniques.techniques[2];
178 | expect(technique3.program).toBe(0);
179 | expect(technique3.uniforms.u_diffuse.type).toBe(WebGLConstants.FLOAT_VEC4);
180 | expect(
181 | gltf.materials[1].extensions.KHR_techniques_webgl.values.u_diffuse,
182 | ).toEqual([0.2, 0.2, 0.2, 1.0]);
183 | });
184 | });
185 |
--------------------------------------------------------------------------------
/specs/lib/numberOfComponentsForTypeSpec.js:
--------------------------------------------------------------------------------
1 | "use strict";
2 | const numberOfComponentsForType = require("../../lib/numberOfComponentsForType");
3 |
4 | describe("numberOfComponentsForType", () => {
5 | it("numberOfComponentsForType", () => {
6 | expect(numberOfComponentsForType("SCALAR")).toBe(1);
7 | expect(numberOfComponentsForType("VEC2")).toBe(2);
8 | expect(numberOfComponentsForType("VEC3")).toBe(3);
9 | expect(numberOfComponentsForType("VEC4")).toBe(4);
10 | expect(numberOfComponentsForType("MAT2")).toBe(4);
11 | expect(numberOfComponentsForType("MAT3")).toBe(9);
12 | expect(numberOfComponentsForType("MAT4")).toBe(16);
13 | });
14 | });
15 |
--------------------------------------------------------------------------------
/specs/lib/parseGlbSpec.js:
--------------------------------------------------------------------------------
1 | "use strict";
2 |
3 | const { RuntimeError } = require("cesium");
4 |
5 | const parseGlb = require("../../lib/parseGlb");
6 | const removePipelineExtras = require("../../lib/removePipelineExtras");
7 |
8 | describe("parseGlb", () => {
9 | it("throws an error with invalid magic", () => {
10 | const glb = Buffer.alloc(20);
11 | glb.write("NOPE", 0);
12 |
13 | let thrownError;
14 | try {
15 | parseGlb(glb);
16 | } catch (e) {
17 | thrownError = e;
18 | }
19 | expect(thrownError).toEqual(
20 | new RuntimeError("File is not valid binary glTF"),
21 | );
22 | });
23 |
24 | it("throws an error if version is not 1 or 2", () => {
25 | const glb = Buffer.alloc(20);
26 | glb.write("glTF", 0);
27 | glb.writeUInt32LE(3, 4);
28 |
29 | let thrownError;
30 | try {
31 | parseGlb(glb);
32 | } catch (e) {
33 | thrownError = e;
34 | }
35 | expect(thrownError).toEqual(
36 | new RuntimeError("Binary glTF version is not 1 or 2"),
37 | );
38 | });
39 |
40 | describe("1.0", () => {
41 | it("throws an error if content format is not JSON", () => {
42 | const glb = Buffer.alloc(20);
43 | glb.write("glTF", 0);
44 | glb.writeUInt32LE(1, 4);
45 | glb.writeUInt32LE(20, 8);
46 | glb.writeUInt32LE(0, 12);
47 | glb.writeUInt32LE(1, 16);
48 |
49 | let thrownError;
50 | try {
51 | parseGlb(glb);
52 | } catch (e) {
53 | thrownError = e;
54 | }
55 | expect(thrownError).toEqual(
56 | new RuntimeError("Binary glTF scene format is not JSON"),
57 | );
58 | });
59 |
60 | it("loads binary glTF", () => {
61 | const binaryData = Buffer.from([0, 1, 2, 3, 4, 5]);
62 | const gltf = {
63 | bufferViews: {
64 | imageBufferView: {
65 | byteLength: 0,
66 | },
67 | shaderBufferView: {
68 | byteLength: 0,
69 | },
70 | },
71 | buffers: {
72 | binary_glTF: {
73 | byteLength: binaryData.length,
74 | uri: "data:,",
75 | },
76 | },
77 | images: {
78 | image: {
79 | extensions: {
80 | KHR_binary_glTF: {
81 | bufferView: "imageBufferView",
82 | mimeType: "image/jpg",
83 | },
84 | },
85 | },
86 | },
87 | shaders: {
88 | shader: {
89 | extensions: {
90 | KHR_binary_glTF: {
91 | bufferView: "shaderBufferView",
92 | },
93 | },
94 | },
95 | },
96 | extensionsUsed: ["KHR_binary_glTF"],
97 | };
98 | let gltfString = JSON.stringify(gltf);
99 | while (gltfString.length % 4 !== 0) {
100 | gltfString += " ";
101 | }
102 | const glb = Buffer.alloc(20 + gltfString.length + binaryData.length);
103 | glb.write("glTF", 0);
104 | glb.writeUInt32LE(1, 4);
105 | glb.writeUInt32LE(20 + gltfString.length + binaryData.length, 8);
106 | glb.writeUInt32LE(gltfString.length, 12);
107 | glb.writeUInt32LE(0, 16);
108 | glb.write(gltfString, 20);
109 | binaryData.copy(glb, 20 + gltfString.length);
110 |
111 | const parsedGltf = parseGlb(glb);
112 | expect(parsedGltf.extensionsUsed).toBeUndefined();
113 | const buffer = parsedGltf.buffers.binary_glTF;
114 | for (let i = 0; i < binaryData.length; i++) {
115 | expect(buffer.extras._pipeline.source[i]).toEqual(binaryData[i]);
116 | expect(buffer.uri).toBeUndefined();
117 | }
118 |
119 | const image = parsedGltf.images.image;
120 | expect(image.extensions.KHR_binary_glTF).toBeDefined();
121 | expect(image.extensions.KHR_binary_glTF.bufferView).toBe(
122 | "imageBufferView",
123 | );
124 | expect(image.extensions.KHR_binary_glTF.mimeType).toBe("image/jpg");
125 | const shader = parsedGltf.shaders.shader;
126 | expect(shader.extensions.KHR_binary_glTF).toBeDefined();
127 | expect(shader.extensions.KHR_binary_glTF.bufferView).toBe(
128 | "shaderBufferView",
129 | );
130 | });
131 | });
132 |
133 | describe("2.0", () => {
134 | it("loads binary glTF", () => {
135 | let i;
136 | const binaryData = Buffer.from([0, 1, 2, 3, 4, 5]);
137 | const gltf = {
138 | asset: {
139 | version: "2.0",
140 | },
141 | buffers: [
142 | {
143 | byteLength: binaryData.length,
144 | },
145 | ],
146 | images: [
147 | {
148 | bufferView: 0,
149 | mimeType: "image/jpg",
150 | },
151 | ],
152 | };
153 | let gltfString = JSON.stringify(gltf);
154 | while (gltfString.length % 4 !== 0) {
155 | gltfString += " ";
156 | }
157 | const glb = Buffer.alloc(28 + gltfString.length + binaryData.length);
158 | glb.write("glTF", 0);
159 | glb.writeUInt32LE(2, 4);
160 | glb.writeUInt32LE(12 + 8 + gltfString.length + 8 + binaryData.length, 8);
161 | glb.writeUInt32LE(gltfString.length, 12);
162 | glb.writeUInt32LE(0x4e4f534a, 16);
163 | glb.write(gltfString, 20);
164 | glb.writeUInt32LE(binaryData.length, 20 + gltfString.length);
165 | glb.writeUInt32LE(0x004e4942, 24 + gltfString.length);
166 | binaryData.copy(glb, 28 + gltfString.length);
167 |
168 | const parsedGltf = parseGlb(glb);
169 | const buffer = parsedGltf.buffers[0];
170 | for (i = 0; i < binaryData.length; i++) {
171 | expect(buffer.extras._pipeline.source[i]).toEqual(binaryData[i]);
172 | }
173 | removePipelineExtras(parsedGltf);
174 | expect(parsedGltf).toEqual(gltf);
175 | });
176 | });
177 | });
178 |
--------------------------------------------------------------------------------
/specs/lib/processGlbSpec.js:
--------------------------------------------------------------------------------
1 | "use strict";
2 | const fsExtra = require("fs-extra");
3 | const processGlb = require("../../lib/processGlb");
4 |
5 | const glbPath = "specs/data/2.0/box-textured-binary/box-textured-binary.glb";
6 |
7 | describe("processGlb", () => {
8 | it("processGlb", async () => {
9 | spyOn(console, "log");
10 | const glb = fsExtra.readFileSync(glbPath);
11 | const options = {
12 | separate: true,
13 | stats: true,
14 | };
15 | const results = await processGlb(glb, options);
16 | expect(Buffer.isBuffer(results.glb)).toBe(true);
17 | expect(results.separateResources).toBeDefined();
18 | expect(console.log).toHaveBeenCalled();
19 | });
20 | });
21 |
--------------------------------------------------------------------------------
/specs/lib/readAccessorPackedSpec.js:
--------------------------------------------------------------------------------
1 | "use strict";
2 | const readAccessorPacked = require("../../lib/readAccessorPacked");
3 | const readResources = require("../../lib/readResources");
4 |
5 | const contiguousData = [
6 | -1.0, 1.0, -1.0, 0.0, 0.0, 0.0, 3.0, 2.0, 1.0, -1.0, -2.0, -3.0,
7 | ];
8 |
9 | const nan = Number.NaN;
10 | const nonContiguousData = [
11 | -1.0,
12 | 1.0,
13 | -1.0,
14 | nan,
15 | nan,
16 | nan,
17 | 0.0,
18 | 0.0,
19 | 0.0,
20 | nan,
21 | nan,
22 | nan,
23 | 3.0,
24 | 2.0,
25 | 1.0,
26 | nan,
27 | nan,
28 | nan,
29 | -1.0,
30 | -2.0,
31 | -3.0,
32 | nan,
33 | nan,
34 | nan,
35 | ];
36 |
37 | function createGltf(elements, byteStride) {
38 | const buffer = Buffer.from(new Float32Array(elements).buffer);
39 | const byteLength = buffer.length;
40 | const dataUri = `data:application/octet-stream;base64,${buffer.toString(
41 | "base64",
42 | )}`;
43 | const gltf = {
44 | accessors: [
45 | {
46 | bufferView: 0,
47 | byteOffset: 0,
48 | componentType: 5126,
49 | count: 4,
50 | type: "VEC3",
51 | },
52 | ],
53 | bufferViews: [
54 | {
55 | buffer: 0,
56 | byteOffset: 0,
57 | byteLength: byteLength,
58 | byteStride: byteStride,
59 | },
60 | ],
61 | buffers: [
62 | {
63 | uri: dataUri,
64 | byteLength: byteLength,
65 | },
66 | ],
67 | };
68 | return readResources(gltf);
69 | }
70 |
71 | describe("readAccessorPacked", () => {
72 | it("reads contiguous accessor", async () => {
73 | const gltf = await createGltf(contiguousData, 12);
74 | expect(readAccessorPacked(gltf, gltf.accessors[0])).toEqual(contiguousData);
75 | });
76 |
77 | it("reads non-contiguous accessor", async () => {
78 | const gltf = await createGltf(nonContiguousData, 24);
79 | expect(readAccessorPacked(gltf, gltf.accessors[0])).toEqual(contiguousData);
80 | });
81 |
82 | it("reads accessor that does not have a buffer view", () => {
83 | const gltf = {
84 | accessors: [
85 | {
86 | componentType: 5126,
87 | count: 4,
88 | type: "VEC3",
89 | },
90 | ],
91 | };
92 | const expected = new Array(12).fill(0);
93 | expect(readAccessorPacked(gltf, gltf.accessors[0])).toEqual(expected);
94 | });
95 | });
96 |
--------------------------------------------------------------------------------
/specs/lib/readResourcesSpec.js:
--------------------------------------------------------------------------------
1 | "use strict";
2 | const Cesium = require("cesium");
3 | const fsExtra = require("fs-extra");
4 | const path = require("path");
5 | const ForEach = require("../../lib/ForEach");
6 | const parseGlb = require("../../lib/parseGlb");
7 | const readResources = require("../../lib/readResources");
8 |
9 | const RuntimeError = Cesium.RuntimeError;
10 |
11 | const boxTexturedSeparate1Path =
12 | "specs/data/1.0/box-textured-separate/box-textured-separate.gltf";
13 | const boxTexturedBinarySeparate1Path =
14 | "specs/data/1.0/box-textured-binary-separate/box-textured-binary-separate.glb";
15 | const boxTexturedBinary1Path =
16 | "specs/data/1.0/box-textured-binary/box-textured-binary.glb";
17 | const boxTexturedEmbedded1Path =
18 | "specs/data/1.0/box-textured-embedded/box-textured-embedded.gltf";
19 | const boxTexturedSeparate2Path =
20 | "specs/data/2.0/box-textured-separate/box-textured-separate.gltf";
21 | const boxTexturedBinarySeparate2Path =
22 | "specs/data/2.0/box-textured-binary-separate/box-textured-binary-separate.glb";
23 | const boxTexturedBinary2Path =
24 | "specs/data/2.0/box-textured-binary/box-textured-binary.glb";
25 | const boxTexturedEmbedded2Path =
26 | "specs/data/2.0/box-textured-embedded/box-textured-embedded.gltf";
27 |
28 | function readGltf(gltfPath, binary) {
29 | if (binary) {
30 | const glb = fsExtra.readFileSync(gltfPath);
31 | return parseGlb(glb);
32 | }
33 | return fsExtra.readJsonSync(gltfPath);
34 | }
35 |
36 | function checkPaths(object, resourceDirectory) {
37 | const pipelineExtras = object.extras._pipeline;
38 | const absolutePath = pipelineExtras.absolutePath;
39 | const relativePath = pipelineExtras.relativePath;
40 | expect(path.basename(relativePath)).toBe(relativePath);
41 | expect(absolutePath).toBe(path.join(resourceDirectory, relativePath));
42 | expect(object.name).toBe(
43 | path.basename(relativePath, path.extname(relativePath)),
44 | );
45 | }
46 |
47 | async function readsResources(gltfPath, binary, separate) {
48 | const gltf = readGltf(gltfPath, binary);
49 | const resourceDirectory = path.resolve(path.dirname(gltfPath));
50 | const options = {
51 | resourceDirectory: resourceDirectory,
52 | };
53 | await readResources(gltf, options);
54 | ForEach.shader(gltf, (shader) => {
55 | const shaderText = shader.extras._pipeline.source;
56 | expect(typeof shaderText === "string").toBe(true);
57 | expect(shaderText.length).toBeGreaterThan(0);
58 | expect(shader.uri).toBeUndefined();
59 | if (separate) {
60 | checkPaths(shader, resourceDirectory);
61 | }
62 | });
63 | ForEach.image(gltf, (image) => {
64 | const imageSource = image.extras._pipeline.source;
65 | expect(Buffer.isBuffer(imageSource)).toBe(true);
66 | expect(image.uri).toBeUndefined();
67 | if (separate) {
68 | checkPaths(image, resourceDirectory);
69 | }
70 | });
71 | ForEach.buffer(gltf, (buffer) => {
72 | const bufferSource = buffer.extras._pipeline.source;
73 | expect(Buffer.isBuffer(bufferSource)).toBe(true);
74 | expect(buffer.uri).toBeUndefined();
75 | if (separate && !binary) {
76 | checkPaths(buffer, resourceDirectory);
77 | }
78 | });
79 | }
80 |
81 | describe("readResources", () => {
82 | it("reads separate resources from 1.0 model", async () => {
83 | await readsResources(boxTexturedSeparate1Path, false, true);
84 | });
85 |
86 | it("reads separate resources from 1.0 glb", async () => {
87 | await readsResources(boxTexturedBinarySeparate1Path, true, true);
88 | });
89 |
90 | it("reads embedded resources from 1.0 model", async () => {
91 | await readsResources(boxTexturedEmbedded1Path, false, false);
92 | });
93 |
94 | it("reads resources from 1.0 glb", async () => {
95 | await readsResources(boxTexturedBinary1Path, true, false);
96 | });
97 |
98 | it("reads separate resources from model", async () => {
99 | await readsResources(boxTexturedSeparate2Path, false, true);
100 | });
101 |
102 | it("reads separate resources from glb", async () => {
103 | await readsResources(boxTexturedBinarySeparate2Path, true, true);
104 | });
105 |
106 | it("reads embedded resources from model", async () => {
107 | await readsResources(boxTexturedEmbedded2Path, false, false);
108 | });
109 |
110 | it("reads resources from glb", async () => {
111 | await readsResources(boxTexturedBinary2Path, true, false);
112 | });
113 |
114 | it("rejects if gltf contains separate resources but no resource directory is supplied", async () => {
115 | const gltf = readGltf(boxTexturedSeparate2Path);
116 |
117 | let thrownError;
118 | try {
119 | await readResources(gltf);
120 | } catch (e) {
121 | thrownError = e;
122 | }
123 | expect(thrownError).toEqual(
124 | new RuntimeError(
125 | "glTF model references separate files but no resourceDirectory is supplied",
126 | ),
127 | );
128 | });
129 | });
130 |
--------------------------------------------------------------------------------
/specs/lib/removeDefaultsSpec.js:
--------------------------------------------------------------------------------
1 | "use strict";
2 | const removeDefaults = require("../../lib/removeDefaults");
3 |
4 | describe("removeDefaults", () => {
5 | it("removeDefaults", () => {
6 | const gltf = {
7 | nodes: [
8 | {
9 | matrix: [1, 0, 0, 0, 0, 1, 0, 0, 0, 0, 1, 0, 0, 0, 0, 1],
10 | },
11 | {
12 | matrix: [2, 0, 0, 0, 0, 1, 0, 0, 0, 0, 1, 0, 0, 0, 0, 1],
13 | },
14 | ],
15 | accessors: [
16 | {
17 | bufferView: 0,
18 | normalized: false,
19 | },
20 | ],
21 | };
22 | removeDefaults(gltf);
23 | expect(gltf.nodes[0].matrix).toBeUndefined();
24 | expect(gltf.nodes[1].matrix).toBeDefined();
25 | expect(gltf.accessors[0].normalized).toBeUndefined();
26 | });
27 | });
28 |
--------------------------------------------------------------------------------
/specs/lib/removeExtensionSpec.js:
--------------------------------------------------------------------------------
1 | "use strict";
2 | const Cesium = require("cesium");
3 | const removeExtension = require("../../lib/removeExtension");
4 |
5 | const WebGLConstants = Cesium.WebGLConstants;
6 |
7 | describe("removeExtension", () => {
8 | it("removes extension", () => {
9 | const gltf = {
10 | extensionsRequired: ["extension1", "extension2", "extension3"],
11 | extensionsUsed: ["extension1", "extension2", "extension3"],
12 | extensions: {
13 | extension1: {
14 | value: 9,
15 | },
16 | extension2: [0, 1, 2],
17 | },
18 | materials: [
19 | {
20 | baseColorFactor: [1.0, 0.0, 0.0, 1.0],
21 | extensions: {
22 | extension1: {
23 | value: 10,
24 | },
25 | },
26 | },
27 | {
28 | baseColorFactor: [0.0, 0.0, 1.0, 1.0],
29 | extensions: {
30 | extension1: {
31 | value: 11,
32 | },
33 | },
34 | },
35 | ],
36 | cameras: [
37 | {
38 | extensions: {
39 | extension1: {
40 | value: 9,
41 | },
42 | extension2: [3, 4, 5],
43 | },
44 | },
45 | ],
46 | };
47 | const extension1 = removeExtension(gltf, "extension1");
48 | expect(gltf.extensionsRequired).toEqual(["extension2", "extension3"]);
49 | expect(gltf.extensionsUsed).toEqual(["extension2", "extension3"]);
50 | expect(gltf.extensions).toEqual({
51 | extension2: [0, 1, 2],
52 | });
53 | expect(gltf.materials[0].extensions).toBeUndefined();
54 | expect(gltf.materials[1].extensions).toBeUndefined();
55 | expect(gltf.cameras[0].extensions).toEqual({
56 | extension2: [3, 4, 5],
57 | });
58 | expect(extension1).toEqual({
59 | value: 9,
60 | });
61 |
62 | const extension2 = removeExtension(gltf, "extension2");
63 | expect(gltf.extensionsRequired).toEqual(["extension3"]);
64 | expect(gltf.extensionsUsed).toEqual(["extension3"]);
65 | expect(gltf.extensions).toBeUndefined();
66 | expect(gltf.materials[0].extensions).toBeUndefined();
67 | expect(gltf.materials[1].extensions).toBeUndefined();
68 | expect(gltf.cameras[0].extensions).toBeUndefined();
69 | expect(extension2).toEqual([0, 1, 2]);
70 |
71 | const extension3 = removeExtension(gltf, "extension3");
72 | expect(gltf.extensionsRequired).toBeUndefined();
73 | expect(gltf.extensionsUsed).toBeUndefined();
74 | expect(gltf.extensions).toBeUndefined();
75 | expect(gltf.materials[0].extensions).toBeUndefined();
76 | expect(gltf.materials[1].extensions).toBeUndefined();
77 | expect(gltf.cameras[0].extensions).toBeUndefined();
78 | expect(extension3).toBeUndefined();
79 |
80 | const emptyGltf = {};
81 | removeExtension(gltf, "extension1");
82 | expect(emptyGltf).toEqual({});
83 | });
84 |
85 | it("removes CESIUM_RTC extension", () => {
86 | const gltf = {
87 | extensionsRequired: ["CESIUM_RTC", "KHR_techniques_webgl"],
88 | extensionsUsed: ["CESIUM_RTC", "KHR_techniques_webgl"],
89 | extensions: {
90 | CESIUM_RTC: {
91 | center: [1.0, 2.0, 3.0],
92 | },
93 | KHR_techniques_webgl: {
94 | techniques: [
95 | {
96 | uniforms: {
97 | u_modelViewMatrix: {
98 | type: WebGLConstants.FLOAT_MAT4,
99 | semantic: "CESIUM_RTC_MODELVIEW",
100 | },
101 | },
102 | },
103 | ],
104 | },
105 | },
106 | };
107 | const extension = removeExtension(gltf, "CESIUM_RTC");
108 | expect(extension).toEqual({
109 | center: [1.0, 2.0, 3.0],
110 | });
111 | expect(gltf.extensionsRequired).toEqual(["KHR_techniques_webgl"]);
112 | expect(gltf.extensionsUsed).toEqual(["KHR_techniques_webgl"]);
113 | expect(
114 | gltf.extensions.KHR_techniques_webgl.techniques[0].uniforms
115 | .u_modelViewMatrix.semantic,
116 | ).toBe("MODELVIEW");
117 | });
118 | });
119 |
--------------------------------------------------------------------------------
/specs/lib/removeExtensionsRequiredSpec.js:
--------------------------------------------------------------------------------
1 | "use strict";
2 | const removeExtensionsRequired = require("../../lib/removeExtensionsRequired");
3 |
4 | describe("removeExtensionsRequired", () => {
5 | it("removes extension from extensionsRequired", () => {
6 | const gltf = {
7 | extensionsRequired: ["extension1", "extension2"],
8 | extensionsUsed: ["extension1", "extension2"],
9 | };
10 | removeExtensionsRequired(gltf, "extension1");
11 | expect(gltf.extensionsRequired).toEqual(["extension2"]);
12 | removeExtensionsRequired(gltf, "extension2");
13 | expect(gltf.extensionsRequired).toBeUndefined();
14 | expect(gltf.extensionsUsed).toEqual(["extension1", "extension2"]);
15 |
16 | const emptyGltf = {};
17 | removeExtensionsRequired(gltf, "extension1");
18 | expect(emptyGltf).toEqual({});
19 | });
20 | });
21 |
--------------------------------------------------------------------------------
/specs/lib/removeExtensionsUsedSpec.js:
--------------------------------------------------------------------------------
1 | "use strict";
2 | const removeExtensionsUsed = require("../../lib/removeExtensionsUsed");
3 |
4 | describe("removeExtensionsUsed", () => {
5 | it("removes extension from extensionsUsed", () => {
6 | const gltf = {
7 | extensionsRequired: ["extension1", "extension2"],
8 | extensionsUsed: ["extension1", "extension2"],
9 | };
10 | removeExtensionsUsed(gltf, "extension1");
11 | expect(gltf.extensionsRequired).toEqual(["extension2"]);
12 | expect(gltf.extensionsUsed).toEqual(["extension2"]);
13 |
14 | removeExtensionsUsed(gltf, "extension2");
15 | expect(gltf.extensionsRequired).toBeUndefined();
16 | expect(gltf.extensionsUsed).toBeUndefined();
17 |
18 | const emptyGltf = {};
19 | removeExtensionsUsed(gltf, "extension1");
20 | expect(emptyGltf).toEqual({});
21 | });
22 | });
23 |
--------------------------------------------------------------------------------
/specs/lib/removePipelineExtrasSpec.js:
--------------------------------------------------------------------------------
1 | "use strict";
2 | const Cesium = require("cesium");
3 | const addPipelineExtras = require("../../lib/addPipelineExtras");
4 | const removePipelineExtras = require("../../lib/removePipelineExtras");
5 |
6 | const WebGLConstants = Cesium.WebGLConstants;
7 |
8 | describe("removePipelineExtras", () => {
9 | it("removes pipeline extras", () => {
10 | const gltf = {
11 | buffers: [
12 | {
13 | byteLength: 100,
14 | },
15 | ],
16 | extensions: {
17 | KHR_techniques_webgl: {
18 | shaders: [
19 | {
20 | type: WebGLConstants.VERTEX_SHADER,
21 | uri: "data:,",
22 | },
23 | ],
24 | },
25 | },
26 | };
27 | const gltfWithExtrasRemoved = removePipelineExtras(addPipelineExtras(gltf));
28 | expect(gltfWithExtrasRemoved.buffers[0].extras).toBeUndefined();
29 | expect(
30 | gltfWithExtrasRemoved.extensions.KHR_techniques_webgl.shaders[0].extras,
31 | ).toBeUndefined();
32 | });
33 | });
34 |
--------------------------------------------------------------------------------
/specs/lib/updateAccessorComponentTypeSpec.js:
--------------------------------------------------------------------------------
1 | "use strict";
2 | const Cesium = require("cesium");
3 | const readResources = require("../../lib/readResources");
4 | const updateAccessorComponentTypes = require("../../lib/updateAccessorComponentTypes");
5 |
6 | const WebGLConstants = Cesium.WebGLConstants;
7 |
8 | let buffer;
9 |
10 | describe("updateAccessorComponentTypes", () => {
11 | beforeAll(() => {
12 | // Note: TypedArray constructors initialize all elements to zero
13 | const byteBuffer = Buffer.from(new Int8Array(96).buffer);
14 | const floatBuffer = Buffer.from(new Float32Array(96).buffer);
15 | const unsignedShortBuffer = Buffer.from(new Uint16Array(96).buffer);
16 | const source = Buffer.concat([
17 | byteBuffer,
18 | floatBuffer,
19 | unsignedShortBuffer,
20 | ]);
21 | const byteLength = source.length;
22 | const dataUri = `data:application/octet-stream;base64,${source.toString(
23 | "base64",
24 | )}`;
25 | buffer = {
26 | uri: dataUri,
27 | byteLength: byteLength,
28 | };
29 | });
30 |
31 | it("converts joints accessor types", async () => {
32 | const gltf = {
33 | meshes: [
34 | {
35 | primitives: [
36 | {
37 | attributes: {
38 | JOINTS_0: 0,
39 | },
40 | },
41 | {
42 | attributes: {
43 | JOINTS_0: 1,
44 | },
45 | },
46 | {
47 | attributes: {
48 | JOINTS_0: 2,
49 | },
50 | },
51 | ],
52 | },
53 | ],
54 | accessors: [
55 | {
56 | bufferView: 0,
57 | componentType: WebGLConstants.BYTE,
58 | count: 24,
59 | type: "VEC4",
60 | },
61 | {
62 | bufferView: 1,
63 | componentType: WebGLConstants.FLOAT,
64 | byteOffset: 12,
65 | count: 24,
66 | type: "VEC4",
67 | },
68 | {
69 | bufferView: 2,
70 | componentType: WebGLConstants.UNSIGNED_SHORT,
71 | count: 24,
72 | type: "VEC4",
73 | },
74 | ],
75 | bufferViews: [
76 | {
77 | buffer: 0,
78 | byteOffset: 0,
79 | byteLength: 96,
80 | },
81 | {
82 | buffer: 0,
83 | byteOffset: 96,
84 | byteLength: 384,
85 | },
86 | {
87 | buffer: 0,
88 | byteOffset: 480,
89 | byteLength: 192,
90 | },
91 | ],
92 | buffers: [buffer],
93 | };
94 |
95 | await readResources(gltf);
96 | updateAccessorComponentTypes(gltf);
97 |
98 | expect(gltf.accessors.length).toBe(3);
99 | expect(gltf.bufferViews.length).toBe(5);
100 | expect(gltf.buffers.length).toBe(3);
101 |
102 | expect(gltf.accessors[0].componentType).toBe(WebGLConstants.UNSIGNED_BYTE);
103 | expect(gltf.accessors[0].bufferView).toBe(3);
104 | expect(gltf.bufferViews[3].buffer).toBe(1);
105 | expect(gltf.bufferViews[3].byteLength).toBe(96);
106 |
107 | expect(gltf.accessors[1].componentType).toBe(WebGLConstants.UNSIGNED_SHORT);
108 | expect(gltf.accessors[1].bufferView).toBe(4);
109 | expect(gltf.accessors[1].byteOffset).toBe(0);
110 | expect(gltf.bufferViews[4].buffer).toBe(2);
111 | expect(gltf.bufferViews[4].byteLength).toBe(192);
112 |
113 | expect(gltf.accessors[2].componentType).toBe(WebGLConstants.UNSIGNED_SHORT);
114 | expect(gltf.accessors[2].bufferView).toBe(2);
115 | });
116 |
117 | it("converts weights accessor types", async () => {
118 | const gltf = {
119 | meshes: [
120 | {
121 | primitives: [
122 | {
123 | attributes: {
124 | WEIGHTS_0: 0,
125 | },
126 | },
127 | {
128 | attributes: {
129 | WEIGHTS_0: 1,
130 | },
131 | },
132 | {
133 | attributes: {
134 | WEIGHTS_0: 2,
135 | },
136 | },
137 | ],
138 | },
139 | ],
140 | accessors: [
141 | {
142 | bufferView: 0,
143 | componentType: WebGLConstants.FLOAT,
144 | count: 24,
145 | type: "VEC4",
146 | },
147 | {
148 | bufferView: 1,
149 | componentType: WebGLConstants.BYTE,
150 | byteOffset: 12,
151 | count: 24,
152 | type: "VEC4",
153 | },
154 | {
155 | bufferView: 2,
156 | componentType: WebGLConstants.SHORT,
157 | count: 24,
158 | type: "VEC4",
159 | },
160 | ],
161 | bufferViews: [
162 | {
163 | buffer: 0,
164 | byteLength: 12,
165 | },
166 | {
167 | buffer: 0,
168 | byteOffset: 12,
169 | byteLength: 12,
170 | },
171 | {
172 | buffer: 0,
173 | byteLength: 12,
174 | },
175 | ],
176 | buffers: [buffer],
177 | };
178 |
179 | await readResources(gltf);
180 | updateAccessorComponentTypes(gltf);
181 |
182 | expect(gltf.accessors.length).toBe(3);
183 | expect(gltf.bufferViews.length).toBe(5);
184 | expect(gltf.buffers.length).toBe(3);
185 |
186 | expect(gltf.accessors[0].componentType).toBe(WebGLConstants.FLOAT);
187 | expect(gltf.accessors[0].bufferView).toBe(0);
188 |
189 | expect(gltf.accessors[1].componentType).toBe(WebGLConstants.UNSIGNED_BYTE);
190 | expect(gltf.accessors[1].bufferView).toBe(3);
191 | expect(gltf.accessors[1].byteOffset).toBe(0);
192 | expect(gltf.bufferViews[3].buffer).toBe(1);
193 | expect(gltf.bufferViews[3].byteLength).toBe(96);
194 |
195 | expect(gltf.accessors[2].componentType).toBe(WebGLConstants.UNSIGNED_SHORT);
196 | expect(gltf.accessors[2].bufferView).toBe(4);
197 | expect(gltf.bufferViews[4].buffer).toBe(2);
198 | expect(gltf.bufferViews[4].byteLength).toBe(192);
199 | });
200 | });
201 |
--------------------------------------------------------------------------------
/specs/lib/usesExtensionSpec.js:
--------------------------------------------------------------------------------
1 | "use strict";
2 | const usesExtension = require("../../lib/usesExtension");
3 |
4 | describe("usesExtension", () => {
5 | it("uses extension", () => {
6 | const gltf = {
7 | extensionsUsed: ["extension1", "extension2"],
8 | };
9 | expect(usesExtension(gltf, "extension1")).toBe(true);
10 | expect(usesExtension(gltf, "extension2")).toBe(true);
11 | expect(usesExtension(gltf, "extension3")).toBe(false);
12 |
13 | const emptyGltf = {};
14 | expect(usesExtension(emptyGltf, "extension1")).toBe(false);
15 | });
16 | });
17 |
--------------------------------------------------------------------------------