From eb4ec2036bf267d064dd68be1afcff7eff55046b Mon Sep 17 00:00:00 2001
From: Marco Hutter
Date: Tue, 17 Sep 2024 17:01:41 +0200
Subject: [PATCH 01/38] First, partial support of GPM classes
---
packages/engine/Source/Scene/GltfGpmLoader.js | 469 ++++++++++++++++++
.../Model/Extensions/Gpm/AnchorPointDirect.js | 50 ++
.../Extensions/Gpm/AnchorPointIndirect.js | 66 +++
.../Model/Extensions/Gpm/CorrelationGroup.js | 67 +++
.../Model/Extensions/Gpm/GltfGpmLocal.js | 145 ++++++
.../Extensions/Gpm/MeshPrimitiveGpmLocal.js | 27 +
.../Scene/Model/Extensions/Gpm/PpeMetadata.js | 67 +++
.../Scene/Model/Extensions/Gpm/PpeSource.js | 72 +++
.../Scene/Model/Extensions/Gpm/PpeTexture.js | 115 +++++
.../Scene/Model/Extensions/Gpm/Spdcf.js | 87 ++++
.../Scene/Model/Extensions/Gpm/StorageType.js | 27 +
11 files changed, 1192 insertions(+)
create mode 100644 packages/engine/Source/Scene/GltfGpmLoader.js
create mode 100644 packages/engine/Source/Scene/Model/Extensions/Gpm/AnchorPointDirect.js
create mode 100644 packages/engine/Source/Scene/Model/Extensions/Gpm/AnchorPointIndirect.js
create mode 100644 packages/engine/Source/Scene/Model/Extensions/Gpm/CorrelationGroup.js
create mode 100644 packages/engine/Source/Scene/Model/Extensions/Gpm/GltfGpmLocal.js
create mode 100644 packages/engine/Source/Scene/Model/Extensions/Gpm/MeshPrimitiveGpmLocal.js
create mode 100644 packages/engine/Source/Scene/Model/Extensions/Gpm/PpeMetadata.js
create mode 100644 packages/engine/Source/Scene/Model/Extensions/Gpm/PpeSource.js
create mode 100644 packages/engine/Source/Scene/Model/Extensions/Gpm/PpeTexture.js
create mode 100644 packages/engine/Source/Scene/Model/Extensions/Gpm/Spdcf.js
create mode 100644 packages/engine/Source/Scene/Model/Extensions/Gpm/StorageType.js
diff --git a/packages/engine/Source/Scene/GltfGpmLoader.js b/packages/engine/Source/Scene/GltfGpmLoader.js
new file mode 100644
index 000000000000..b3463abb8054
--- /dev/null
+++ b/packages/engine/Source/Scene/GltfGpmLoader.js
@@ -0,0 +1,469 @@
+import Check from "../Core/Check.js";
+import defaultValue from "../Core/defaultValue.js";
+import defined from "../Core/defined.js";
+import ResourceCache from "./ResourceCache.js";
+import ResourceLoader from "./ResourceLoader.js";
+import ResourceLoaderState from "./ResourceLoaderState.js";
+import PropertyTexture from "./PropertyTexture.js";
+import StructuralMetadata from "./StructuralMetadata.js";
+import MetadataSchema from "./MetadataSchema.js";
+
+/**
+ * Loads glTF NGA_gpm_local
+ *
+ * Implements the {@link ResourceLoader} interface.
+ *
+ * Implementation note: This is an experimental implementation. It is based
+ * on a GltfStructuralMetadataLoader, by removing everything that is not
+ * related to property textures, and translating the "ppeTextures" of
+ * the NGA_gpm_local extension into property textures. These will be
+ * returned as part of a `StructuralMetadata` object, which may override
+ * any `StructuralMetadata` that was read directly from the glTF.
+ *
+ * @alias GltfGpmLoader
+ * @constructor
+ * @augments ResourceLoader
+ *
+ * @param {object} options Object with the following properties:
+ * @param {object} options.gltf The glTF JSON.
+ * @param {string} [options.extension] The NGA_gpm_local
extension object.
+ * @param {Resource} options.gltfResource The {@link Resource} containing the glTF.
+ * @param {Resource} options.baseResource The {@link Resource} that paths in the glTF JSON are relative to.
+ * @param {SupportedImageFormats} options.supportedImageFormats The supported image formats.
+ * @param {FrameState} options.frameState The frame state.
+ * @param {string} [options.cacheKey] The cache key of the resource.
+ * @param {boolean} [options.asynchronous=true] Determines if WebGL resource creation will be spread out over several frames or block until all WebGL resources are created.
+ *
+ * @private
+ * @experimental This feature is subject to change without Cesium's standard deprecation policy.
+ */
+function GltfGpmLoader(options) {
+ options = defaultValue(options, defaultValue.EMPTY_OBJECT);
+ const gltf = options.gltf;
+ const extension = options.extension;
+ const gltfResource = options.gltfResource;
+ const baseResource = options.baseResource;
+ const supportedImageFormats = options.supportedImageFormats;
+ const frameState = options.frameState;
+ const cacheKey = options.cacheKey;
+ const asynchronous = defaultValue(options.asynchronous, true);
+
+ //>>includeStart('debug', pragmas.debug);
+ Check.typeOf.object("options.gltf", gltf);
+ Check.typeOf.object("options.extension", extension);
+ Check.typeOf.object("options.gltfResource", gltfResource);
+ Check.typeOf.object("options.baseResource", baseResource);
+ Check.typeOf.object("options.supportedImageFormats", supportedImageFormats);
+ Check.typeOf.object("options.frameState", frameState);
+ //>>includeEnd('debug');
+
+ this._gltfResource = gltfResource;
+ this._baseResource = baseResource;
+ this._gltf = gltf;
+ this._extension = extension;
+ this._supportedImageFormats = supportedImageFormats;
+ this._frameState = frameState;
+ this._cacheKey = cacheKey;
+ this._asynchronous = asynchronous;
+ this._textureLoaders = [];
+ this._textureIds = [];
+ this._structuralMetadata = undefined;
+ this._state = ResourceLoaderState.UNLOADED;
+ this._promise = undefined;
+}
+
+if (defined(Object.create)) {
+ GltfGpmLoader.prototype = Object.create(ResourceLoader.prototype);
+ GltfGpmLoader.prototype.constructor = GltfGpmLoader;
+}
+
+Object.defineProperties(GltfGpmLoader.prototype, {
+ /**
+ * The cache key of the resource.
+ *
+ * @memberof GltfGpmLoader.prototype
+ *
+ * @type {string}
+ * @readonly
+ * @private
+ */
+ cacheKey: {
+ get: function () {
+ return this._cacheKey;
+ },
+ },
+ /**
+ * The parsed structural metadata
+ *
+ * @memberof GltfGpmLoader.prototype
+ *
+ * @type {StructuralMetadata}
+ * @readonly
+ * @private
+ */
+ structuralMetadata: {
+ get: function () {
+ return this._structuralMetadata;
+ },
+ },
+});
+
+async function loadResources(loader) {
+ try {
+ const texturesPromise = loadTextures(loader);
+ await texturesPromise;
+
+ if (loader.isDestroyed()) {
+ return;
+ }
+
+ loader._gltf = undefined; // No longer need to hold onto the glTF
+
+ loader._state = ResourceLoaderState.LOADED;
+ return loader;
+ } catch (error) {
+ if (loader.isDestroyed()) {
+ return;
+ }
+
+ loader.unload();
+ loader._state = ResourceLoaderState.FAILED;
+ const errorMessage = "Failed to load GPM data";
+ throw loader.getError(errorMessage, error);
+ }
+}
+
+/**
+ * Loads the resource.
+ * @returns {Promise} A promise which resolves to the loader when the resource loading is completed.
+ * @private
+ */
+GltfGpmLoader.prototype.load = function () {
+ if (defined(this._promise)) {
+ return this._promise;
+ }
+
+ this._state = ResourceLoaderState.LOADING;
+ this._promise = loadResources(this);
+ return this._promise;
+};
+
+function gatherUsedTextureIds(gpmExtension) {
+ // Gather the used textures
+ const textureIds = {};
+ const ppeTextures = gpmExtension.ppeTextures;
+ if (defined(ppeTextures)) {
+ for (let i = 0; i < ppeTextures.length; i++) {
+ const ppeTexture = ppeTextures[i];
+ // The texture is a valid textureInfo.
+ textureIds[ppeTexture.index] = ppeTexture;
+ }
+ }
+ return textureIds;
+}
+
+function loadTextures(gpmLoader) {
+ let textureIds;
+ if (defined(gpmLoader._extension)) {
+ textureIds = gatherUsedTextureIds(gpmLoader._extension);
+ }
+
+ const gltf = gpmLoader._gltf;
+ const gltfResource = gpmLoader._gltfResource;
+ const baseResource = gpmLoader._baseResource;
+ const supportedImageFormats = gpmLoader._supportedImageFormats;
+ const frameState = gpmLoader._frameState;
+ const asynchronous = gpmLoader._asynchronous;
+
+ // Load the textures
+ const texturePromises = [];
+ for (const textureId in textureIds) {
+ if (textureIds.hasOwnProperty(textureId)) {
+ const textureLoader = ResourceCache.getTextureLoader({
+ gltf: gltf,
+ textureInfo: textureIds[textureId],
+ gltfResource: gltfResource,
+ baseResource: baseResource,
+ supportedImageFormats: supportedImageFormats,
+ frameState: frameState,
+ asynchronous: asynchronous,
+ });
+ gpmLoader._textureLoaders.push(textureLoader);
+ gpmLoader._textureIds.push(textureId);
+ texturePromises.push(textureLoader.load());
+ }
+ }
+
+ return Promise.all(texturePromises);
+}
+
+/**
+ * A static mapping from PPE texture property identifier keys
+ * to `MetadataSchema` instances. This is used to create each
+ * schema (with a certain structure) only ONCE in
+ * obtainPpeTexturesMetadataSchema
+ *
+ * @private
+ */
+GltfGpmLoader.ppeTexturesMetadataSchemaCache = new Map();
+
+/**
+ * Create the JSON description of a metadata class that treats
+ * the given PPE texture as a property texture property(!).
+ *
+ * @param {any} ppeTexture - The PPE texture
+ * @param {number} index - The index of the texture in the extension
+ * @returns The class JSON
+ */
+GltfGpmLoader.prototype.createPpeTextureClassJson = function (
+ ppeTexture,
+ index
+) {
+ const traits = ppeTexture.traits;
+ const ppePropertyName = traits.source;
+
+ // XXX_UNCERTAINTY: The ppeTexture will have a structure
+ // like this:
+ //
+ //"ppeTextures" : [
+ // {
+ // "traits" : {
+ // "source" : "SIGZ",
+ // "min" : 0.0,
+ // "max" : 16.0
+ // },
+ // "index" : 2,
+ // "noData" : 255,
+ // "offset" : 0.0,
+ // "scale" : 0.06274509803921569,
+ // "texCoord" : 1
+ // },
+ //
+ // Note that in GPM 1.2d, the min/max should be part of the
+ // texture, but in the actual data (GPM 1.2i), they are in
+ // the traits (ppeMetadata). And there, they/ seem to denote
+ // the min/max values that actually appear in the texture.
+ // So they are integrated into the 'scale' factor here:
+ const min = traits.min ?? 0;
+ const max = traits.max ?? 255;
+ const minMaxScale = 255.0 / (max - min);
+ const offset = ppeTexture.offset;
+ const scale = ppeTexture.scale * minMaxScale;
+ const classJson = {
+ name: `PPE texture class ${index}`,
+ properties: {
+ [ppePropertyName]: {
+ name: "PPE",
+ type: "SCALAR",
+ componentType: "UINT8",
+ normalized: true,
+ offset: offset,
+ scale: scale,
+ },
+ },
+ };
+ return classJson;
+};
+
+/**
+ * Returns the `MetadataSchema` for the PPE textures in this instance.
+ *
+ * This method will return a (statically/globally) cached metadata
+ * schema that reflects the structure of the PPE textures in this
+ * instance, creating and caching it if necessary.
+ *
+ * For details on the cache key, see `collectPpeTexturePropertyIdentifiers`
+ *
+ * @returns The `MetadataSchema`
+ */
+GltfGpmLoader.prototype.obtainPpeTexturesMetadataSchema = function () {
+ const ppeTexturePropertyIdentifiers = this.collectPpeTexturePropertyIdentifiers();
+ const key = ppeTexturePropertyIdentifiers.toString();
+ let ppeTexturesMetadataSchema = GltfGpmLoader.ppeTexturesMetadataSchemaCache.get(
+ key
+ );
+ if (defined(ppeTexturesMetadataSchema)) {
+ // XXX_UNCERTAINTY Debug log
+ //console.log(`Using cached schema for GPM PPE textures with key ${key}`);
+ return ppeTexturesMetadataSchema;
+ }
+
+ // XXX_UNCERTAINTY Debug log - if caching works, this should be printed only ONCE!
+ console.log(`Creating schema for GPM PPE textures with key ${key}`);
+ const schemaId = `PPE_TEXTURE_SCHEMA_${GltfGpmLoader.ppeTexturesMetadataSchemaCache.size}`;
+ const ppeTexturesMetadataSchemaJson = {
+ id: schemaId,
+ classes: {},
+ };
+
+ const extension = this._extension;
+ if (defined(extension.ppeTextures)) {
+ for (let i = 0; i < extension.ppeTextures.length; i++) {
+ const ppeTexture = extension.ppeTextures[i];
+ const classId = `ppeTexture_${i}`;
+ const classJson = this.createPpeTextureClassJson(ppeTexture, i);
+ ppeTexturesMetadataSchemaJson.classes[classId] = classJson;
+ }
+ }
+
+ ppeTexturesMetadataSchema = MetadataSchema.fromJson(
+ ppeTexturesMetadataSchemaJson
+ );
+ GltfGpmLoader.ppeTexturesMetadataSchemaCache.set(
+ key,
+ ppeTexturesMetadataSchema
+ );
+ return ppeTexturesMetadataSchema;
+};
+
+/**
+ * Creates an array of strings that serve as identifiers for PPE textures.
+ *
+ * Each glTF may define multiple `ppeTexture` objects within the
+ * `NGA_gpm_local` extensions. Each of these textures corresponds
+ * to one 'property texture property(!)' in a metadata schema.
+ *
+ * This method will create an array where each element is a (JSON)
+ * string representation of the parts of a GPM PPE texture definition
+ * that are relevant for distinguishing two PPE textures in terms
+ * of their structure within a `StructuralMetadata`.
+ *
+ * @returns The identifiers
+ */
+GltfGpmLoader.prototype.collectPpeTexturePropertyIdentifiers = function () {
+ const extension = this._extension;
+ const ppeTexturePropertyIdentifiers = [];
+ if (defined(extension.ppeTextures)) {
+ for (let i = 0; i < extension.ppeTextures.length; i++) {
+ const ppeTexture = extension.ppeTextures[i];
+ // The following will create an identifier that can be used
+ // to define two PPE textures as "representing the same
+ // property texture property" within a structural metadata
+ // schema.
+ const classJson = this.createPpeTextureClassJson(ppeTexture, i);
+ const ppeTexturePropertyIdentifier = JSON.stringify(classJson);
+ ppeTexturePropertyIdentifiers.push(ppeTexturePropertyIdentifier);
+ }
+ }
+ return ppeTexturePropertyIdentifiers;
+};
+
+/**
+ * Processes the resource until it becomes ready.
+ *
+ * @param {FrameState} frameState The frame state.
+ * @private
+ */
+GltfGpmLoader.prototype.process = function (frameState) {
+ //>>includeStart('debug', pragmas.debug);
+ Check.typeOf.object("frameState", frameState);
+ //>>includeEnd('debug');
+
+ if (this._state === ResourceLoaderState.READY) {
+ return true;
+ }
+
+ if (this._state !== ResourceLoaderState.LOADED) {
+ return false;
+ }
+
+ const textureLoaders = this._textureLoaders;
+ const textureLoadersLength = textureLoaders.length;
+ let ready = true;
+ for (let i = 0; i < textureLoadersLength; ++i) {
+ const textureLoader = textureLoaders[i];
+ const textureReady = textureLoader.process(frameState);
+ ready = ready && textureReady;
+ }
+
+ if (!ready) {
+ return false;
+ }
+
+ const textures = {};
+ for (let i = 0; i < this._textureIds.length; ++i) {
+ const textureId = this._textureIds[i];
+ const textureLoader = textureLoaders[i];
+ if (!textureLoader.isDestroyed()) {
+ textures[textureId] = textureLoader.texture;
+ }
+ }
+
+ const ppeTexturesMetadataSchema = this.obtainPpeTexturesMetadataSchema();
+ const extension = this._extension;
+ const propertyTextures = [];
+ if (defined(extension.ppeTextures)) {
+ for (let i = 0; i < extension.ppeTextures.length; i++) {
+ const ppeTexture = extension.ppeTextures[i];
+ const classId = `ppeTexture_${i}`;
+ const traits = ppeTexture.traits;
+ const ppePropertyName = traits.source;
+ const metadataClass = ppeTexturesMetadataSchema.classes[classId];
+
+ // XXX_UNCERTAINTY Debug log
+ //console.log(
+ // `Creating property texture with class ${classId} and property ${ppePropertyName}`
+ //);
+
+ const ppeTextureAsPropertyTexture = {
+ class: classId,
+ properties: {
+ [ppePropertyName]: {
+ index: ppeTexture.index,
+ texCoord: ppeTexture.texCoord,
+ },
+ },
+ };
+ propertyTextures.push(
+ new PropertyTexture({
+ id: i,
+ name: ppeTexture.name,
+ propertyTexture: ppeTextureAsPropertyTexture,
+ class: metadataClass,
+ textures: textures,
+ })
+ );
+ }
+ }
+
+ const structuralMetadata = new StructuralMetadata({
+ schema: ppeTexturesMetadataSchema,
+ propertyTables: [],
+ propertyTextures: propertyTextures,
+ propertyAttributes: [],
+ statistics: extension.statistics,
+ extras: extension.extras,
+ extensions: extension.extensions,
+ });
+ this._structuralMetadata = structuralMetadata;
+
+ this._state = ResourceLoaderState.READY;
+ return true;
+};
+
+function unloadTextures(gpmLoader) {
+ const textureLoaders = gpmLoader._textureLoaders;
+ const textureLoadersLength = textureLoaders.length;
+ for (let i = 0; i < textureLoadersLength; ++i) {
+ ResourceCache.unload(textureLoaders[i]);
+ }
+ gpmLoader._textureLoaders.length = 0;
+ gpmLoader._textureIds.length = 0;
+}
+
+/**
+ * Unloads the resource.
+ * @private
+ */
+GltfGpmLoader.prototype.unload = function () {
+ unloadTextures(this);
+
+ if (defined(this._schemaLoader)) {
+ ResourceCache.unload(this._schemaLoader);
+ }
+ this._schemaLoader = undefined;
+
+ this._structuralMetadata = undefined;
+};
+
+export default GltfGpmLoader;
diff --git a/packages/engine/Source/Scene/Model/Extensions/Gpm/AnchorPointDirect.js b/packages/engine/Source/Scene/Model/Extensions/Gpm/AnchorPointDirect.js
new file mode 100644
index 000000000000..79e71895214a
--- /dev/null
+++ b/packages/engine/Source/Scene/Model/Extensions/Gpm/AnchorPointDirect.js
@@ -0,0 +1,50 @@
+import Check from "../Core/Check.js";
+
+/**
+ * Metadata for one stored anchor point using direct storage
+ *
+ * @private
+ * @experimental This feature is not final and is subject to change without Cesium's standard deprecation policy.
+ */
+function AnchorPointDirect(options) {
+ //>>includeStart('debug', pragmas.debug);
+ Check.typeOf.object("options.position", options.property);
+ Check.typeOf.object("options.adjustmentParams", options.adjustmentParams);
+ //>>includeEnd('debug');
+
+ this._position = options.position;
+ this._adjustmentParams = options.adjustmentParams;
+}
+
+Object.defineProperties(AnchorPointDirect.prototype, {
+ /**
+ * Anchor point geographic coordinates in meters as X/Easting, Y/Northing, Z/HAE
+ *
+ * @memberof AnchorPointDirect.prototype
+ * @type {Cartesian3}
+ * @readonly
+ * @private
+ */
+ position: {
+ get: function () {
+ return this._position;
+ },
+ },
+
+ /**
+ * The delta-x delta-y delta-z adjustment values in meters per anchor
+ * point.
+ *
+ * @memberof AnchorPointDirect.prototype
+ * @type {Cartesian3}
+ * @readonly
+ * @private
+ */
+ adjustmentParams: {
+ get: function () {
+ return this._adjustmentParams;
+ },
+ },
+});
+
+export default AnchorPointDirect;
diff --git a/packages/engine/Source/Scene/Model/Extensions/Gpm/AnchorPointIndirect.js b/packages/engine/Source/Scene/Model/Extensions/Gpm/AnchorPointIndirect.js
new file mode 100644
index 000000000000..c70996f33485
--- /dev/null
+++ b/packages/engine/Source/Scene/Model/Extensions/Gpm/AnchorPointIndirect.js
@@ -0,0 +1,66 @@
+import Check from "../Core/Check.js";
+
+/**
+ * Metadata for one stored anchor point.
+ *
+ * @private
+ * @experimental This feature is not final and is subject to change without Cesium's standard deprecation policy.
+ */
+function AnchorPointIndirect(options) {
+ //>>includeStart('debug', pragmas.debug);
+ Check.typeOf.object("options.position", options.property);
+ Check.typeOf.object("options.adjustmentParams", options.adjustmentParams);
+ Check.typeOf.object("options.covarianceMatrix", options.covarianceMatrix);
+ //>>includeEnd('debug');
+
+ this._position = options.position;
+ this._adjustmentParams = options.adjustmentParams;
+ this._covarianceMatrix = options.covarianceMatrix;
+}
+
+Object.defineProperties(AnchorPointIndirect.prototype, {
+ /**
+ * Anchor point geographic coordinates in meters as X/Easting, Y/Northing, Z/HAE
+ *
+ * @memberof AnchorPointIndirect.prototype
+ * @type {Cartesian3}
+ * @readonly
+ * @private
+ */
+ position: {
+ get: function () {
+ return this._position;
+ },
+ },
+
+ /**
+ * The delta-x delta-y delta-z adjustment values in meters per anchor
+ * point.
+ *
+ * @memberof AnchorPointIndirect.prototype
+ * @type {Cartesian3}
+ * @readonly
+ * @private
+ */
+ adjustmentParams: {
+ get: function () {
+ return this._adjustmentParams;
+ },
+ },
+
+ /**
+ * The 3x3 covariance matrix.
+ *
+ * @memberof AnchorPointIndirect.prototype
+ * @type {Matrix3}
+ * @readonly
+ * @private
+ */
+ covarianceMatrix: {
+ get: function () {
+ return this._covarianceMatrix;
+ },
+ },
+});
+
+export default AnchorPointIndirect;
diff --git a/packages/engine/Source/Scene/Model/Extensions/Gpm/CorrelationGroup.js b/packages/engine/Source/Scene/Model/Extensions/Gpm/CorrelationGroup.js
new file mode 100644
index 000000000000..e535ea6a43a2
--- /dev/null
+++ b/packages/engine/Source/Scene/Model/Extensions/Gpm/CorrelationGroup.js
@@ -0,0 +1,67 @@
+import Check from "../Core/Check.js";
+
+/**
+ * Metadata identifying parameters using same correlation modeling and
+ * associated correlation parameters.
+ *
+ * @private
+ * @experimental This feature is not final and is subject to change without Cesium's standard deprecation policy.
+ */
+function CorrelationGroup(options) {
+ //>>includeStart('debug', pragmas.debug);
+ Check.typeOf.object("options.groupFlags", options.groupFlags);
+ Check.typeOf.object("options.rotationThetas", options.rotationThetas);
+ Check.typeOf.object("options.params", options.params);
+ //>>includeEnd('debug');
+
+ this._groupFlags = options.groupFlags;
+ this._rotationThetas = options.rotationThetas;
+ this._params = options.params;
+}
+
+Object.defineProperties(CorrelationGroup.prototype, {
+ /**
+ * Array of 3 booleans indicating if parameters delta-x delta-y delta-z
+ * used in the correlation group
+ *
+ * @memberof CorrelationGroup.prototype
+ * @type {boolean[]}
+ * @readonly
+ * @private
+ */
+ groupFlags: {
+ get: function () {
+ return this._groupFlags;
+ },
+ },
+
+ /**
+ * Rotations in milliradians about X, Y, Z axes, respectively
+ *
+ * @memberof CorrelationGroup.prototype
+ * @type {Cartesian3}
+ * @readonly
+ * @private
+ */
+ rotationThetas: {
+ get: function () {
+ return this._rotationThetas;
+ },
+ },
+
+ /**
+ * Array of 3 sets of SPDCF parameters, for the U, V, W directions, respectively
+ *
+ * @memberof CorrelationGroup.prototype
+ * @type {Spdcf[]}
+ * @readonly
+ * @private
+ */
+ params: {
+ get: function () {
+ return this._params;
+ },
+ },
+});
+
+export default CorrelationGroup;
diff --git a/packages/engine/Source/Scene/Model/Extensions/Gpm/GltfGpmLocal.js b/packages/engine/Source/Scene/Model/Extensions/Gpm/GltfGpmLocal.js
new file mode 100644
index 000000000000..016c27bf81cb
--- /dev/null
+++ b/packages/engine/Source/Scene/Model/Extensions/Gpm/GltfGpmLocal.js
@@ -0,0 +1,145 @@
+import defined from "../Core/defined.js";
+import Check from "../Core/Check.js";
+import StorageType from "./StorageType.js";
+import RuntimeError from "../Core/RuntimeError.js";
+
+/**
+ * The GPM metadata for a Ground-Space Indirect implementation stored
+ * locally (i.e. a tile and/or leaf node).
+ *
+ * @private
+ * @experimental This feature is not final and is subject to change without Cesium's standard deprecation policy.
+ */
+function GltfGpmLocal(options) {
+ //>>includeStart('debug', pragmas.debug);
+ Check.typeOf.object("options.storageType", options.storageType);
+ //>>includeEnd('debug');
+
+ this._storageType = options.storageType;
+ this._anchorPointsIndirect = options.anchorPointsIndirect;
+ this._anchorPointsDirect = options.anchorPointsDirect;
+ this._intraTileCorrelationGroups = options.intraTileCorrelationGroups;
+ this._covarianceDirect = options.covarianceDirect;
+
+ //>>includeStart('debug', pragmas.debug);
+ if (this.storageType === StorageType.Indirect) {
+ if (!defined(this.anchorPointsIndirect)) {
+ throw new RuntimeError(
+ "The anchorPointsIndirect are required for 'Indirect' storage"
+ );
+ }
+ if (!defined(this.intraTileCorrelationGroups)) {
+ throw new RuntimeError(
+ "The intraTileCorrelationGroups are required for 'Indirect' storage"
+ );
+ }
+ if (defined(this.anchorPointsDirect)) {
+ throw new RuntimeError(
+ "The anchorPointsDirect must be omitted for 'Indirect' storage"
+ );
+ }
+ if (defined(this.covarianceDirect)) {
+ throw new RuntimeError(
+ "The covarianceDirect must be omitted for 'Indirect' storage"
+ );
+ }
+ } else {
+ // Direct storage
+ if (!defined(this.anchorPointsDirect)) {
+ throw new RuntimeError(
+ "The anchorPointsDirect are required for 'Direct' storage"
+ );
+ }
+ if (!defined(this.covarianceDirect)) {
+ throw new RuntimeError(
+ "The covarianceDirect is required for 'Direct' storage"
+ );
+ }
+ if (defined(this.anchorPointsIndirect)) {
+ throw new RuntimeError(
+ "The anchorPointsIndirect must be omitted for 'Direct' storage"
+ );
+ }
+ if (defined(this.intraTileCorrelationGroups)) {
+ throw new RuntimeError(
+ "The intraTileCorrelationGroups must be omitted for 'Direct' storage"
+ );
+ }
+ }
+ //>>includeEnd('debug');
+}
+
+Object.defineProperties(GltfGpmLocal.prototype, {
+ /**
+ * Specifies if covariance storage is indirect or direct.
+ *
+ * @memberof GltfGpmLocal.prototype
+ * @type {StorageType}
+ * @readonly
+ * @private
+ */
+ storageType: {
+ get: function () {
+ return this._storageType;
+ },
+ },
+
+ /**
+ * Array of stored indirect anchor points
+ *
+ * @memberof GltfGpmLocal.prototype
+ * @type {AnchorPointIndirect[]}
+ * @readonly
+ * @private
+ */
+ anchorPointsIndirect: {
+ get: function () {
+ return this._anchorPointsIndirect;
+ },
+ },
+
+ /**
+ * Array of stored direct anchor points
+ *
+ * @memberof GltfGpmLocal.prototype
+ * @type {AnchorPointDirect[]}
+ * @readonly
+ * @private
+ */
+ anchorPointsDirect: {
+ get: function () {
+ return this._anchorPointsDirect;
+ },
+ },
+
+ /**
+ * Metadata identifying parameters using same correlation modeling and
+ * associated correlation parameters
+ *
+ * @memberof GltfGpmLocal.prototype
+ * @type {CorrelationGroup[]}
+ * @readonly
+ * @private
+ */
+ intraTileCorrelationGroups: {
+ get: function () {
+ return this._intraTileCorrelationGroups;
+ },
+ },
+
+ /**
+ * The full covariance of anchor point parameters
+ *
+ * @memberof GltfGpmLocal.prototype
+ * @type {Matrix3}
+ * @readonly
+ * @private
+ */
+ covarianceDirect: {
+ get: function () {
+ return this._covarianceDirect;
+ },
+ },
+});
+
+export default GltfGpmLocal;
diff --git a/packages/engine/Source/Scene/Model/Extensions/Gpm/MeshPrimitiveGpmLocal.js b/packages/engine/Source/Scene/Model/Extensions/Gpm/MeshPrimitiveGpmLocal.js
new file mode 100644
index 000000000000..1ca122137744
--- /dev/null
+++ b/packages/engine/Source/Scene/Model/Extensions/Gpm/MeshPrimitiveGpmLocal.js
@@ -0,0 +1,27 @@
+/**
+ * Local Generic Point-cloud Model information about a glTF primitive.
+ *
+ * @private
+ * @experimental This feature is not final and is subject to change without Cesium's standard deprecation policy.
+ */
+function MeshPrimitiveGpmLocal(options) {
+ this._ppeTextures = options.ppeTextures;
+}
+
+Object.defineProperties(MeshPrimitiveGpmLocal.prototype, {
+ /**
+ * An array of ppe textures.
+ *
+ * @memberof MeshPrimitiveGpmLocal.prototype
+ * @type {PpeTexture[]|undefined}
+ * @readonly
+ * @private
+ */
+ ppeTextures: {
+ get: function () {
+ return this._ppeTextures;
+ },
+ },
+});
+
+export default MeshPrimitiveGpmLocal;
diff --git a/packages/engine/Source/Scene/Model/Extensions/Gpm/PpeMetadata.js b/packages/engine/Source/Scene/Model/Extensions/Gpm/PpeMetadata.js
new file mode 100644
index 000000000000..26f0d152284a
--- /dev/null
+++ b/packages/engine/Source/Scene/Model/Extensions/Gpm/PpeMetadata.js
@@ -0,0 +1,67 @@
+import Check from "../Core/Check.js";
+
+/**
+ * Metadata related to the stored PPE data.
+ *
+ * @private
+ * @experimental This feature is not final and is subject to change without Cesium's standard deprecation policy.
+ */
+function PpeMetadata(options) {
+ //>>includeStart('debug', pragmas.debug);
+ Check.typeOf.object("options.source", options.source);
+ //>>includeEnd('debug');
+
+ this._min = options.min;
+ this._max = options.max;
+ this._source = options.source;
+}
+
+Object.defineProperties(PpeMetadata.prototype, {
+ /**
+ * Minimum allowed value for the property. This is the minimum of all
+ * values after the transforms based on the offset and scale properties
+ * have been applied.
+ *
+ * @memberof PpeMetadata.prototype
+ * @type {number|undefined}
+ * @readonly
+ * @private
+ */
+ min: {
+ get: function () {
+ return this._min;
+ },
+ },
+
+ /**
+ * Maximum allowed value for the property. This is the maximum of all
+ * values after the transforms based on the offset and scale properties
+ * have been applied.
+ *
+ * @memberof PpeMetadata.prototype
+ * @type {number|undefined}
+ * @readonly
+ * @private
+ */
+ max: {
+ get: function () {
+ return this._max;
+ },
+ },
+
+ /**
+ * Possible error source contents
+ *
+ * @memberof PpeMetadata.prototype
+ * @type {PpeSource}
+ * @readonly
+ * @private
+ */
+ source: {
+ get: function () {
+ return this._source;
+ },
+ },
+});
+
+export default PpeMetadata;
diff --git a/packages/engine/Source/Scene/Model/Extensions/Gpm/PpeSource.js b/packages/engine/Source/Scene/Model/Extensions/Gpm/PpeSource.js
new file mode 100644
index 000000000000..d44acabe4e1c
--- /dev/null
+++ b/packages/engine/Source/Scene/Model/Extensions/Gpm/PpeSource.js
@@ -0,0 +1,72 @@
+/**
+ * An enum of per-point error sources
+ *
+ * @enum {string}
+ * @experimental This feature is not final and is subject to change without Cesium's standard deprecation policy.
+ */
+const PpeSource = {
+ /**
+ * The PPE standard deviation of error in the x dimension of the MCS (sigma x). Value will be squared
+ * and used to populate the (1,1) element in the PPE covariance matrix.
+ *
+ * @type {string}
+ * @constant
+ */
+ SIGX: "SIGX",
+
+ /**
+ * The PPE standard deviation of error in the y dimension of the MCS (sigma y). Value will be squared
+ * and used to populate the (2,2) element in the PPE covariance matrix.
+ *
+ * @type {string}
+ * @constant
+ */
+ SIGY: "SIGY",
+
+ /**
+ * The PPE standard deviation of error in the z dimension of the MCS (sigma z). Value will be squared
+ * and used to populate the (3,3) element in the PPE covariance matrix.
+ *
+ * @type {string}
+ * @constant
+ */
+ SIGZ: "SIGZ",
+
+ /**
+ * The PPE variance of error in the x dimension of the MCS (sigma x2). Value will be used to populate
+ * the (1,1) element in the PPE covariance matrix.
+ *
+ * @type {string}
+ * @constant
+ */
+ VARX: "VARX",
+
+ /**
+ * The PPE variance of error in the y dimension of the MCS (sigma y2). Value will be used to populate
+ * the (2,2) element in the PPE covariance matrix.
+ *
+ * @type {string}
+ * @constant
+ */
+ VARY: "VARY",
+
+ /**
+ * The PPE variance of error in the z dimension of the MCS (sigma z2). Value will be used to populate
+ * the (3,3) element in the PPE covariance matrix.
+ *
+ * @type {string}
+ * @constant
+ */
+ VARZ: "VARZ",
+
+ /**
+ * The PPE radial error in the horizontal dimension (x-y) of the MCS (sigma radial) . Value will be squared
+ * and used to populate the (1,1) and (2,2) element in the PPE covariance matrix.
+ *
+ * @type {string}
+ * @constant
+ */
+ SIGR: "VARZ",
+};
+
+export default Object.freeze(PpeSource);
diff --git a/packages/engine/Source/Scene/Model/Extensions/Gpm/PpeTexture.js b/packages/engine/Source/Scene/Model/Extensions/Gpm/PpeTexture.js
new file mode 100644
index 000000000000..07d2fb5b6be8
--- /dev/null
+++ b/packages/engine/Source/Scene/Model/Extensions/Gpm/PpeTexture.js
@@ -0,0 +1,115 @@
+import Check from "../Core/Check.js";
+
+/**
+ * PPE (Per-Point Error) texture in `NGA_gpm_local`.
+ *
+ * This is a valid glTF `TextureInfo` object (with a required `index`
+ * and an optional `texCoord)`, with additional properties that
+ * describe the structure of the metdata that is stored in the texture.
+ *
+ * @private
+ * @experimental This feature is not final and is subject to change without Cesium's standard deprecation policy.
+ */
+function PpeTexture(options) {
+ //>>includeStart('debug', pragmas.debug);
+ Check.typeOf.object("options.traits", options.traits);
+ Check.typeOf.object("options.index", options.index);
+ //>>includeEnd('debug');
+
+ this._traits = options.traits;
+ this._noData = options.noData;
+ this._offset = options.offset;
+ this._scale = options.scale;
+ this._index = options.index;
+ this._texCoord = options.texCoord;
+}
+
+Object.defineProperties(PpeTexture.prototype, {
+ /**
+ * The data contained here applies to this node and corresponding
+ * texture.
+ *
+ * @memberof PpeTexture.prototype
+ * @type {PpeMetadata}
+ * @readonly
+ * @private
+ */
+ traits: {
+ get: function () {
+ return this._traits;
+ },
+ },
+
+ /**
+ * A value to represent missing data - also known as a sentinel value -
+ * wherever it appears.
+ *
+ * @memberof PpeTexture.prototype
+ * @type {number|undefined}
+ * @readonly
+ * @private
+ */
+ noData: {
+ get: function () {
+ return this._noData;
+ },
+ },
+
+ /**
+ * An offset to apply to property values.
+ *
+ * @memberof PpeTexture.prototype
+ * @type {number|undefined}
+ * @readonly
+ * @private
+ */
+ offset: {
+ get: function () {
+ return this._offset;
+ },
+ },
+
+ /**
+ * An scale to apply to property values.
+ *
+ * @memberof PpeTexture.prototype
+ * @type {number|undefined}
+ * @readonly
+ * @private
+ */
+ scale: {
+ get: function () {
+ return this._scale;
+ },
+ },
+
+ /**
+ * The index of the texture
+ *
+ * @memberof PpeTexture.prototype
+ * @type {number}
+ * @readonly
+ * @private
+ */
+ index: {
+ get: function () {
+ return this._index;
+ },
+ },
+
+ /**
+ * The set index of texture's TEXCOORD attribute used for texture coordinate mapping.
+ *
+ * @memberof PpeTexture.prototype
+ * @type {number|undefined}
+ * @readonly
+ * @private
+ */
+ texCoord: {
+ get: function () {
+ return this._texCoord;
+ },
+ },
+});
+
+export default PpeTexture;
diff --git a/packages/engine/Source/Scene/Model/Extensions/Gpm/Spdcf.js b/packages/engine/Source/Scene/Model/Extensions/Gpm/Spdcf.js
new file mode 100644
index 000000000000..548752d533e4
--- /dev/null
+++ b/packages/engine/Source/Scene/Model/Extensions/Gpm/Spdcf.js
@@ -0,0 +1,87 @@
+import Check from "../Core/Check.js";
+
+/**
+ * Variables for a Strictly Positive-Definite Correlation Function.
+ *
+ * Parameters (A, alpha, beta, T) used to describe the correlation decrease
+ * between points as a function of delta time.
+ *
+ * @private
+ * @experimental This feature is not final and is subject to change without Cesium's standard deprecation policy.
+ */
+function Spdcf(options) {
+ //>>includeStart('debug', pragmas.debug);
+ Check.typeOf.number.greaterThan("options.A", options.A, 0.0);
+ Check.typeOf.number.lessThanOrEquals("options.A", options.A, 1.0);
+ Check.typeOf.number.greaterThanOrEquals("options.alpha", options.alpha, 0.0);
+ Check.typeOf.number.lessThan("options.alpha", options.alpha, 1.0);
+ Check.typeOf.number.greaterThanOrEquals("options.beta", options.beta, 0.0);
+ Check.typeOf.number.lessThanOrEquals("options.beta", options.beta, 10.0);
+ Check.typeOf.number.greaterThan("options.T", options.T, 0.0);
+ //>>includeEnd('debug');
+
+ this._A = options.A;
+ this._alpha = options.alpha;
+ this._beta = options.beta;
+ this._T = options.T;
+}
+
+Object.defineProperties(Spdcf.prototype, {
+ /**
+ * In (0, 1]
+ *
+ * @memberof Spdcf.prototype
+ * @type {number}
+ * @readonly
+ * @private
+ */
+ A: {
+ get: function () {
+ return this._A;
+ },
+ },
+
+ /**
+ * In [0, 1)
+ *
+ * @memberof Spdcf.prototype
+ * @type {number}
+ * @readonly
+ * @private
+ */
+ alpha: {
+ get: function () {
+ return this._alpha;
+ },
+ },
+
+ /**
+ * In [0, 10]
+ *
+ * @memberof Spdcf.prototype
+ * @type {number}
+ * @readonly
+ * @private
+ */
+ beta: {
+ get: function () {
+ return this._beta;
+ },
+ },
+
+ /**
+ * In (0, +inf)
+ *
+ * @memberof Spdcf.prototype
+ * @type {number}
+ * @readonly
+ * @private
+ */
+ T: {
+ get: function () {
+ return this._T;
+ },
+ },
+});
+
+export default Spdcf;
diff --git a/packages/engine/Source/Scene/Model/Extensions/Gpm/StorageType.js b/packages/engine/Source/Scene/Model/Extensions/Gpm/StorageType.js
new file mode 100644
index 000000000000..f706c32b63e1
--- /dev/null
+++ b/packages/engine/Source/Scene/Model/Extensions/Gpm/StorageType.js
@@ -0,0 +1,27 @@
+/**
+ * An enum of storage types for covariance information
+ *
+ * @enum {string}
+ * @experimental This feature is not final and is subject to change without Cesium's standard deprecation policy.
+ */
+const StorageType = {
+ /**
+ * Store the full error covariance of the anchor points, to include the cross-covariance terms
+ *
+ * @type {string}
+ * @constant
+ */
+ Direct: "Direct",
+
+ /**
+ * A full covariance matrix is stored for each of the anchor points. However, in this case the
+ * cross-covariance terms are not directly stored, but can be computed by a set of spatial
+ * correlation function parameters which are stored in the metadata.
+ *
+ * @type {string}
+ * @constant
+ */
+ Indirect: "Indirect",
+};
+
+export default Object.freeze(StorageType);
From d7f1c9195afb7016cd5e0c290f7dc740195903d9 Mon Sep 17 00:00:00 2001
From: Marco Hutter
Date: Wed, 18 Sep 2024 18:30:57 +0200
Subject: [PATCH 02/38] Fix import and usage of 'Check'
---
.../Scene/Model/Extensions/Gpm/AnchorPointDirect.js | 4 ++--
.../Scene/Model/Extensions/Gpm/AnchorPointIndirect.js | 4 ++--
.../Source/Scene/Model/Extensions/Gpm/CorrelationGroup.js | 2 +-
.../Source/Scene/Model/Extensions/Gpm/GltfGpmLocal.js | 8 ++++----
.../Source/Scene/Model/Extensions/Gpm/PpeMetadata.js | 2 +-
.../Source/Scene/Model/Extensions/Gpm/PpeTexture.js | 2 +-
.../engine/Source/Scene/Model/Extensions/Gpm/Spdcf.js | 2 +-
7 files changed, 12 insertions(+), 12 deletions(-)
diff --git a/packages/engine/Source/Scene/Model/Extensions/Gpm/AnchorPointDirect.js b/packages/engine/Source/Scene/Model/Extensions/Gpm/AnchorPointDirect.js
index 79e71895214a..903330dada7f 100644
--- a/packages/engine/Source/Scene/Model/Extensions/Gpm/AnchorPointDirect.js
+++ b/packages/engine/Source/Scene/Model/Extensions/Gpm/AnchorPointDirect.js
@@ -1,4 +1,4 @@
-import Check from "../Core/Check.js";
+import Check from "../../../../Core/Check.js";
/**
* Metadata for one stored anchor point using direct storage
@@ -8,7 +8,7 @@ import Check from "../Core/Check.js";
*/
function AnchorPointDirect(options) {
//>>includeStart('debug', pragmas.debug);
- Check.typeOf.object("options.position", options.property);
+ Check.typeOf.object("options.position", options.position);
Check.typeOf.object("options.adjustmentParams", options.adjustmentParams);
//>>includeEnd('debug');
diff --git a/packages/engine/Source/Scene/Model/Extensions/Gpm/AnchorPointIndirect.js b/packages/engine/Source/Scene/Model/Extensions/Gpm/AnchorPointIndirect.js
index c70996f33485..f0102db32943 100644
--- a/packages/engine/Source/Scene/Model/Extensions/Gpm/AnchorPointIndirect.js
+++ b/packages/engine/Source/Scene/Model/Extensions/Gpm/AnchorPointIndirect.js
@@ -1,4 +1,4 @@
-import Check from "../Core/Check.js";
+import Check from "../../../../Core/Check.js";
/**
* Metadata for one stored anchor point.
@@ -8,7 +8,7 @@ import Check from "../Core/Check.js";
*/
function AnchorPointIndirect(options) {
//>>includeStart('debug', pragmas.debug);
- Check.typeOf.object("options.position", options.property);
+ Check.typeOf.object("options.position", options.position);
Check.typeOf.object("options.adjustmentParams", options.adjustmentParams);
Check.typeOf.object("options.covarianceMatrix", options.covarianceMatrix);
//>>includeEnd('debug');
diff --git a/packages/engine/Source/Scene/Model/Extensions/Gpm/CorrelationGroup.js b/packages/engine/Source/Scene/Model/Extensions/Gpm/CorrelationGroup.js
index e535ea6a43a2..cc82812aa608 100644
--- a/packages/engine/Source/Scene/Model/Extensions/Gpm/CorrelationGroup.js
+++ b/packages/engine/Source/Scene/Model/Extensions/Gpm/CorrelationGroup.js
@@ -1,4 +1,4 @@
-import Check from "../Core/Check.js";
+import Check from "../../../../Core/Check.js";
/**
* Metadata identifying parameters using same correlation modeling and
diff --git a/packages/engine/Source/Scene/Model/Extensions/Gpm/GltfGpmLocal.js b/packages/engine/Source/Scene/Model/Extensions/Gpm/GltfGpmLocal.js
index 016c27bf81cb..f6b9e4aab5df 100644
--- a/packages/engine/Source/Scene/Model/Extensions/Gpm/GltfGpmLocal.js
+++ b/packages/engine/Source/Scene/Model/Extensions/Gpm/GltfGpmLocal.js
@@ -1,7 +1,7 @@
-import defined from "../Core/defined.js";
-import Check from "../Core/Check.js";
+import defined from "../../../../Core/defined.js";
+import Check from "../../../../Core/Check.js";
+import RuntimeError from "../../../../Core/RuntimeError.js";
import StorageType from "./StorageType.js";
-import RuntimeError from "../Core/RuntimeError.js";
/**
* The GPM metadata for a Ground-Space Indirect implementation stored
@@ -12,7 +12,7 @@ import RuntimeError from "../Core/RuntimeError.js";
*/
function GltfGpmLocal(options) {
//>>includeStart('debug', pragmas.debug);
- Check.typeOf.object("options.storageType", options.storageType);
+ Check.typeOf.string("options.storageType", options.storageType);
//>>includeEnd('debug');
this._storageType = options.storageType;
diff --git a/packages/engine/Source/Scene/Model/Extensions/Gpm/PpeMetadata.js b/packages/engine/Source/Scene/Model/Extensions/Gpm/PpeMetadata.js
index 26f0d152284a..de255f5b6dd9 100644
--- a/packages/engine/Source/Scene/Model/Extensions/Gpm/PpeMetadata.js
+++ b/packages/engine/Source/Scene/Model/Extensions/Gpm/PpeMetadata.js
@@ -1,4 +1,4 @@
-import Check from "../Core/Check.js";
+import Check from "../../../../Core/Check.js";
/**
* Metadata related to the stored PPE data.
diff --git a/packages/engine/Source/Scene/Model/Extensions/Gpm/PpeTexture.js b/packages/engine/Source/Scene/Model/Extensions/Gpm/PpeTexture.js
index 07d2fb5b6be8..5214a6e34ac6 100644
--- a/packages/engine/Source/Scene/Model/Extensions/Gpm/PpeTexture.js
+++ b/packages/engine/Source/Scene/Model/Extensions/Gpm/PpeTexture.js
@@ -1,4 +1,4 @@
-import Check from "../Core/Check.js";
+import Check from "../../../../Core/Check.js";
/**
* PPE (Per-Point Error) texture in `NGA_gpm_local`.
diff --git a/packages/engine/Source/Scene/Model/Extensions/Gpm/Spdcf.js b/packages/engine/Source/Scene/Model/Extensions/Gpm/Spdcf.js
index 548752d533e4..7687ed32ed8d 100644
--- a/packages/engine/Source/Scene/Model/Extensions/Gpm/Spdcf.js
+++ b/packages/engine/Source/Scene/Model/Extensions/Gpm/Spdcf.js
@@ -1,4 +1,4 @@
-import Check from "../Core/Check.js";
+import Check from "../../../../Core/Check.js";
/**
* Variables for a Strictly Positive-Definite Correlation Function.
From e09cb04035f4e654249cccab80e582f5aac0f222 Mon Sep 17 00:00:00 2001
From: Marco Hutter
Date: Wed, 18 Sep 2024 18:32:04 +0200
Subject: [PATCH 03/38] Start wiring in GPM loader
---
packages/engine/Source/Scene/GltfGpmLoader.js | 466 +++++-------------
packages/engine/Source/Scene/GltfLoader.js | 105 ++++
.../Scene/GltfMeshPrimitiveGpmLoader.js | 465 +++++++++++++++++
packages/engine/Source/Scene/Model/Model.js | 18 +
.../Source/Scene/Model/Model3DTileContent.js | 10 +-
.../engine/Source/Scene/ModelComponents.js | 6 +
6 files changed, 737 insertions(+), 333 deletions(-)
create mode 100644 packages/engine/Source/Scene/GltfMeshPrimitiveGpmLoader.js
diff --git a/packages/engine/Source/Scene/GltfGpmLoader.js b/packages/engine/Source/Scene/GltfGpmLoader.js
index b3463abb8054..4fe1cf50c46b 100644
--- a/packages/engine/Source/Scene/GltfGpmLoader.js
+++ b/packages/engine/Source/Scene/GltfGpmLoader.js
@@ -1,24 +1,23 @@
+import Cartesian3 from "../Core/Cartesian3.js";
import Check from "../Core/Check.js";
import defaultValue from "../Core/defaultValue.js";
import defined from "../Core/defined.js";
-import ResourceCache from "./ResourceCache.js";
+import Matrix3 from "../Core/Matrix3.js";
+import AnchorPointDirect from "./Model/Extensions/Gpm/AnchorPointDirect.js";
+import AnchorPointIndirect from "./Model/Extensions/Gpm/AnchorPointIndirect.js";
+import CorrelationGroup from "./Model/Extensions/Gpm/CorrelationGroup.js";
+import GltfGpmLocal from "./Model/Extensions/Gpm/GltfGpmLocal.js";
+import Spdcf from "./Model/Extensions/Gpm/Spdcf.js";
+import StorageType from "./Model/Extensions/Gpm/StorageType.js";
import ResourceLoader from "./ResourceLoader.js";
import ResourceLoaderState from "./ResourceLoaderState.js";
-import PropertyTexture from "./PropertyTexture.js";
-import StructuralMetadata from "./StructuralMetadata.js";
-import MetadataSchema from "./MetadataSchema.js";
/**
- * Loads glTF NGA_gpm_local
+ * Loads glTF NGA_gpm_local from the root of a glTF object
*
* Implements the {@link ResourceLoader} interface.
*
- * Implementation note: This is an experimental implementation. It is based
- * on a GltfStructuralMetadataLoader, by removing everything that is not
- * related to property textures, and translating the "ppeTextures" of
- * the NGA_gpm_local extension into property textures. These will be
- * returned as part of a `StructuralMetadata` object, which may override
- * any `StructuralMetadata` that was read directly from the glTF.
+ * Implementation note: This is an experimental implementation.
*
* @alias GltfGpmLoader
* @constructor
@@ -29,8 +28,6 @@ import MetadataSchema from "./MetadataSchema.js";
* @param {string} [options.extension] The NGA_gpm_local
extension object.
* @param {Resource} options.gltfResource The {@link Resource} containing the glTF.
* @param {Resource} options.baseResource The {@link Resource} that paths in the glTF JSON are relative to.
- * @param {SupportedImageFormats} options.supportedImageFormats The supported image formats.
- * @param {FrameState} options.frameState The frame state.
* @param {string} [options.cacheKey] The cache key of the resource.
* @param {boolean} [options.asynchronous=true] Determines if WebGL resource creation will be spread out over several frames or block until all WebGL resources are created.
*
@@ -43,8 +40,6 @@ function GltfGpmLoader(options) {
const extension = options.extension;
const gltfResource = options.gltfResource;
const baseResource = options.baseResource;
- const supportedImageFormats = options.supportedImageFormats;
- const frameState = options.frameState;
const cacheKey = options.cacheKey;
const asynchronous = defaultValue(options.asynchronous, true);
@@ -53,22 +48,19 @@ function GltfGpmLoader(options) {
Check.typeOf.object("options.extension", extension);
Check.typeOf.object("options.gltfResource", gltfResource);
Check.typeOf.object("options.baseResource", baseResource);
- Check.typeOf.object("options.supportedImageFormats", supportedImageFormats);
- Check.typeOf.object("options.frameState", frameState);
//>>includeEnd('debug');
this._gltfResource = gltfResource;
this._baseResource = baseResource;
this._gltf = gltf;
this._extension = extension;
- this._supportedImageFormats = supportedImageFormats;
- this._frameState = frameState;
this._cacheKey = cacheKey;
this._asynchronous = asynchronous;
- this._textureLoaders = [];
- this._textureIds = [];
- this._structuralMetadata = undefined;
- this._state = ResourceLoaderState.UNLOADED;
+ this._gltfGpmLocal = undefined;
+
+ // Immediately go into the "LOADED" state, since there
+ // are no resources to wait for
+ this._state = ResourceLoaderState.LOADED;
this._promise = undefined;
}
@@ -93,46 +85,21 @@ Object.defineProperties(GltfGpmLoader.prototype, {
},
},
/**
- * The parsed structural metadata
+ * The parsed GltfGpmLocal object
*
* @memberof GltfGpmLoader.prototype
*
- * @type {StructuralMetadata}
+ * @type {GltfGpmLocal}
* @readonly
* @private
*/
- structuralMetadata: {
+ gltfGpmLocal: {
get: function () {
- return this._structuralMetadata;
+ return this._gltfGpmLocal;
},
},
});
-async function loadResources(loader) {
- try {
- const texturesPromise = loadTextures(loader);
- await texturesPromise;
-
- if (loader.isDestroyed()) {
- return;
- }
-
- loader._gltf = undefined; // No longer need to hold onto the glTF
-
- loader._state = ResourceLoaderState.LOADED;
- return loader;
- } catch (error) {
- if (loader.isDestroyed()) {
- return;
- }
-
- loader.unload();
- loader._state = ResourceLoaderState.FAILED;
- const errorMessage = "Failed to load GPM data";
- throw loader.getError(errorMessage, error);
- }
-}
-
/**
* Loads the resource.
* @returns {Promise} A promise which resolves to the loader when the resource loading is completed.
@@ -142,211 +109,32 @@ GltfGpmLoader.prototype.load = function () {
if (defined(this._promise)) {
return this._promise;
}
-
- this._state = ResourceLoaderState.LOADING;
- this._promise = loadResources(this);
+ this._promise = Promise.resolve(this);
return this._promise;
};
-function gatherUsedTextureIds(gpmExtension) {
- // Gather the used textures
- const textureIds = {};
- const ppeTextures = gpmExtension.ppeTextures;
- if (defined(ppeTextures)) {
- for (let i = 0; i < ppeTextures.length; i++) {
- const ppeTexture = ppeTextures[i];
- // The texture is a valid textureInfo.
- textureIds[ppeTexture.index] = ppeTexture;
- }
- }
- return textureIds;
-}
-
-function loadTextures(gpmLoader) {
- let textureIds;
- if (defined(gpmLoader._extension)) {
- textureIds = gatherUsedTextureIds(gpmLoader._extension);
- }
-
- const gltf = gpmLoader._gltf;
- const gltfResource = gpmLoader._gltfResource;
- const baseResource = gpmLoader._baseResource;
- const supportedImageFormats = gpmLoader._supportedImageFormats;
- const frameState = gpmLoader._frameState;
- const asynchronous = gpmLoader._asynchronous;
-
- // Load the textures
- const texturePromises = [];
- for (const textureId in textureIds) {
- if (textureIds.hasOwnProperty(textureId)) {
- const textureLoader = ResourceCache.getTextureLoader({
- gltf: gltf,
- textureInfo: textureIds[textureId],
- gltfResource: gltfResource,
- baseResource: baseResource,
- supportedImageFormats: supportedImageFormats,
- frameState: frameState,
- asynchronous: asynchronous,
- });
- gpmLoader._textureLoaders.push(textureLoader);
- gpmLoader._textureIds.push(textureId);
- texturePromises.push(textureLoader.load());
- }
- }
-
- return Promise.all(texturePromises);
-}
-
-/**
- * A static mapping from PPE texture property identifier keys
- * to `MetadataSchema` instances. This is used to create each
- * schema (with a certain structure) only ONCE in
- * obtainPpeTexturesMetadataSchema
- *
- * @private
- */
-GltfGpmLoader.ppeTexturesMetadataSchemaCache = new Map();
-
-/**
- * Create the JSON description of a metadata class that treats
- * the given PPE texture as a property texture property(!).
- *
- * @param {any} ppeTexture - The PPE texture
- * @param {number} index - The index of the texture in the extension
- * @returns The class JSON
- */
-GltfGpmLoader.prototype.createPpeTextureClassJson = function (
- ppeTexture,
- index
-) {
- const traits = ppeTexture.traits;
- const ppePropertyName = traits.source;
-
- // XXX_UNCERTAINTY: The ppeTexture will have a structure
- // like this:
- //
- //"ppeTextures" : [
- // {
- // "traits" : {
- // "source" : "SIGZ",
- // "min" : 0.0,
- // "max" : 16.0
- // },
- // "index" : 2,
- // "noData" : 255,
- // "offset" : 0.0,
- // "scale" : 0.06274509803921569,
- // "texCoord" : 1
- // },
- //
- // Note that in GPM 1.2d, the min/max should be part of the
- // texture, but in the actual data (GPM 1.2i), they are in
- // the traits (ppeMetadata). And there, they/ seem to denote
- // the min/max values that actually appear in the texture.
- // So they are integrated into the 'scale' factor here:
- const min = traits.min ?? 0;
- const max = traits.max ?? 255;
- const minMaxScale = 255.0 / (max - min);
- const offset = ppeTexture.offset;
- const scale = ppeTexture.scale * minMaxScale;
- const classJson = {
- name: `PPE texture class ${index}`,
- properties: {
- [ppePropertyName]: {
- name: "PPE",
- type: "SCALAR",
- componentType: "UINT8",
- normalized: true,
- offset: offset,
- scale: scale,
- },
- },
- };
- return classJson;
-};
-
/**
- * Returns the `MetadataSchema` for the PPE textures in this instance.
- *
- * This method will return a (statically/globally) cached metadata
- * schema that reflects the structure of the PPE textures in this
- * instance, creating and caching it if necessary.
+ * Creates a Matrix3 that describes a covariance matrix (which is
+ * symmetric) from the array containing the upper triangle, in
+ * column-major order.
*
- * For details on the cache key, see `collectPpeTexturePropertyIdentifiers`
- *
- * @returns The `MetadataSchema`
+ * @param {number[]} array The input array
+ * @returns The Matrix3
*/
-GltfGpmLoader.prototype.obtainPpeTexturesMetadataSchema = function () {
- const ppeTexturePropertyIdentifiers = this.collectPpeTexturePropertyIdentifiers();
- const key = ppeTexturePropertyIdentifiers.toString();
- let ppeTexturesMetadataSchema = GltfGpmLoader.ppeTexturesMetadataSchemaCache.get(
- key
- );
- if (defined(ppeTexturesMetadataSchema)) {
- // XXX_UNCERTAINTY Debug log
- //console.log(`Using cached schema for GPM PPE textures with key ${key}`);
- return ppeTexturesMetadataSchema;
- }
-
- // XXX_UNCERTAINTY Debug log - if caching works, this should be printed only ONCE!
- console.log(`Creating schema for GPM PPE textures with key ${key}`);
- const schemaId = `PPE_TEXTURE_SCHEMA_${GltfGpmLoader.ppeTexturesMetadataSchemaCache.size}`;
- const ppeTexturesMetadataSchemaJson = {
- id: schemaId,
- classes: {},
- };
-
- const extension = this._extension;
- if (defined(extension.ppeTextures)) {
- for (let i = 0; i < extension.ppeTextures.length; i++) {
- const ppeTexture = extension.ppeTextures[i];
- const classId = `ppeTexture_${i}`;
- const classJson = this.createPpeTextureClassJson(ppeTexture, i);
- ppeTexturesMetadataSchemaJson.classes[classId] = classJson;
- }
- }
-
- ppeTexturesMetadataSchema = MetadataSchema.fromJson(
- ppeTexturesMetadataSchemaJson
+function createCovarianceMatrixFromUpperTriangle(array) {
+ const covarianceMatrix = new Matrix3(
+ array[0],
+ array[1],
+ array[3],
+ array[1],
+ array[2],
+ array[4],
+ array[3],
+ array[4],
+ array[5]
);
- GltfGpmLoader.ppeTexturesMetadataSchemaCache.set(
- key,
- ppeTexturesMetadataSchema
- );
- return ppeTexturesMetadataSchema;
-};
-
-/**
- * Creates an array of strings that serve as identifiers for PPE textures.
- *
- * Each glTF may define multiple `ppeTexture` objects within the
- * `NGA_gpm_local` extensions. Each of these textures corresponds
- * to one 'property texture property(!)' in a metadata schema.
- *
- * This method will create an array where each element is a (JSON)
- * string representation of the parts of a GPM PPE texture definition
- * that are relevant for distinguishing two PPE textures in terms
- * of their structure within a `StructuralMetadata`.
- *
- * @returns The identifiers
- */
-GltfGpmLoader.prototype.collectPpeTexturePropertyIdentifiers = function () {
- const extension = this._extension;
- const ppeTexturePropertyIdentifiers = [];
- if (defined(extension.ppeTextures)) {
- for (let i = 0; i < extension.ppeTextures.length; i++) {
- const ppeTexture = extension.ppeTextures[i];
- // The following will create an identifier that can be used
- // to define two PPE textures as "representing the same
- // property texture property" within a structural metadata
- // schema.
- const classJson = this.createPpeTextureClassJson(ppeTexture, i);
- const ppeTexturePropertyIdentifier = JSON.stringify(classJson);
- ppeTexturePropertyIdentifiers.push(ppeTexturePropertyIdentifier);
- }
- }
- return ppeTexturePropertyIdentifiers;
-};
+ return covarianceMatrix;
+}
/**
* Processes the resource until it becomes ready.
@@ -362,108 +150,122 @@ GltfGpmLoader.prototype.process = function (frameState) {
if (this._state === ResourceLoaderState.READY) {
return true;
}
-
+ if (this._state === ResourceLoaderState.FAILED) {
+ return true;
+ }
if (this._state !== ResourceLoaderState.LOADED) {
return false;
}
- const textureLoaders = this._textureLoaders;
- const textureLoadersLength = textureLoaders.length;
- let ready = true;
- for (let i = 0; i < textureLoadersLength; ++i) {
- const textureLoader = textureLoaders[i];
- const textureReady = textureLoader.process(frameState);
- ready = ready && textureReady;
- }
+ console.log("PARSE AND STORE HERE! ", this._extension);
- if (!ready) {
- return false;
- }
+ const extensionJson = this._extension;
- const textures = {};
- for (let i = 0; i < this._textureIds.length; ++i) {
- const textureId = this._textureIds[i];
- const textureLoader = textureLoaders[i];
- if (!textureLoader.isDestroyed()) {
- textures[textureId] = textureLoader.texture;
+ const storageType = extensionJson.storageType;
+ if (storageType === StorageType.Direct) {
+ const anchorPointsDirect = [];
+ const anchorPointsDirectJson = extensionJson.anchorPointsDirect;
+ for (const anchorPointDirectJson of anchorPointsDirectJson) {
+ const position = Cartesian3.fromArray(
+ anchorPointDirectJson.position,
+ 0,
+ new Cartesian3()
+ );
+ const adjustmentParams = Cartesian3.fromArray(
+ anchorPointDirectJson.adjustmentParams,
+ 0,
+ new Cartesian3()
+ );
+ const anchorPointDirect = new AnchorPointDirect({
+ position: position,
+ adjustmentParams: adjustmentParams,
+ });
+ anchorPointsDirect.push(anchorPointDirect);
}
+ const covarianceDirect = createCovarianceMatrixFromUpperTriangle(
+ extensionJson.covarianceDirectUpperTriangle
+ );
+
+ this._gltfGpmLocal = new GltfGpmLocal({
+ storageType: storageType,
+ anchorPointsDirect: anchorPointsDirect,
+ covarianceDirect: covarianceDirect,
+ });
+ this._state = ResourceLoaderState.READY;
+ return true;
}
-
- const ppeTexturesMetadataSchema = this.obtainPpeTexturesMetadataSchema();
- const extension = this._extension;
- const propertyTextures = [];
- if (defined(extension.ppeTextures)) {
- for (let i = 0; i < extension.ppeTextures.length; i++) {
- const ppeTexture = extension.ppeTextures[i];
- const classId = `ppeTexture_${i}`;
- const traits = ppeTexture.traits;
- const ppePropertyName = traits.source;
- const metadataClass = ppeTexturesMetadataSchema.classes[classId];
-
- // XXX_UNCERTAINTY Debug log
- //console.log(
- // `Creating property texture with class ${classId} and property ${ppePropertyName}`
- //);
-
- const ppeTextureAsPropertyTexture = {
- class: classId,
- properties: {
- [ppePropertyName]: {
- index: ppeTexture.index,
- texCoord: ppeTexture.texCoord,
- },
- },
- };
- propertyTextures.push(
- new PropertyTexture({
- id: i,
- name: ppeTexture.name,
- propertyTexture: ppeTextureAsPropertyTexture,
- class: metadataClass,
- textures: textures,
- })
+ if (storageType === StorageType.Indirect) {
+ const anchorPointsIndirect = [];
+ const anchorPointsIndirectJson = extensionJson.anchorPointsIndirect;
+ for (const anchorPointIndirectJson of anchorPointsIndirectJson) {
+ const position = Cartesian3.fromArray(
+ anchorPointIndirectJson.position,
+ 0,
+ new Cartesian3()
+ );
+ const adjustmentParams = Cartesian3.fromArray(
+ anchorPointIndirectJson.adjustmentParams,
+ 0,
+ new Cartesian3()
+ );
+ const covarianceMatrix = createCovarianceMatrixFromUpperTriangle(
+ anchorPointIndirectJson.covarianceMatrix
);
+ const anchorPointIndirect = new AnchorPointIndirect({
+ position: position,
+ adjustmentParams: adjustmentParams,
+ covarianceMatrix: covarianceMatrix,
+ });
+ anchorPointsIndirect.push(anchorPointIndirect);
}
- }
- const structuralMetadata = new StructuralMetadata({
- schema: ppeTexturesMetadataSchema,
- propertyTables: [],
- propertyTextures: propertyTextures,
- propertyAttributes: [],
- statistics: extension.statistics,
- extras: extension.extras,
- extensions: extension.extensions,
- });
- this._structuralMetadata = structuralMetadata;
+ const intraTileCorrelationGroupsJson =
+ extensionJson.intraTileCorrelationGroups;
+ const intraTileCorrelationGroups = [];
- this._state = ResourceLoaderState.READY;
- return true;
-};
+ for (const correlationGroupJson of intraTileCorrelationGroupsJson) {
+ const groupFlags = correlationGroupJson.groupFlags;
+ const rotationThetas = Cartesian3.fromArray(
+ correlationGroupJson.rotationThetas,
+ 0,
+ new Cartesian3()
+ );
+ const params = [];
+ for (const paramJson of correlationGroupJson.params) {
+ const param = new Spdcf({
+ A: paramJson.A,
+ alpha: paramJson.alpha,
+ beta: paramJson.beta,
+ T: paramJson.T,
+ });
+ params.push(param);
+ }
+ const correlationGroup = new CorrelationGroup({
+ groupFlags: groupFlags,
+ rotationThetas: rotationThetas,
+ params: params,
+ });
+ intraTileCorrelationGroups.push(correlationGroup);
+ }
-function unloadTextures(gpmLoader) {
- const textureLoaders = gpmLoader._textureLoaders;
- const textureLoadersLength = textureLoaders.length;
- for (let i = 0; i < textureLoadersLength; ++i) {
- ResourceCache.unload(textureLoaders[i]);
+ this._gltfGpmLocal = new GltfGpmLocal({
+ storageType: storageType,
+ anchorPointsIndirect: anchorPointsIndirect,
+ intraTileCorrelationGroups: intraTileCorrelationGroups,
+ });
+ this._state = ResourceLoaderState.READY;
+ return true;
}
- gpmLoader._textureLoaders.length = 0;
- gpmLoader._textureIds.length = 0;
-}
+ this._state = ResourceLoaderState.FAILED;
+ return false;
+};
/**
* Unloads the resource.
* @private
*/
GltfGpmLoader.prototype.unload = function () {
- unloadTextures(this);
-
- if (defined(this._schemaLoader)) {
- ResourceCache.unload(this._schemaLoader);
- }
- this._schemaLoader = undefined;
-
- this._structuralMetadata = undefined;
+ this._gltfGpmLocal = undefined;
};
export default GltfGpmLoader;
diff --git a/packages/engine/Source/Scene/GltfLoader.js b/packages/engine/Source/Scene/GltfLoader.js
index 18f825a6ae4e..490cbfb087af 100644
--- a/packages/engine/Source/Scene/GltfLoader.js
+++ b/packages/engine/Source/Scene/GltfLoader.js
@@ -29,6 +29,8 @@ import ResourceCache from "./ResourceCache.js";
import ResourceLoader from "./ResourceLoader.js";
import SupportedImageFormats from "./SupportedImageFormats.js";
import VertexAttributeSemantic from "./VertexAttributeSemantic.js";
+import GltfGpmLoader from "./GltfGpmLoader.js";
+import GltfMeshPrimitiveGpmLoader from "./GltfMeshPrimitiveGpmLoader.js";
const {
Attribute,
@@ -260,6 +262,8 @@ function GltfLoader(options) {
this._geometryLoaders = [];
this._geometryCallbacks = [];
this._structuralMetadataLoader = undefined;
+ this._gpmLoader = undefined;
+ this._meshPrimitiveGpmLoader = undefined;
this._loadResourcesPromise = undefined;
this._resourcesLoaded = false;
this._texturesLoaded = false;
@@ -471,6 +475,29 @@ function processLoaders(loader, frameState) {
ready = ready && metadataReady;
}
+ const gpmLoader = loader._gpmLoader;
+ if (defined(gpmLoader)) {
+ const gpmReady = gpmLoader.process(frameState);
+ if (gpmReady) {
+ // XXX
+ console.log("Got GPM data, storing ", gpmLoader._gltfGpmLocal);
+ loader._components.extensions["NGA_gpm_local"] = gpmLoader._gltfGpmLocal;
+ }
+ ready = ready && gpmReady;
+ }
+
+ const meshPrimitiveGpmLoader = loader._meshPrimitiveGpmLoader;
+ if (defined(meshPrimitiveGpmLoader)) {
+ const metadataReady = meshPrimitiveGpmLoader.process(frameState);
+ if (metadataReady) {
+ // XXX
+ console.log("Overwriting structural metadata with GPM data");
+ loader._components.structuralMetadata =
+ meshPrimitiveGpmLoader.structuralMetadata;
+ }
+ ready = ready && metadataReady;
+ }
+
if (ready) {
// Geometry requires further processing
loader._state = GltfLoaderState.POST_PROCESSING;
@@ -2453,6 +2480,32 @@ async function loadStructuralMetadata(
return structuralMetadataLoader.load();
}
+async function loadGpm(loader, gltf, extension) {
+ const gpmLoader = new GltfGpmLoader({
+ gltf: gltf,
+ extension: extension,
+ gltfResource: loader._gltfResource,
+ baseResource: loader._baseResource,
+ asynchronous: loader._asynchronous,
+ });
+ loader._gpmLoader = gpmLoader;
+ return gpmLoader.load();
+}
+
+async function loadMeshPrimitiveGpm(loader, gltf, extension, frameState) {
+ const meshPrimitiveGpmLoader = new GltfMeshPrimitiveGpmLoader({
+ gltf: gltf,
+ extension: extension,
+ gltfResource: loader._gltfResource,
+ baseResource: loader._baseResource,
+ supportedImageFormats: loader._supportedImageFormats,
+ frameState: frameState,
+ asynchronous: loader._asynchronous,
+ });
+ loader._meshPrimitiveGpmLoader = meshPrimitiveGpmLoader;
+ return meshPrimitiveGpmLoader.load();
+}
+
function loadAnimationSampler(loader, gltfSampler) {
const animationSampler = new AnimationSampler();
const accessors = loader.gltfJson.accessors;
@@ -2681,6 +2734,39 @@ function parse(loader, frameState) {
loader._loaderPromises.push(promise);
}
+ // Load NGA_gpm_local from root object
+ const gpmExtension = extensions.NGA_gpm_local;
+ if (defined(gpmExtension)) {
+ console.log("Loading GPM from root");
+ const promise = loadGpm(loader, gltf, gpmExtension);
+ loader._loaderPromises.push(promise);
+ }
+
+ // Load NGA_gpm_local from mesh primitives
+ const meshes = gltf.meshes;
+ if (defined(meshes)) {
+ for (const mesh of meshes) {
+ const primitives = mesh.primitives;
+ if (defined(primitives)) {
+ for (const primitive of primitives) {
+ const primitiveExtensions = primitive.extensions;
+ const meshPrimitiveGpmExtension = primitiveExtensions.NGA_gpm_local;
+ if (defined(meshPrimitiveGpmExtension)) {
+ // XXX
+ console.log("Loading GPM from mesh primitive");
+ const promise = loadMeshPrimitiveGpm(
+ loader,
+ gltf,
+ meshPrimitiveGpmExtension,
+ frameState
+ );
+ loader._loaderPromises.push(promise);
+ }
+ }
+ }
+ }
+ }
+
// Gather promises and handle any errors
const readyPromises = [];
readyPromises.push.apply(readyPromises, loader._loaderPromises);
@@ -2745,6 +2831,23 @@ function unloadStructuralMetadata(loader) {
}
}
+function unloadGpm(loader) {
+ if (defined(loader._gpmLoader) && !loader._gpmLoader.isDestroyed()) {
+ loader._gpmLoader.destroy();
+ loader._gpmLoader = undefined;
+ }
+}
+
+function unloadMeshPrimitiveGpm(loader) {
+ if (
+ defined(loader._meshPrimitiveGpmLoader) &&
+ !loader._meshPrimitiveGpmLoader.isDestroyed()
+ ) {
+ loader._meshPrimitiveGpmLoader.destroy();
+ loader._meshPrimitiveGpmLoader = undefined;
+ }
+}
+
/**
* Returns whether the resource has been unloaded.
* @private
@@ -2768,6 +2871,8 @@ GltfLoader.prototype.unload = function () {
unloadGeometry(this);
unloadGeneratedAttributes(this);
unloadStructuralMetadata(this);
+ unloadGpm(this);
+ unloadMeshPrimitiveGpm(this);
this._components = undefined;
this._typedArray = undefined;
diff --git a/packages/engine/Source/Scene/GltfMeshPrimitiveGpmLoader.js b/packages/engine/Source/Scene/GltfMeshPrimitiveGpmLoader.js
new file mode 100644
index 000000000000..724b1d8b1a47
--- /dev/null
+++ b/packages/engine/Source/Scene/GltfMeshPrimitiveGpmLoader.js
@@ -0,0 +1,465 @@
+import Check from "../Core/Check.js";
+import defaultValue from "../Core/defaultValue.js";
+import defined from "../Core/defined.js";
+import ResourceCache from "./ResourceCache.js";
+import ResourceLoader from "./ResourceLoader.js";
+import ResourceLoaderState from "./ResourceLoaderState.js";
+import PropertyTexture from "./PropertyTexture.js";
+import StructuralMetadata from "./StructuralMetadata.js";
+import MetadataSchema from "./MetadataSchema.js";
+
+/**
+ * Loads glTF NGA_gpm_local from a glTF mesh primitive.
+ *
+ * Implements the {@link ResourceLoader} interface.
+ *
+ * Implementation note: This is an experimental implementation. It is based
+ * on a GltfStructuralMetadataLoader, by removing everything that is not
+ * related to property textures, and translating the "ppeTextures" of
+ * the NGA_gpm_local extension into property texture properties. These will
+ * be returned as part of a `StructuralMetadata` object, which may override
+ * any `StructuralMetadata` that was read directly from the glTF.
+ *
+ * @alias GltfMeshPrimitiveGpmLoader
+ * @constructor
+ * @augments ResourceLoader
+ *
+ * @param {object} options Object with the following properties:
+ * @param {object} options.gltf The glTF JSON.
+ * @param {string} [options.extension] The NGA_gpm_local
extension object.
+ * @param {Resource} options.gltfResource The {@link Resource} containing the glTF.
+ * @param {Resource} options.baseResource The {@link Resource} that paths in the glTF JSON are relative to.
+ * @param {SupportedImageFormats} options.supportedImageFormats The supported image formats.
+ * @param {FrameState} options.frameState The frame state.
+ * @param {string} [options.cacheKey] The cache key of the resource.
+ * @param {boolean} [options.asynchronous=true] Determines if WebGL resource creation will be spread out over several frames or block until all WebGL resources are created.
+ *
+ * @private
+ * @experimental This feature is subject to change without Cesium's standard deprecation policy.
+ */
+function GltfMeshPrimitiveGpmLoader(options) {
+ options = defaultValue(options, defaultValue.EMPTY_OBJECT);
+ const gltf = options.gltf;
+ const extension = options.extension;
+ const gltfResource = options.gltfResource;
+ const baseResource = options.baseResource;
+ const supportedImageFormats = options.supportedImageFormats;
+ const frameState = options.frameState;
+ const cacheKey = options.cacheKey;
+ const asynchronous = defaultValue(options.asynchronous, true);
+
+ //>>includeStart('debug', pragmas.debug);
+ Check.typeOf.object("options.gltf", gltf);
+ Check.typeOf.object("options.extension", extension);
+ Check.typeOf.object("options.gltfResource", gltfResource);
+ Check.typeOf.object("options.baseResource", baseResource);
+ Check.typeOf.object("options.supportedImageFormats", supportedImageFormats);
+ Check.typeOf.object("options.frameState", frameState);
+ //>>includeEnd('debug');
+
+ this._gltfResource = gltfResource;
+ this._baseResource = baseResource;
+ this._gltf = gltf;
+ this._extension = extension;
+ this._supportedImageFormats = supportedImageFormats;
+ this._frameState = frameState;
+ this._cacheKey = cacheKey;
+ this._asynchronous = asynchronous;
+ this._textureLoaders = [];
+ this._textureIds = [];
+ this._structuralMetadata = undefined;
+ this._state = ResourceLoaderState.UNLOADED;
+ this._promise = undefined;
+}
+
+if (defined(Object.create)) {
+ GltfMeshPrimitiveGpmLoader.prototype = Object.create(
+ ResourceLoader.prototype
+ );
+ GltfMeshPrimitiveGpmLoader.prototype.constructor = GltfMeshPrimitiveGpmLoader;
+}
+
+Object.defineProperties(GltfMeshPrimitiveGpmLoader.prototype, {
+ /**
+ * The cache key of the resource.
+ *
+ * @memberof GltfMeshPrimitiveGpmLoader.prototype
+ *
+ * @type {string}
+ * @readonly
+ * @private
+ */
+ cacheKey: {
+ get: function () {
+ return this._cacheKey;
+ },
+ },
+ /**
+ * The parsed structural metadata
+ *
+ * @memberof GltfMeshPrimitiveGpmLoader.prototype
+ *
+ * @type {StructuralMetadata}
+ * @readonly
+ * @private
+ */
+ structuralMetadata: {
+ get: function () {
+ return this._structuralMetadata;
+ },
+ },
+});
+
+async function loadResources(loader) {
+ try {
+ const texturesPromise = loadTextures(loader);
+ await texturesPromise;
+
+ if (loader.isDestroyed()) {
+ return;
+ }
+
+ loader._gltf = undefined; // No longer need to hold onto the glTF
+
+ loader._state = ResourceLoaderState.LOADED;
+ return loader;
+ } catch (error) {
+ if (loader.isDestroyed()) {
+ return;
+ }
+
+ loader.unload();
+ loader._state = ResourceLoaderState.FAILED;
+ const errorMessage = "Failed to load GPM data";
+ throw loader.getError(errorMessage, error);
+ }
+}
+
+/**
+ * Loads the resource.
+ * @returns {Promise} A promise which resolves to the loader when the resource loading is completed.
+ * @private
+ */
+GltfMeshPrimitiveGpmLoader.prototype.load = function () {
+ if (defined(this._promise)) {
+ return this._promise;
+ }
+
+ this._state = ResourceLoaderState.LOADING;
+ this._promise = loadResources(this);
+ return this._promise;
+};
+
+function gatherUsedTextureIds(gpmExtension) {
+ // Gather the used textures
+ const textureIds = {};
+ const ppeTextures = gpmExtension.ppeTextures;
+ if (defined(ppeTextures)) {
+ for (let i = 0; i < ppeTextures.length; i++) {
+ const ppeTexture = ppeTextures[i];
+ // The texture is a valid textureInfo.
+ textureIds[ppeTexture.index] = ppeTexture;
+ }
+ }
+ return textureIds;
+}
+
+function loadTextures(meshPrimitiveGpmLoader) {
+ let textureIds;
+ if (defined(meshPrimitiveGpmLoader._extension)) {
+ textureIds = gatherUsedTextureIds(meshPrimitiveGpmLoader._extension);
+ }
+
+ const gltf = meshPrimitiveGpmLoader._gltf;
+ const gltfResource = meshPrimitiveGpmLoader._gltfResource;
+ const baseResource = meshPrimitiveGpmLoader._baseResource;
+ const supportedImageFormats = meshPrimitiveGpmLoader._supportedImageFormats;
+ const frameState = meshPrimitiveGpmLoader._frameState;
+ const asynchronous = meshPrimitiveGpmLoader._asynchronous;
+
+ // Load the textures
+ const texturePromises = [];
+ for (const textureId in textureIds) {
+ if (textureIds.hasOwnProperty(textureId)) {
+ const textureLoader = ResourceCache.getTextureLoader({
+ gltf: gltf,
+ textureInfo: textureIds[textureId],
+ gltfResource: gltfResource,
+ baseResource: baseResource,
+ supportedImageFormats: supportedImageFormats,
+ frameState: frameState,
+ asynchronous: asynchronous,
+ });
+ meshPrimitiveGpmLoader._textureLoaders.push(textureLoader);
+ meshPrimitiveGpmLoader._textureIds.push(textureId);
+ texturePromises.push(textureLoader.load());
+ }
+ }
+
+ return Promise.all(texturePromises);
+}
+
+/**
+ * A static mapping from PPE texture property identifier keys
+ * to `MetadataSchema` instances. This is used to create each
+ * schema (with a certain structure) only ONCE in
+ * obtainPpeTexturesMetadataSchema
+ *
+ * @private
+ */
+GltfMeshPrimitiveGpmLoader.ppeTexturesMetadataSchemaCache = new Map();
+
+/**
+ * Create the JSON description of a metadata class that treats
+ * the given PPE texture as a property texture property(!).
+ *
+ * @param {any} ppeTexture - The PPE texture
+ * @param {number} index - The index of the texture in the extension
+ * @returns The class JSON
+ */
+GltfMeshPrimitiveGpmLoader.prototype.createPpeTextureClassJson = function (
+ ppeTexture,
+ index
+) {
+ const traits = ppeTexture.traits;
+ const ppePropertyName = traits.source;
+
+ // XXX_UNCERTAINTY: The ppeTexture will have a structure
+ // like this:
+ //
+ //"ppeTextures" : [
+ // {
+ // "traits" : {
+ // "source" : "SIGZ",
+ // "min" : 0.0,
+ // "max" : 16.0
+ // },
+ // "index" : 2,
+ // "noData" : 255,
+ // "offset" : 0.0,
+ // "scale" : 0.06274509803921569,
+ // "texCoord" : 1
+ // },
+ //
+ // Note that in GPM 1.2d, the min/max should be part of the
+ // texture, but in the actual data (GPM 1.2i), they are in
+ // the traits (ppeMetadata). And there, they/ seem to denote
+ // the min/max values that actually appear in the texture.
+ // So they are integrated into the 'scale' factor here:
+ const min = traits.min ?? 0;
+ const max = traits.max ?? 255;
+ const minMaxScale = 255.0 / (max - min);
+ const offset = ppeTexture.offset;
+ const scale = ppeTexture.scale * minMaxScale;
+ const classJson = {
+ name: `PPE texture class ${index}`,
+ properties: {
+ [ppePropertyName]: {
+ name: "PPE",
+ type: "SCALAR",
+ componentType: "UINT8",
+ normalized: true,
+ offset: offset,
+ scale: scale,
+ },
+ },
+ };
+ return classJson;
+};
+
+/**
+ * Returns the `MetadataSchema` for the PPE textures in this instance.
+ *
+ * This method will return a (statically/globally) cached metadata
+ * schema that reflects the structure of the PPE textures in this
+ * instance, creating and caching it if necessary.
+ *
+ * For details on the cache key, see `collectPpeTexturePropertyIdentifiers`
+ *
+ * @returns The `MetadataSchema`
+ */
+GltfMeshPrimitiveGpmLoader.prototype.obtainPpeTexturesMetadataSchema = function () {
+ const ppeTexturePropertyIdentifiers = this.collectPpeTexturePropertyIdentifiers();
+ const key = ppeTexturePropertyIdentifiers.toString();
+ let ppeTexturesMetadataSchema = GltfMeshPrimitiveGpmLoader.ppeTexturesMetadataSchemaCache.get(
+ key
+ );
+ if (defined(ppeTexturesMetadataSchema)) {
+ // XXX_UNCERTAINTY Debug log
+ //console.log(`Using cached schema for GPM PPE textures with key ${key}`);
+ return ppeTexturesMetadataSchema;
+ }
+
+ // XXX_UNCERTAINTY Debug log - if caching works, this should be printed only ONCE!
+ console.log(`Creating schema for GPM PPE textures with key ${key}`);
+ const schemaId = `PPE_TEXTURE_SCHEMA_${GltfMeshPrimitiveGpmLoader.ppeTexturesMetadataSchemaCache.size}`;
+ const ppeTexturesMetadataSchemaJson = {
+ id: schemaId,
+ classes: {},
+ };
+
+ const extension = this._extension;
+ if (defined(extension.ppeTextures)) {
+ for (let i = 0; i < extension.ppeTextures.length; i++) {
+ const ppeTexture = extension.ppeTextures[i];
+ const classId = `ppeTexture_${i}`;
+ const classJson = this.createPpeTextureClassJson(ppeTexture, i);
+ ppeTexturesMetadataSchemaJson.classes[classId] = classJson;
+ }
+ }
+
+ ppeTexturesMetadataSchema = MetadataSchema.fromJson(
+ ppeTexturesMetadataSchemaJson
+ );
+ GltfMeshPrimitiveGpmLoader.ppeTexturesMetadataSchemaCache.set(
+ key,
+ ppeTexturesMetadataSchema
+ );
+ return ppeTexturesMetadataSchema;
+};
+
+/**
+ * Creates an array of strings that serve as identifiers for PPE textures.
+ *
+ * Each glTF may define multiple `ppeTexture` objects within the
+ * `NGA_gpm_local` extensions. Each of these textures corresponds
+ * to one 'property texture property(!)' in a metadata schema.
+ *
+ * This method will create an array where each element is a (JSON)
+ * string representation of the parts of a GPM PPE texture definition
+ * that are relevant for distinguishing two PPE textures in terms
+ * of their structure within a `StructuralMetadata`.
+ *
+ * @returns The identifiers
+ */
+GltfMeshPrimitiveGpmLoader.prototype.collectPpeTexturePropertyIdentifiers = function () {
+ const extension = this._extension;
+ const ppeTexturePropertyIdentifiers = [];
+ if (defined(extension.ppeTextures)) {
+ for (let i = 0; i < extension.ppeTextures.length; i++) {
+ const ppeTexture = extension.ppeTextures[i];
+ // The following will create an identifier that can be used
+ // to define two PPE textures as "representing the same
+ // property texture property" within a structural metadata
+ // schema.
+ const classJson = this.createPpeTextureClassJson(ppeTexture, i);
+ const ppeTexturePropertyIdentifier = JSON.stringify(classJson);
+ ppeTexturePropertyIdentifiers.push(ppeTexturePropertyIdentifier);
+ }
+ }
+ return ppeTexturePropertyIdentifiers;
+};
+
+/**
+ * Processes the resource until it becomes ready.
+ *
+ * @param {FrameState} frameState The frame state.
+ * @private
+ */
+GltfMeshPrimitiveGpmLoader.prototype.process = function (frameState) {
+ //>>includeStart('debug', pragmas.debug);
+ Check.typeOf.object("frameState", frameState);
+ //>>includeEnd('debug');
+
+ if (this._state === ResourceLoaderState.READY) {
+ return true;
+ }
+
+ if (this._state !== ResourceLoaderState.LOADED) {
+ return false;
+ }
+
+ const textureLoaders = this._textureLoaders;
+ const textureLoadersLength = textureLoaders.length;
+ let ready = true;
+ for (let i = 0; i < textureLoadersLength; ++i) {
+ const textureLoader = textureLoaders[i];
+ const textureReady = textureLoader.process(frameState);
+ ready = ready && textureReady;
+ }
+
+ if (!ready) {
+ return false;
+ }
+
+ const textures = {};
+ for (let i = 0; i < this._textureIds.length; ++i) {
+ const textureId = this._textureIds[i];
+ const textureLoader = textureLoaders[i];
+ if (!textureLoader.isDestroyed()) {
+ textures[textureId] = textureLoader.texture;
+ }
+ }
+
+ const ppeTexturesMetadataSchema = this.obtainPpeTexturesMetadataSchema();
+ const extension = this._extension;
+ const propertyTextures = [];
+ if (defined(extension.ppeTextures)) {
+ for (let i = 0; i < extension.ppeTextures.length; i++) {
+ const ppeTexture = extension.ppeTextures[i];
+ const classId = `ppeTexture_${i}`;
+ const traits = ppeTexture.traits;
+ const ppePropertyName = traits.source;
+ const metadataClass = ppeTexturesMetadataSchema.classes[classId];
+
+ // XXX_UNCERTAINTY Debug log
+ //console.log(
+ // `Creating property texture with class ${classId} and property ${ppePropertyName}`
+ //);
+
+ const ppeTextureAsPropertyTexture = {
+ class: classId,
+ properties: {
+ [ppePropertyName]: {
+ index: ppeTexture.index,
+ texCoord: ppeTexture.texCoord,
+ },
+ },
+ };
+ propertyTextures.push(
+ new PropertyTexture({
+ id: i,
+ name: ppeTexture.name,
+ propertyTexture: ppeTextureAsPropertyTexture,
+ class: metadataClass,
+ textures: textures,
+ })
+ );
+ }
+ }
+
+ const structuralMetadata = new StructuralMetadata({
+ schema: ppeTexturesMetadataSchema,
+ propertyTables: [],
+ propertyTextures: propertyTextures,
+ propertyAttributes: [],
+ statistics: extension.statistics,
+ extras: extension.extras,
+ extensions: extension.extensions,
+ });
+ this._structuralMetadata = structuralMetadata;
+
+ this._state = ResourceLoaderState.READY;
+ return true;
+};
+
+function unloadTextures(meshPrimitiveGpmLoader) {
+ const textureLoaders = meshPrimitiveGpmLoader._textureLoaders;
+ const textureLoadersLength = textureLoaders.length;
+ for (let i = 0; i < textureLoadersLength; ++i) {
+ ResourceCache.unload(textureLoaders[i]);
+ }
+ meshPrimitiveGpmLoader._textureLoaders.length = 0;
+ meshPrimitiveGpmLoader._textureIds.length = 0;
+}
+
+/**
+ * Unloads the resource.
+ * @private
+ */
+GltfMeshPrimitiveGpmLoader.prototype.unload = function () {
+ unloadTextures(this);
+ this._structuralMetadata = undefined;
+};
+
+export default GltfMeshPrimitiveGpmLoader;
diff --git a/packages/engine/Source/Scene/Model/Model.js b/packages/engine/Source/Scene/Model/Model.js
index e5415a11587c..7d481d2088d1 100644
--- a/packages/engine/Source/Scene/Model/Model.js
+++ b/packages/engine/Source/Scene/Model/Model.js
@@ -1772,6 +1772,24 @@ Model.prototype.applyArticulations = function () {
this._sceneGraph.applyArticulations();
};
+/**
+ * XXX_TODO_COMMENT
+ *
+ * @exception {DeveloperError} The model is not loaded. Use Model.readyEvent or wait for Model.ready to be true.
+ */
+Model.prototype.getExtension = function (extensionName) {
+ //>>includeStart('debug', pragmas.debug);
+ Check.typeOf.string("extensionName", extensionName);
+ if (!this._ready) {
+ throw new DeveloperError(
+ "The model is not loaded. Use Model.readyEvent or wait for Model.ready to be true."
+ );
+ }
+ //>>includeEnd('debug');
+ const components = this._loader.components;
+ return components.extensions[extensionName];
+};
+
/**
* Marks the model's {@link Model#style} as dirty, which forces all features
* to re-evaluate the style in the next frame the model is visible.
diff --git a/packages/engine/Source/Scene/Model/Model3DTileContent.js b/packages/engine/Source/Scene/Model/Model3DTileContent.js
index a32206d3dba6..f2240619e744 100644
--- a/packages/engine/Source/Scene/Model/Model3DTileContent.js
+++ b/packages/engine/Source/Scene/Model/Model3DTileContent.js
@@ -152,6 +152,12 @@ Object.defineProperties(Model3DTileContent.prototype, {
},
});
+Model3DTileContent.prototype.getExtension = function (extensionName) {
+ const model = this._model;
+ const extension = model.getExtension(extensionName);
+ return extension;
+};
+
Model3DTileContent.prototype.getFeature = function (featureId) {
const model = this._model;
const featureTableId = model.featureTableId;
@@ -468,9 +474,11 @@ Model3DTileContent.prototype.pick = function (ray, frameState, result) {
};
function makeModelOptions(tileset, tile, content, additionalOptions) {
+ console.log("XXX KEEPING glTF in Model3DTileContent");
const mainOptions = {
cull: false, // The model is already culled by 3D Tiles
- releaseGltfJson: true, // Models are unique and will not benefit from caching so save memory
+ //releaseGltfJson: true, // Models are unique and will not benefit from caching so save memory
+ releaseGltfJson: false,
opaquePass: Pass.CESIUM_3D_TILE, // Draw opaque portions of the model during the 3D Tiles pass
modelMatrix: tile.computedTransform,
upAxis: tileset._modelUpAxis,
diff --git a/packages/engine/Source/Scene/ModelComponents.js b/packages/engine/Source/Scene/ModelComponents.js
index 42b3834c6db7..428e410dd8ac 100644
--- a/packages/engine/Source/Scene/ModelComponents.js
+++ b/packages/engine/Source/Scene/ModelComponents.js
@@ -1141,6 +1141,12 @@ function Components() {
* @private
*/
this.transform = Matrix4.clone(Matrix4.IDENTITY);
+
+ /**
+ * A mapping from extension names like `"EXT_example_extension"` to
+ * the object that was created from the extension input
+ */
+ this.extensions = {};
}
/**
From 7e1479105782cf4b385acdbe4b877abe1aa1d14b Mon Sep 17 00:00:00 2001
From: Marco Hutter
Date: Thu, 19 Sep 2024 13:49:18 +0200
Subject: [PATCH 04/38] Move loaders to extension directory
---
packages/engine/Source/Scene/GltfLoader.js | 4 +--
.../Extensions/Gpm}/GltfGpmLoader.js | 26 +++++++++----------
.../Gpm}/GltfMeshPrimitiveGpmLoader.js | 18 ++++++-------
3 files changed, 24 insertions(+), 24 deletions(-)
rename packages/engine/Source/Scene/{ => Model/Extensions/Gpm}/GltfGpmLoader.js (90%)
rename packages/engine/Source/Scene/{ => Model/Extensions/Gpm}/GltfMeshPrimitiveGpmLoader.js (96%)
diff --git a/packages/engine/Source/Scene/GltfLoader.js b/packages/engine/Source/Scene/GltfLoader.js
index 490cbfb087af..bb25a498d4d1 100644
--- a/packages/engine/Source/Scene/GltfLoader.js
+++ b/packages/engine/Source/Scene/GltfLoader.js
@@ -29,8 +29,8 @@ import ResourceCache from "./ResourceCache.js";
import ResourceLoader from "./ResourceLoader.js";
import SupportedImageFormats from "./SupportedImageFormats.js";
import VertexAttributeSemantic from "./VertexAttributeSemantic.js";
-import GltfGpmLoader from "./GltfGpmLoader.js";
-import GltfMeshPrimitiveGpmLoader from "./GltfMeshPrimitiveGpmLoader.js";
+import GltfGpmLoader from "./Model/Extensions/Gpm/GltfGpmLoader.js";
+import GltfMeshPrimitiveGpmLoader from "./Model/Extensions/Gpm/GltfMeshPrimitiveGpmLoader.js";
const {
Attribute,
diff --git a/packages/engine/Source/Scene/GltfGpmLoader.js b/packages/engine/Source/Scene/Model/Extensions/Gpm/GltfGpmLoader.js
similarity index 90%
rename from packages/engine/Source/Scene/GltfGpmLoader.js
rename to packages/engine/Source/Scene/Model/Extensions/Gpm/GltfGpmLoader.js
index 4fe1cf50c46b..520a96b44769 100644
--- a/packages/engine/Source/Scene/GltfGpmLoader.js
+++ b/packages/engine/Source/Scene/Model/Extensions/Gpm/GltfGpmLoader.js
@@ -1,16 +1,16 @@
-import Cartesian3 from "../Core/Cartesian3.js";
-import Check from "../Core/Check.js";
-import defaultValue from "../Core/defaultValue.js";
-import defined from "../Core/defined.js";
-import Matrix3 from "../Core/Matrix3.js";
-import AnchorPointDirect from "./Model/Extensions/Gpm/AnchorPointDirect.js";
-import AnchorPointIndirect from "./Model/Extensions/Gpm/AnchorPointIndirect.js";
-import CorrelationGroup from "./Model/Extensions/Gpm/CorrelationGroup.js";
-import GltfGpmLocal from "./Model/Extensions/Gpm/GltfGpmLocal.js";
-import Spdcf from "./Model/Extensions/Gpm/Spdcf.js";
-import StorageType from "./Model/Extensions/Gpm/StorageType.js";
-import ResourceLoader from "./ResourceLoader.js";
-import ResourceLoaderState from "./ResourceLoaderState.js";
+import Cartesian3 from "../../../../Core/Cartesian3.js";
+import Check from "../../../../Core/Check.js";
+import defaultValue from "../../../../Core/defaultValue.js";
+import defined from "../../../../Core/defined.js";
+import Matrix3 from "../../../../Core/Matrix3.js";
+import ResourceLoader from "./../../../ResourceLoader.js";
+import ResourceLoaderState from "./../../../ResourceLoaderState.js";
+import AnchorPointDirect from "./AnchorPointDirect.js";
+import AnchorPointIndirect from "./AnchorPointIndirect.js";
+import CorrelationGroup from "./CorrelationGroup.js";
+import GltfGpmLocal from "./GltfGpmLocal.js";
+import Spdcf from "./Spdcf.js";
+import StorageType from "./StorageType.js";
/**
* Loads glTF NGA_gpm_local from the root of a glTF object
diff --git a/packages/engine/Source/Scene/GltfMeshPrimitiveGpmLoader.js b/packages/engine/Source/Scene/Model/Extensions/Gpm/GltfMeshPrimitiveGpmLoader.js
similarity index 96%
rename from packages/engine/Source/Scene/GltfMeshPrimitiveGpmLoader.js
rename to packages/engine/Source/Scene/Model/Extensions/Gpm/GltfMeshPrimitiveGpmLoader.js
index 724b1d8b1a47..0668183ea0f7 100644
--- a/packages/engine/Source/Scene/GltfMeshPrimitiveGpmLoader.js
+++ b/packages/engine/Source/Scene/Model/Extensions/Gpm/GltfMeshPrimitiveGpmLoader.js
@@ -1,12 +1,12 @@
-import Check from "../Core/Check.js";
-import defaultValue from "../Core/defaultValue.js";
-import defined from "../Core/defined.js";
-import ResourceCache from "./ResourceCache.js";
-import ResourceLoader from "./ResourceLoader.js";
-import ResourceLoaderState from "./ResourceLoaderState.js";
-import PropertyTexture from "./PropertyTexture.js";
-import StructuralMetadata from "./StructuralMetadata.js";
-import MetadataSchema from "./MetadataSchema.js";
+import Check from "../../../../Core/Check.js";
+import defaultValue from "../../../../Core/defaultValue.js";
+import defined from "../../../../Core/defined.js";
+import ResourceCache from "../../../ResourceCache.js";
+import ResourceLoader from "../../../ResourceLoader.js";
+import ResourceLoaderState from "../../../ResourceLoaderState.js";
+import PropertyTexture from "../../../PropertyTexture.js";
+import StructuralMetadata from "../../../StructuralMetadata.js";
+import MetadataSchema from "../../../MetadataSchema.js";
/**
* Loads glTF NGA_gpm_local from a glTF mesh primitive.
From 6ee73444dbe75acd6a9b7e1450af023c4a297ec8 Mon Sep 17 00:00:00 2001
From: Marco Hutter
Date: Thu, 19 Sep 2024 15:16:48 +0200
Subject: [PATCH 05/38] Added implementation note in GltfGpmLoader
---
.../Source/Scene/Model/Extensions/Gpm/GltfGpmLoader.js | 7 +++++++
1 file changed, 7 insertions(+)
diff --git a/packages/engine/Source/Scene/Model/Extensions/Gpm/GltfGpmLoader.js b/packages/engine/Source/Scene/Model/Extensions/Gpm/GltfGpmLoader.js
index 520a96b44769..3091184142a3 100644
--- a/packages/engine/Source/Scene/Model/Extensions/Gpm/GltfGpmLoader.js
+++ b/packages/engine/Source/Scene/Model/Extensions/Gpm/GltfGpmLoader.js
@@ -19,6 +19,13 @@ import StorageType from "./StorageType.js";
*
* Implementation note: This is an experimental implementation.
*
+ * TODO This implements ResourceLoader, even though it does not really
+ * load other resources (in contrast to GltfMeshPrimitiveGpmLoader,
+ * which loads textures). Parsing the JSON into a GltfGpmLocal
+ * could be done directly (synchronously in the GltfLoader class).
+ * But it should be carved out into a dedicated class, regardless
+ * of how this is eventually USED from the GltfLoader.
+ *
* @alias GltfGpmLoader
* @constructor
* @augments ResourceLoader
From c2a064c9bb2ebd7ab610f683a7dadc511fd7ae61 Mon Sep 17 00:00:00 2001
From: Marco Hutter
Date: Thu, 19 Sep 2024 15:17:18 +0200
Subject: [PATCH 06/38] Fix handling of min, max, offset, scale for PPE
textures
---
.../Gpm/GltfMeshPrimitiveGpmLoader.js | 34 +++++++++++--------
1 file changed, 20 insertions(+), 14 deletions(-)
diff --git a/packages/engine/Source/Scene/Model/Extensions/Gpm/GltfMeshPrimitiveGpmLoader.js b/packages/engine/Source/Scene/Model/Extensions/Gpm/GltfMeshPrimitiveGpmLoader.js
index 0668183ea0f7..c97ae55d7c61 100644
--- a/packages/engine/Source/Scene/Model/Extensions/Gpm/GltfMeshPrimitiveGpmLoader.js
+++ b/packages/engine/Source/Scene/Model/Extensions/Gpm/GltfMeshPrimitiveGpmLoader.js
@@ -224,8 +224,7 @@ GltfMeshPrimitiveGpmLoader.prototype.createPpeTextureClassJson = function (
const traits = ppeTexture.traits;
const ppePropertyName = traits.source;
- // XXX_UNCERTAINTY: The ppeTexture will have a structure
- // like this:
+ // The ppeTexture will have a structure like this:
//
//"ppeTextures" : [
// {
@@ -241,16 +240,16 @@ GltfMeshPrimitiveGpmLoader.prototype.createPpeTextureClassJson = function (
// "texCoord" : 1
// },
//
- // Note that in GPM 1.2d, the min/max should be part of the
- // texture, but in the actual data (GPM 1.2i), they are in
- // the traits (ppeMetadata). And there, they/ seem to denote
- // the min/max values that actually appear in the texture.
- // So they are integrated into the 'scale' factor here:
- const min = traits.min ?? 0;
- const max = traits.max ?? 255;
- const minMaxScale = 255.0 / (max - min);
- const offset = ppeTexture.offset;
- const scale = ppeTexture.scale * minMaxScale;
+ // This is translated into a single class property here, that defines
+ // the structure of the property texture property.
+ //
+ // Given that `offset` and `scale` may only be applied to integer
+ // property values when they are `normalized`, the values will be
+ // declared as `normalized` here.
+ // The and the normalization factor will later have to be
+ // cancelled out, when integrating the `scale` into the
+ // actual property texture property. In the property texture
+ // property, the `scale` has to be multiplied by 255.
const classJson = {
name: `PPE texture class ${index}`,
properties: {
@@ -259,8 +258,8 @@ GltfMeshPrimitiveGpmLoader.prototype.createPpeTextureClassJson = function (
type: "SCALAR",
componentType: "UINT8",
normalized: true,
- offset: offset,
- scale: scale,
+ min: traits.min,
+ max: traits.max,
},
},
};
@@ -407,12 +406,19 @@ GltfMeshPrimitiveGpmLoader.prototype.process = function (frameState) {
// `Creating property texture with class ${classId} and property ${ppePropertyName}`
//);
+ // The class property has been declared as `normalized`, so
+ // that `offset` and `scale` can be applied. The normalization
+ // factor has to be cancelled out here, by multiplying the
+ // `scale` with 255.
+ const scale = ppeTexture.scale * 255.0;
const ppeTextureAsPropertyTexture = {
class: classId,
properties: {
[ppePropertyName]: {
index: ppeTexture.index,
texCoord: ppeTexture.texCoord,
+ offset: ppeTexture.offset,
+ scale: scale,
},
},
};
From 50b072702973c8ca5b2dc39476509331b44db5e1 Mon Sep 17 00:00:00 2001
From: Marco Hutter
Date: Sat, 21 Sep 2024 17:27:19 +0200
Subject: [PATCH 07/38] Enable releaseGltfJson
---
packages/engine/Source/Scene/Model/Model3DTileContent.js | 4 +---
1 file changed, 1 insertion(+), 3 deletions(-)
diff --git a/packages/engine/Source/Scene/Model/Model3DTileContent.js b/packages/engine/Source/Scene/Model/Model3DTileContent.js
index f2240619e744..1a1b8a1aceaf 100644
--- a/packages/engine/Source/Scene/Model/Model3DTileContent.js
+++ b/packages/engine/Source/Scene/Model/Model3DTileContent.js
@@ -474,11 +474,9 @@ Model3DTileContent.prototype.pick = function (ray, frameState, result) {
};
function makeModelOptions(tileset, tile, content, additionalOptions) {
- console.log("XXX KEEPING glTF in Model3DTileContent");
const mainOptions = {
cull: false, // The model is already culled by 3D Tiles
- //releaseGltfJson: true, // Models are unique and will not benefit from caching so save memory
- releaseGltfJson: false,
+ releaseGltfJson: true, // Models are unique and will not benefit from caching so save memory
opaquePass: Pass.CESIUM_3D_TILE, // Draw opaque portions of the model during the 3D Tiles pass
modelMatrix: tile.computedTransform,
upAxis: tileset._modelUpAxis,
From cac091f95a14e28d762c38b2848873309f5b7ef7 Mon Sep 17 00:00:00 2001
From: Marco Hutter
Date: Sat, 21 Sep 2024 17:27:54 +0200
Subject: [PATCH 08/38] Let GltfGpmLoader not extend ResourceLoader
---
packages/engine/Source/Scene/GltfLoader.js | 36 +-
.../Model/Extensions/Gpm/GltfGpmLoader.js | 372 ++++++++----------
2 files changed, 167 insertions(+), 241 deletions(-)
diff --git a/packages/engine/Source/Scene/GltfLoader.js b/packages/engine/Source/Scene/GltfLoader.js
index bb25a498d4d1..d7abc6d3fee0 100644
--- a/packages/engine/Source/Scene/GltfLoader.js
+++ b/packages/engine/Source/Scene/GltfLoader.js
@@ -262,7 +262,6 @@ function GltfLoader(options) {
this._geometryLoaders = [];
this._geometryCallbacks = [];
this._structuralMetadataLoader = undefined;
- this._gpmLoader = undefined;
this._meshPrimitiveGpmLoader = undefined;
this._loadResourcesPromise = undefined;
this._resourcesLoaded = false;
@@ -475,17 +474,6 @@ function processLoaders(loader, frameState) {
ready = ready && metadataReady;
}
- const gpmLoader = loader._gpmLoader;
- if (defined(gpmLoader)) {
- const gpmReady = gpmLoader.process(frameState);
- if (gpmReady) {
- // XXX
- console.log("Got GPM data, storing ", gpmLoader._gltfGpmLocal);
- loader._components.extensions["NGA_gpm_local"] = gpmLoader._gltfGpmLocal;
- }
- ready = ready && gpmReady;
- }
-
const meshPrimitiveGpmLoader = loader._meshPrimitiveGpmLoader;
if (defined(meshPrimitiveGpmLoader)) {
const metadataReady = meshPrimitiveGpmLoader.process(frameState);
@@ -2480,18 +2468,6 @@ async function loadStructuralMetadata(
return structuralMetadataLoader.load();
}
-async function loadGpm(loader, gltf, extension) {
- const gpmLoader = new GltfGpmLoader({
- gltf: gltf,
- extension: extension,
- gltfResource: loader._gltfResource,
- baseResource: loader._baseResource,
- asynchronous: loader._asynchronous,
- });
- loader._gpmLoader = gpmLoader;
- return gpmLoader.load();
-}
-
async function loadMeshPrimitiveGpm(loader, gltf, extension, frameState) {
const meshPrimitiveGpmLoader = new GltfMeshPrimitiveGpmLoader({
gltf: gltf,
@@ -2738,8 +2714,8 @@ function parse(loader, frameState) {
const gpmExtension = extensions.NGA_gpm_local;
if (defined(gpmExtension)) {
console.log("Loading GPM from root");
- const promise = loadGpm(loader, gltf, gpmExtension);
- loader._loaderPromises.push(promise);
+ const gltfGpmLocal = GltfGpmLoader.load(gpmExtension);
+ loader._components.extensions["NGA_gpm_local"] = gltfGpmLocal;
}
// Load NGA_gpm_local from mesh primitives
@@ -2831,13 +2807,6 @@ function unloadStructuralMetadata(loader) {
}
}
-function unloadGpm(loader) {
- if (defined(loader._gpmLoader) && !loader._gpmLoader.isDestroyed()) {
- loader._gpmLoader.destroy();
- loader._gpmLoader = undefined;
- }
-}
-
function unloadMeshPrimitiveGpm(loader) {
if (
defined(loader._meshPrimitiveGpmLoader) &&
@@ -2871,7 +2840,6 @@ GltfLoader.prototype.unload = function () {
unloadGeometry(this);
unloadGeneratedAttributes(this);
unloadStructuralMetadata(this);
- unloadGpm(this);
unloadMeshPrimitiveGpm(this);
this._components = undefined;
diff --git a/packages/engine/Source/Scene/Model/Extensions/Gpm/GltfGpmLoader.js b/packages/engine/Source/Scene/Model/Extensions/Gpm/GltfGpmLoader.js
index 3091184142a3..c3a1a0c0bbb9 100644
--- a/packages/engine/Source/Scene/Model/Extensions/Gpm/GltfGpmLoader.js
+++ b/packages/engine/Source/Scene/Model/Extensions/Gpm/GltfGpmLoader.js
@@ -1,10 +1,7 @@
import Cartesian3 from "../../../../Core/Cartesian3.js";
import Check from "../../../../Core/Check.js";
-import defaultValue from "../../../../Core/defaultValue.js";
-import defined from "../../../../Core/defined.js";
+import DeveloperError from "../../../../Core/DeveloperError.js";
import Matrix3 from "../../../../Core/Matrix3.js";
-import ResourceLoader from "./../../../ResourceLoader.js";
-import ResourceLoaderState from "./../../../ResourceLoaderState.js";
import AnchorPointDirect from "./AnchorPointDirect.js";
import AnchorPointIndirect from "./AnchorPointIndirect.js";
import CorrelationGroup from "./CorrelationGroup.js";
@@ -15,110 +12,19 @@ import StorageType from "./StorageType.js";
/**
* Loads glTF NGA_gpm_local from the root of a glTF object
*
- * Implements the {@link ResourceLoader} interface.
- *
* Implementation note: This is an experimental implementation.
*
- * TODO This implements ResourceLoader, even though it does not really
- * load other resources (in contrast to GltfMeshPrimitiveGpmLoader,
- * which loads textures). Parsing the JSON into a GltfGpmLocal
- * could be done directly (synchronously in the GltfLoader class).
- * But it should be carved out into a dedicated class, regardless
- * of how this is eventually USED from the GltfLoader.
- *
* @alias GltfGpmLoader
* @constructor
- * @augments ResourceLoader
*
* @param {object} options Object with the following properties:
* @param {object} options.gltf The glTF JSON.
* @param {string} [options.extension] The NGA_gpm_local
extension object.
- * @param {Resource} options.gltfResource The {@link Resource} containing the glTF.
- * @param {Resource} options.baseResource The {@link Resource} that paths in the glTF JSON are relative to.
- * @param {string} [options.cacheKey] The cache key of the resource.
- * @param {boolean} [options.asynchronous=true] Determines if WebGL resource creation will be spread out over several frames or block until all WebGL resources are created.
*
* @private
* @experimental This feature is subject to change without Cesium's standard deprecation policy.
*/
-function GltfGpmLoader(options) {
- options = defaultValue(options, defaultValue.EMPTY_OBJECT);
- const gltf = options.gltf;
- const extension = options.extension;
- const gltfResource = options.gltfResource;
- const baseResource = options.baseResource;
- const cacheKey = options.cacheKey;
- const asynchronous = defaultValue(options.asynchronous, true);
-
- //>>includeStart('debug', pragmas.debug);
- Check.typeOf.object("options.gltf", gltf);
- Check.typeOf.object("options.extension", extension);
- Check.typeOf.object("options.gltfResource", gltfResource);
- Check.typeOf.object("options.baseResource", baseResource);
- //>>includeEnd('debug');
-
- this._gltfResource = gltfResource;
- this._baseResource = baseResource;
- this._gltf = gltf;
- this._extension = extension;
- this._cacheKey = cacheKey;
- this._asynchronous = asynchronous;
- this._gltfGpmLocal = undefined;
-
- // Immediately go into the "LOADED" state, since there
- // are no resources to wait for
- this._state = ResourceLoaderState.LOADED;
- this._promise = undefined;
-}
-
-if (defined(Object.create)) {
- GltfGpmLoader.prototype = Object.create(ResourceLoader.prototype);
- GltfGpmLoader.prototype.constructor = GltfGpmLoader;
-}
-
-Object.defineProperties(GltfGpmLoader.prototype, {
- /**
- * The cache key of the resource.
- *
- * @memberof GltfGpmLoader.prototype
- *
- * @type {string}
- * @readonly
- * @private
- */
- cacheKey: {
- get: function () {
- return this._cacheKey;
- },
- },
- /**
- * The parsed GltfGpmLocal object
- *
- * @memberof GltfGpmLoader.prototype
- *
- * @type {GltfGpmLocal}
- * @readonly
- * @private
- */
- gltfGpmLocal: {
- get: function () {
- return this._gltfGpmLocal;
- },
- },
-});
-
-/**
- * Loads the resource.
- * @returns {Promise} A promise which resolves to the loader when the resource loading is completed.
- * @private
- */
-GltfGpmLoader.prototype.load = function () {
- if (defined(this._promise)) {
- return this._promise;
- }
- this._promise = Promise.resolve(this);
- return this._promise;
-};
+function GltfGpmLoader() {}
/**
* Creates a Matrix3 that describes a covariance matrix (which is
@@ -144,135 +50,187 @@ function createCovarianceMatrixFromUpperTriangle(array) {
}
/**
- * Processes the resource until it becomes ready.
+ * Creates an `AnchorPointDirect` from the given JSON representation
*
- * @param {FrameState} frameState The frame state.
+ * @param {object} anchorPointDirectJson The input JSON
+ * @returns The `AnchorPointDirect`
+ */
+function createAnchorPointDirect(anchorPointDirectJson) {
+ const position = Cartesian3.fromArray(
+ anchorPointDirectJson.position,
+ 0,
+ new Cartesian3()
+ );
+ const adjustmentParams = Cartesian3.fromArray(
+ anchorPointDirectJson.adjustmentParams,
+ 0,
+ new Cartesian3()
+ );
+ const anchorPointDirect = new AnchorPointDirect({
+ position: position,
+ adjustmentParams: adjustmentParams,
+ });
+ return anchorPointDirect;
+}
+
+/**
+ * Creates an `AnchorPointIndirect` from the given JSON representation
+ *
+ * @param {object} anchorPointIndirectJson The input JSON
+ * @returns The `AnchorPointIndirect`
+ */
+function createAnchorPointIndirect(anchorPointIndirectJson) {
+ const position = Cartesian3.fromArray(
+ anchorPointIndirectJson.position,
+ 0,
+ new Cartesian3()
+ );
+ const adjustmentParams = Cartesian3.fromArray(
+ anchorPointIndirectJson.adjustmentParams,
+ 0,
+ new Cartesian3()
+ );
+ const covarianceMatrix = createCovarianceMatrixFromUpperTriangle(
+ anchorPointIndirectJson.covarianceMatrix
+ );
+ const anchorPointIndirect = new AnchorPointIndirect({
+ position: position,
+ adjustmentParams: adjustmentParams,
+ covarianceMatrix: covarianceMatrix,
+ });
+ return anchorPointIndirect;
+}
+
+/**
+ * Creates a `CorrelationGroup` from the given JSON representation
+ *
+ * @param {object} correlationGroupJson The input JSON
+ * @returns The `CorrelationGroup`
+ */
+function createCorrelationGroup(correlationGroupJson) {
+ const groupFlags = correlationGroupJson.groupFlags;
+ const rotationThetas = Cartesian3.fromArray(
+ correlationGroupJson.rotationThetas,
+ 0,
+ new Cartesian3()
+ );
+ const params = [];
+ for (const paramJson of correlationGroupJson.params) {
+ const param = new Spdcf({
+ A: paramJson.A,
+ alpha: paramJson.alpha,
+ beta: paramJson.beta,
+ T: paramJson.T,
+ });
+ params.push(param);
+ }
+ const correlationGroup = new CorrelationGroup({
+ groupFlags: groupFlags,
+ rotationThetas: rotationThetas,
+ params: params,
+ });
+ return correlationGroup;
+}
+
+/**
+ * Loads the GPM data from the given JSON that was found as the
+ * `NGA_gpm_local` extension object in the root of the glTF.
+ *
+ * @param {object} gltfGpmLocalJson The extension object
+ * @returns {GltfGpmLocal} The parsed object
* @private
+ * @throws DeveloperError When the given object is `undefined`
+ * or contains invalid structures
*/
-GltfGpmLoader.prototype.process = function (frameState) {
+GltfGpmLoader.load = function (gltfGpmLocalJson) {
//>>includeStart('debug', pragmas.debug);
- Check.typeOf.object("frameState", frameState);
+ Check.typeOf.object("gltfGpmLocalJson", gltfGpmLocalJson);
//>>includeEnd('debug');
- if (this._state === ResourceLoaderState.READY) {
- return true;
+ const storageType = gltfGpmLocalJson.storageType;
+ if (storageType === StorageType.Direct) {
+ return GltfGpmLoader.loadDirect(gltfGpmLocalJson);
}
- if (this._state === ResourceLoaderState.FAILED) {
- return true;
+ if (storageType === StorageType.Indirect) {
+ return GltfGpmLoader.loadIndirect(gltfGpmLocalJson);
}
- if (this._state !== ResourceLoaderState.LOADED) {
- return false;
+ throw new DeveloperError(
+ `Invalid storage type in NGA_gpm_local - expected 'Direct' or 'Indirect', but found ${storageType}`
+ );
+};
+
+/**
+ * Loads the GPM data from the given JSON that was found as the
+ * `NGA_gpm_local` extension object in the root of the glTF,
+ * assuming that the `storageType` of the given object is
+ * `StorageType.Direct`.
+ *
+ * @param {object} gltfGpmLocalJson The extension object
+ * @returns {GltfGpmLocal} The parsed object
+ * @private
+ */
+GltfGpmLoader.loadDirect = function (gltfGpmLocalJson) {
+ //>>includeStart('debug', pragmas.debug);
+ Check.typeOf.object("gltfGpmLocalJson", gltfGpmLocalJson);
+ //>>includeEnd('debug');
+
+ const anchorPointsDirect = [];
+ const anchorPointsDirectJson = gltfGpmLocalJson.anchorPointsDirect;
+ for (const anchorPointDirectJson of anchorPointsDirectJson) {
+ const anchorPointDirect = createAnchorPointDirect(anchorPointDirectJson);
+ anchorPointsDirect.push(anchorPointDirect);
}
+ const covarianceDirect = createCovarianceMatrixFromUpperTriangle(
+ gltfGpmLocalJson.covarianceDirectUpperTriangle
+ );
- console.log("PARSE AND STORE HERE! ", this._extension);
+ const gltfGpmLocal = new GltfGpmLocal({
+ storageType: StorageType.Direct,
+ anchorPointsDirect: anchorPointsDirect,
+ covarianceDirect: covarianceDirect,
+ });
+ return gltfGpmLocal;
+};
- const extensionJson = this._extension;
+/**
+ * Loads the GPM data from the given JSON that was found as the
+ * `NGA_gpm_local` extension object in the root of the glTF,
+ * assuming that the `storageType` of the given object is
+ * `StorageType.Indirect`.
+ *
+ * @param {object} gltfGpmLocalJson The extension object
+ * @returns {GltfGpmLocal} The parsed object
+ * @private
+ */
+GltfGpmLoader.loadIndirect = function (gltfGpmLocalJson) {
+ //>>includeStart('debug', pragmas.debug);
+ Check.typeOf.object("gltfGpmLocalJson", gltfGpmLocalJson);
+ //>>includeEnd('debug');
- const storageType = extensionJson.storageType;
- if (storageType === StorageType.Direct) {
- const anchorPointsDirect = [];
- const anchorPointsDirectJson = extensionJson.anchorPointsDirect;
- for (const anchorPointDirectJson of anchorPointsDirectJson) {
- const position = Cartesian3.fromArray(
- anchorPointDirectJson.position,
- 0,
- new Cartesian3()
- );
- const adjustmentParams = Cartesian3.fromArray(
- anchorPointDirectJson.adjustmentParams,
- 0,
- new Cartesian3()
- );
- const anchorPointDirect = new AnchorPointDirect({
- position: position,
- adjustmentParams: adjustmentParams,
- });
- anchorPointsDirect.push(anchorPointDirect);
- }
- const covarianceDirect = createCovarianceMatrixFromUpperTriangle(
- extensionJson.covarianceDirectUpperTriangle
+ const anchorPointsIndirect = [];
+ const anchorPointsIndirectJson = gltfGpmLocalJson.anchorPointsIndirect;
+ for (const anchorPointIndirectJson of anchorPointsIndirectJson) {
+ const anchorPointIndirect = createAnchorPointIndirect(
+ anchorPointIndirectJson
);
-
- this._gltfGpmLocal = new GltfGpmLocal({
- storageType: storageType,
- anchorPointsDirect: anchorPointsDirect,
- covarianceDirect: covarianceDirect,
- });
- this._state = ResourceLoaderState.READY;
- return true;
+ anchorPointsIndirect.push(anchorPointIndirect);
}
- if (storageType === StorageType.Indirect) {
- const anchorPointsIndirect = [];
- const anchorPointsIndirectJson = extensionJson.anchorPointsIndirect;
- for (const anchorPointIndirectJson of anchorPointsIndirectJson) {
- const position = Cartesian3.fromArray(
- anchorPointIndirectJson.position,
- 0,
- new Cartesian3()
- );
- const adjustmentParams = Cartesian3.fromArray(
- anchorPointIndirectJson.adjustmentParams,
- 0,
- new Cartesian3()
- );
- const covarianceMatrix = createCovarianceMatrixFromUpperTriangle(
- anchorPointIndirectJson.covarianceMatrix
- );
- const anchorPointIndirect = new AnchorPointIndirect({
- position: position,
- adjustmentParams: adjustmentParams,
- covarianceMatrix: covarianceMatrix,
- });
- anchorPointsIndirect.push(anchorPointIndirect);
- }
- const intraTileCorrelationGroupsJson =
- extensionJson.intraTileCorrelationGroups;
- const intraTileCorrelationGroups = [];
+ const intraTileCorrelationGroupsJson =
+ gltfGpmLocalJson.intraTileCorrelationGroups;
+ const intraTileCorrelationGroups = [];
- for (const correlationGroupJson of intraTileCorrelationGroupsJson) {
- const groupFlags = correlationGroupJson.groupFlags;
- const rotationThetas = Cartesian3.fromArray(
- correlationGroupJson.rotationThetas,
- 0,
- new Cartesian3()
- );
- const params = [];
- for (const paramJson of correlationGroupJson.params) {
- const param = new Spdcf({
- A: paramJson.A,
- alpha: paramJson.alpha,
- beta: paramJson.beta,
- T: paramJson.T,
- });
- params.push(param);
- }
- const correlationGroup = new CorrelationGroup({
- groupFlags: groupFlags,
- rotationThetas: rotationThetas,
- params: params,
- });
- intraTileCorrelationGroups.push(correlationGroup);
- }
-
- this._gltfGpmLocal = new GltfGpmLocal({
- storageType: storageType,
- anchorPointsIndirect: anchorPointsIndirect,
- intraTileCorrelationGroups: intraTileCorrelationGroups,
- });
- this._state = ResourceLoaderState.READY;
- return true;
+ for (const correlationGroupJson of intraTileCorrelationGroupsJson) {
+ const correlationGroup = createCorrelationGroup(correlationGroupJson);
+ intraTileCorrelationGroups.push(correlationGroup);
}
- this._state = ResourceLoaderState.FAILED;
- return false;
-};
-/**
- * Unloads the resource.
- * @private
- */
-GltfGpmLoader.prototype.unload = function () {
- this._gltfGpmLocal = undefined;
+ const gltfGpmLocal = new GltfGpmLocal({
+ storageType: StorageType.Indirect,
+ anchorPointsIndirect: anchorPointsIndirect,
+ intraTileCorrelationGroups: intraTileCorrelationGroups,
+ });
+ return gltfGpmLocal;
};
export default GltfGpmLoader;
From 1cc38faedcf3772c804589a05150182d43a2fee1 Mon Sep 17 00:00:00 2001
From: Marco Hutter
Date: Sat, 21 Sep 2024 17:28:06 +0200
Subject: [PATCH 09/38] Minor cleanup and comments
---
.../Extensions/Gpm/GltfMeshPrimitiveGpmLoader.js | 12 +++++++-----
1 file changed, 7 insertions(+), 5 deletions(-)
diff --git a/packages/engine/Source/Scene/Model/Extensions/Gpm/GltfMeshPrimitiveGpmLoader.js b/packages/engine/Source/Scene/Model/Extensions/Gpm/GltfMeshPrimitiveGpmLoader.js
index c97ae55d7c61..485caae1f808 100644
--- a/packages/engine/Source/Scene/Model/Extensions/Gpm/GltfMeshPrimitiveGpmLoader.js
+++ b/packages/engine/Source/Scene/Model/Extensions/Gpm/GltfMeshPrimitiveGpmLoader.js
@@ -246,10 +246,10 @@ GltfMeshPrimitiveGpmLoader.prototype.createPpeTextureClassJson = function (
// Given that `offset` and `scale` may only be applied to integer
// property values when they are `normalized`, the values will be
// declared as `normalized` here.
- // The and the normalization factor will later have to be
- // cancelled out, when integrating the `scale` into the
- // actual property texture property. In the property texture
- // property, the `scale` has to be multiplied by 255.
+ // The normalization factor will later have to be cancelled out,
+ // when integrating the `scale` into the actual property texture
+ // property. In the property texture property, the `scale` has to
+ // be multiplied by 255.
const classJson = {
name: `PPE texture class ${index}`,
properties: {
@@ -410,7 +410,7 @@ GltfMeshPrimitiveGpmLoader.prototype.process = function (frameState) {
// that `offset` and `scale` can be applied. The normalization
// factor has to be cancelled out here, by multiplying the
// `scale` with 255.
- const scale = ppeTexture.scale * 255.0;
+ const scale = (ppeTexture.scale ?? 1.0) * 255.0;
const ppeTextureAsPropertyTexture = {
class: classId,
properties: {
@@ -465,6 +465,8 @@ function unloadTextures(meshPrimitiveGpmLoader) {
*/
GltfMeshPrimitiveGpmLoader.prototype.unload = function () {
unloadTextures(this);
+ this._gltf = undefined;
+ this._extension = undefined;
this._structuralMetadata = undefined;
};
From 4e6d501af04672d5d339765049236877574a45fa Mon Sep 17 00:00:00 2001
From: Marco Hutter
Date: Sat, 21 Sep 2024 18:06:28 +0200
Subject: [PATCH 10/38] Fix type checks
---
.../engine/Source/Scene/Model/Extensions/Gpm/PpeMetadata.js | 2 +-
packages/engine/Source/Scene/Model/Extensions/Gpm/PpeTexture.js | 2 +-
2 files changed, 2 insertions(+), 2 deletions(-)
diff --git a/packages/engine/Source/Scene/Model/Extensions/Gpm/PpeMetadata.js b/packages/engine/Source/Scene/Model/Extensions/Gpm/PpeMetadata.js
index de255f5b6dd9..66d38ef40cb8 100644
--- a/packages/engine/Source/Scene/Model/Extensions/Gpm/PpeMetadata.js
+++ b/packages/engine/Source/Scene/Model/Extensions/Gpm/PpeMetadata.js
@@ -8,7 +8,7 @@ import Check from "../../../../Core/Check.js";
*/
function PpeMetadata(options) {
//>>includeStart('debug', pragmas.debug);
- Check.typeOf.object("options.source", options.source);
+ Check.typeOf.string("options.source", options.source);
//>>includeEnd('debug');
this._min = options.min;
diff --git a/packages/engine/Source/Scene/Model/Extensions/Gpm/PpeTexture.js b/packages/engine/Source/Scene/Model/Extensions/Gpm/PpeTexture.js
index 5214a6e34ac6..8b352956758d 100644
--- a/packages/engine/Source/Scene/Model/Extensions/Gpm/PpeTexture.js
+++ b/packages/engine/Source/Scene/Model/Extensions/Gpm/PpeTexture.js
@@ -13,7 +13,7 @@ import Check from "../../../../Core/Check.js";
function PpeTexture(options) {
//>>includeStart('debug', pragmas.debug);
Check.typeOf.object("options.traits", options.traits);
- Check.typeOf.object("options.index", options.index);
+ Check.typeOf.number.greaterThanOrEquals("options.index", options.index, 0);
//>>includeEnd('debug');
this._traits = options.traits;
From 9e7bfcf0aee24f1d84b23ac57fba20ab1a23ad28 Mon Sep 17 00:00:00 2001
From: Marco Hutter
Date: Sat, 21 Sep 2024 18:06:50 +0200
Subject: [PATCH 11/38] Explicit conversion of PPE into structural metadata
---
.../Gpm/GltfMeshPrimitiveGpmLoader.js | 288 +++++++++++-------
1 file changed, 177 insertions(+), 111 deletions(-)
diff --git a/packages/engine/Source/Scene/Model/Extensions/Gpm/GltfMeshPrimitiveGpmLoader.js b/packages/engine/Source/Scene/Model/Extensions/Gpm/GltfMeshPrimitiveGpmLoader.js
index 485caae1f808..e8d52cb72997 100644
--- a/packages/engine/Source/Scene/Model/Extensions/Gpm/GltfMeshPrimitiveGpmLoader.js
+++ b/packages/engine/Source/Scene/Model/Extensions/Gpm/GltfMeshPrimitiveGpmLoader.js
@@ -7,18 +7,20 @@ import ResourceLoaderState from "../../../ResourceLoaderState.js";
import PropertyTexture from "../../../PropertyTexture.js";
import StructuralMetadata from "../../../StructuralMetadata.js";
import MetadataSchema from "../../../MetadataSchema.js";
+import PpeTexture from "./PpeTexture.js";
+import PpeMetadata from "./PpeMetadata.js";
+import MeshPrimitiveGpmLocal from "./MeshPrimitiveGpmLocal.js";
/**
* Loads glTF NGA_gpm_local from a glTF mesh primitive.
*
* Implements the {@link ResourceLoader} interface.
*
- * Implementation note: This is an experimental implementation. It is based
- * on a GltfStructuralMetadataLoader, by removing everything that is not
- * related to property textures, and translating the "ppeTextures" of
- * the NGA_gpm_local extension into property texture properties. These will
- * be returned as part of a `StructuralMetadata` object, which may override
- * any `StructuralMetadata` that was read directly from the glTF.
+ * This loads the "ppeTextures" of the NGA_gpm_local extension of a mesh primitive
+ * and stores them in a `MeshPrimitiveGpmLocal` object.
+ *
+ * This object will be converted into a `StructuralMetadata` object, which may
+ * override any `StructuralMetadata` that was read directly from the glTF.
*
* @alias GltfMeshPrimitiveGpmLoader
* @constructor
@@ -67,6 +69,7 @@ function GltfMeshPrimitiveGpmLoader(options) {
this._asynchronous = asynchronous;
this._textureLoaders = [];
this._textureIds = [];
+ this._meshPrimitiveGpmLocal = undefined;
this._structuralMetadata = undefined;
this._state = ResourceLoaderState.UNLOADED;
this._promise = undefined;
@@ -110,30 +113,30 @@ Object.defineProperties(GltfMeshPrimitiveGpmLoader.prototype, {
},
});
-async function loadResources(loader) {
+GltfMeshPrimitiveGpmLoader.prototype.loadResources = async function () {
try {
- const texturesPromise = loadTextures(loader);
+ const texturesPromise = this.loadTextures();
await texturesPromise;
- if (loader.isDestroyed()) {
+ if (this.isDestroyed()) {
return;
}
- loader._gltf = undefined; // No longer need to hold onto the glTF
+ this._gltf = undefined; // No longer need to hold onto the glTF
- loader._state = ResourceLoaderState.LOADED;
- return loader;
+ this._state = ResourceLoaderState.LOADED;
+ return this;
} catch (error) {
- if (loader.isDestroyed()) {
+ if (this.isDestroyed()) {
return;
}
- loader.unload();
- loader._state = ResourceLoaderState.FAILED;
+ this.unload();
+ this._state = ResourceLoaderState.FAILED;
const errorMessage = "Failed to load GPM data";
- throw loader.getError(errorMessage, error);
+ throw this.getError(errorMessage, error);
}
-}
+};
/**
* Loads the resource.
@@ -146,7 +149,7 @@ GltfMeshPrimitiveGpmLoader.prototype.load = function () {
}
this._state = ResourceLoaderState.LOADING;
- this._promise = loadResources(this);
+ this._promise = this.loadResources(this);
return this._promise;
};
@@ -164,18 +167,18 @@ function gatherUsedTextureIds(gpmExtension) {
return textureIds;
}
-function loadTextures(meshPrimitiveGpmLoader) {
+GltfMeshPrimitiveGpmLoader.prototype.loadTextures = function () {
let textureIds;
- if (defined(meshPrimitiveGpmLoader._extension)) {
- textureIds = gatherUsedTextureIds(meshPrimitiveGpmLoader._extension);
+ if (defined(this._extension)) {
+ textureIds = gatherUsedTextureIds(this._extension);
}
- const gltf = meshPrimitiveGpmLoader._gltf;
- const gltfResource = meshPrimitiveGpmLoader._gltfResource;
- const baseResource = meshPrimitiveGpmLoader._baseResource;
- const supportedImageFormats = meshPrimitiveGpmLoader._supportedImageFormats;
- const frameState = meshPrimitiveGpmLoader._frameState;
- const asynchronous = meshPrimitiveGpmLoader._asynchronous;
+ const gltf = this._gltf;
+ const gltfResource = this._gltfResource;
+ const baseResource = this._baseResource;
+ const supportedImageFormats = this._supportedImageFormats;
+ const frameState = this._frameState;
+ const asynchronous = this._asynchronous;
// Load the textures
const texturePromises = [];
@@ -190,14 +193,14 @@ function loadTextures(meshPrimitiveGpmLoader) {
frameState: frameState,
asynchronous: asynchronous,
});
- meshPrimitiveGpmLoader._textureLoaders.push(textureLoader);
- meshPrimitiveGpmLoader._textureIds.push(textureId);
+ this._textureLoaders.push(textureLoader);
+ this._textureIds.push(textureId);
texturePromises.push(textureLoader.load());
}
}
return Promise.all(texturePromises);
-}
+};
/**
* A static mapping from PPE texture property identifier keys
@@ -217,7 +220,7 @@ GltfMeshPrimitiveGpmLoader.ppeTexturesMetadataSchemaCache = new Map();
* @param {number} index - The index of the texture in the extension
* @returns The class JSON
*/
-GltfMeshPrimitiveGpmLoader.prototype.createPpeTextureClassJson = function (
+GltfMeshPrimitiveGpmLoader.createPpeTextureClassJson = function (
ppeTexture,
index
) {
@@ -267,18 +270,24 @@ GltfMeshPrimitiveGpmLoader.prototype.createPpeTextureClassJson = function (
};
/**
- * Returns the `MetadataSchema` for the PPE textures in this instance.
+ * Returns the `MetadataSchema` for the PPE textures in the given
+ * `MeshPrimitiveGpmLocal` instance.
*
* This method will return a (statically/globally) cached metadata
- * schema that reflects the structure of the PPE textures in this
- * instance, creating and caching it if necessary.
+ * schema that reflects the structure of the PPE textures in the
+ * given instance, creating and caching it if necessary.
*
* For details on the cache key, see `collectPpeTexturePropertyIdentifiers`
*
+ * @param {MeshPrimitiveGpmLocal} meshPrimitiveGpmLocal The extension object
* @returns The `MetadataSchema`
*/
-GltfMeshPrimitiveGpmLoader.prototype.obtainPpeTexturesMetadataSchema = function () {
- const ppeTexturePropertyIdentifiers = this.collectPpeTexturePropertyIdentifiers();
+GltfMeshPrimitiveGpmLoader.obtainPpeTexturesMetadataSchema = function (
+ meshPrimitiveGpmLocal
+) {
+ const ppeTexturePropertyIdentifiers = GltfMeshPrimitiveGpmLoader.collectPpeTexturePropertyIdentifiers(
+ meshPrimitiveGpmLocal
+ );
const key = ppeTexturePropertyIdentifiers.toString();
let ppeTexturesMetadataSchema = GltfMeshPrimitiveGpmLoader.ppeTexturesMetadataSchemaCache.get(
key
@@ -297,14 +306,15 @@ GltfMeshPrimitiveGpmLoader.prototype.obtainPpeTexturesMetadataSchema = function
classes: {},
};
- const extension = this._extension;
- if (defined(extension.ppeTextures)) {
- for (let i = 0; i < extension.ppeTextures.length; i++) {
- const ppeTexture = extension.ppeTextures[i];
- const classId = `ppeTexture_${i}`;
- const classJson = this.createPpeTextureClassJson(ppeTexture, i);
- ppeTexturesMetadataSchemaJson.classes[classId] = classJson;
- }
+ const ppeTextures = meshPrimitiveGpmLocal.ppeTextures;
+ for (let i = 0; i < ppeTextures.length; i++) {
+ const ppeTexture = ppeTextures[i];
+ const classId = `ppeTexture_${i}`;
+ const classJson = GltfMeshPrimitiveGpmLoader.createPpeTextureClassJson(
+ ppeTexture,
+ i
+ );
+ ppeTexturesMetadataSchemaJson.classes[classId] = classJson;
}
ppeTexturesMetadataSchema = MetadataSchema.fromJson(
@@ -329,26 +339,98 @@ GltfMeshPrimitiveGpmLoader.prototype.obtainPpeTexturesMetadataSchema = function
* that are relevant for distinguishing two PPE textures in terms
* of their structure within a `StructuralMetadata`.
*
+ * @param {MeshPrimitiveGpmLocal} meshPrimitiveGpmLocal The extension object
* @returns The identifiers
*/
-GltfMeshPrimitiveGpmLoader.prototype.collectPpeTexturePropertyIdentifiers = function () {
- const extension = this._extension;
+GltfMeshPrimitiveGpmLoader.collectPpeTexturePropertyIdentifiers = function (
+ meshPrimitiveGpmLocal
+) {
const ppeTexturePropertyIdentifiers = [];
- if (defined(extension.ppeTextures)) {
- for (let i = 0; i < extension.ppeTextures.length; i++) {
- const ppeTexture = extension.ppeTextures[i];
- // The following will create an identifier that can be used
- // to define two PPE textures as "representing the same
- // property texture property" within a structural metadata
- // schema.
- const classJson = this.createPpeTextureClassJson(ppeTexture, i);
- const ppeTexturePropertyIdentifier = JSON.stringify(classJson);
- ppeTexturePropertyIdentifiers.push(ppeTexturePropertyIdentifier);
- }
+ const ppeTextures = meshPrimitiveGpmLocal.ppeTextures;
+ for (let i = 0; i < ppeTextures.length; i++) {
+ const ppeTexture = ppeTextures[i];
+ // The following will create an identifier that can be used
+ // to define two PPE textures as "representing the same
+ // property texture property" within a structural metadata
+ // schema.
+ const classJson = GltfMeshPrimitiveGpmLoader.createPpeTextureClassJson(
+ ppeTexture,
+ i
+ );
+ const ppeTexturePropertyIdentifier = JSON.stringify(classJson);
+ ppeTexturePropertyIdentifiers.push(ppeTexturePropertyIdentifier);
}
return ppeTexturePropertyIdentifiers;
};
+/**
+ * Converts the given `MeshPrimitiveGpmLocal` object into a `StructuralMetadata`
+ * object.
+ *
+ * This will translate the PPE textures from the given object into property
+ * texture properties. The schema will be created based on the the structure
+ * of the PPE textures.
+ *
+ * @param {MeshPrimitiveGpmLocal} meshPrimitiveGpmLocal The extension object
+ * @param {object} textures The mapping from texture ID to texture objects
+ * @returns The `StructuralMetadata` object
+ */
+GltfMeshPrimitiveGpmLoader.convertToStructuralMetadata = function (
+ meshPrimitiveGpmLocal,
+ textures
+) {
+ const propertyTextures = [];
+ const ppeTexturesMetadataSchema = GltfMeshPrimitiveGpmLoader.obtainPpeTexturesMetadataSchema(
+ meshPrimitiveGpmLocal
+ );
+ const ppeTextures = meshPrimitiveGpmLocal.ppeTextures;
+ for (let i = 0; i < ppeTextures.length; i++) {
+ const ppeTexture = ppeTextures[i];
+ const classId = `ppeTexture_${i}`;
+ const traits = ppeTexture.traits;
+ const ppePropertyName = traits.source;
+ const metadataClass = ppeTexturesMetadataSchema.classes[classId];
+
+ // XXX_UNCERTAINTY Debug log
+ //console.log(
+ // `Creating property texture with class ${classId} and property ${ppePropertyName}`
+ //);
+
+ // The class property has been declared as `normalized`, so
+ // that `offset` and `scale` can be applied. The normalization
+ // factor has to be cancelled out here, by multiplying the
+ // `scale` with 255.
+ const scale = (ppeTexture.scale ?? 1.0) * 255.0;
+ const ppeTextureAsPropertyTexture = {
+ class: classId,
+ properties: {
+ [ppePropertyName]: {
+ index: ppeTexture.index,
+ texCoord: ppeTexture.texCoord,
+ offset: ppeTexture.offset,
+ scale: scale,
+ },
+ },
+ };
+ propertyTextures.push(
+ new PropertyTexture({
+ id: i,
+ name: ppeTexture.name,
+ propertyTexture: ppeTextureAsPropertyTexture,
+ class: metadataClass,
+ textures: textures,
+ })
+ );
+ }
+ const structuralMetadata = new StructuralMetadata({
+ schema: ppeTexturesMetadataSchema,
+ propertyTables: [],
+ propertyTextures: propertyTextures,
+ propertyAttributes: [],
+ });
+ return structuralMetadata;
+};
+
/**
* Processes the resource until it becomes ready.
*
@@ -368,6 +450,8 @@ GltfMeshPrimitiveGpmLoader.prototype.process = function (frameState) {
return false;
}
+ // The standard process of loading textures
+ // (from GltfStructuralMetadataLoader)
const textureLoaders = this._textureLoaders;
const textureLoadersLength = textureLoaders.length;
let ready = true;
@@ -381,6 +465,8 @@ GltfMeshPrimitiveGpmLoader.prototype.process = function (frameState) {
return false;
}
+ // More of the standard process of loading textures
+ // (from GltfStructuralMetadataLoader)
const textures = {};
for (let i = 0; i < this._textureIds.length; ++i) {
const textureId = this._textureIds[i];
@@ -390,81 +476,61 @@ GltfMeshPrimitiveGpmLoader.prototype.process = function (frameState) {
}
}
- const ppeTexturesMetadataSchema = this.obtainPpeTexturesMetadataSchema();
+ // Convert the JSON representation of the `ppeTextures` that
+ // are found in the extensjon JSON into `PpeTexture` objects
+ const ppeTextures = [];
const extension = this._extension;
- const propertyTextures = [];
if (defined(extension.ppeTextures)) {
- for (let i = 0; i < extension.ppeTextures.length; i++) {
- const ppeTexture = extension.ppeTextures[i];
- const classId = `ppeTexture_${i}`;
- const traits = ppeTexture.traits;
- const ppePropertyName = traits.source;
- const metadataClass = ppeTexturesMetadataSchema.classes[classId];
-
- // XXX_UNCERTAINTY Debug log
- //console.log(
- // `Creating property texture with class ${classId} and property ${ppePropertyName}`
- //);
-
- // The class property has been declared as `normalized`, so
- // that `offset` and `scale` can be applied. The normalization
- // factor has to be cancelled out here, by multiplying the
- // `scale` with 255.
- const scale = (ppeTexture.scale ?? 1.0) * 255.0;
- const ppeTextureAsPropertyTexture = {
- class: classId,
- properties: {
- [ppePropertyName]: {
- index: ppeTexture.index,
- texCoord: ppeTexture.texCoord,
- offset: ppeTexture.offset,
- scale: scale,
- },
- },
- };
- propertyTextures.push(
- new PropertyTexture({
- id: i,
- name: ppeTexture.name,
- propertyTexture: ppeTextureAsPropertyTexture,
- class: metadataClass,
- textures: textures,
- })
- );
+ const ppeTexturesJson = extension.ppeTextures;
+ for (const ppeTextureJson of ppeTexturesJson) {
+ const traitsJson = ppeTextureJson.traits;
+ const traits = new PpeMetadata({
+ min: traitsJson.min,
+ max: traitsJson.max,
+ source: traitsJson.source,
+ });
+ const ppeTexture = new PpeTexture({
+ traits: traits,
+ noData: ppeTextureJson.noData,
+ offset: ppeTextureJson.offset,
+ scale: ppeTextureJson.scale,
+ index: ppeTextureJson.index,
+ texCoord: ppeTextureJson.texCoord,
+ });
+ ppeTextures.push(ppeTexture);
}
}
-
- const structuralMetadata = new StructuralMetadata({
- schema: ppeTexturesMetadataSchema,
- propertyTables: [],
- propertyTextures: propertyTextures,
- propertyAttributes: [],
- statistics: extension.statistics,
- extras: extension.extras,
- extensions: extension.extensions,
+ const meshPrimitiveGpmLocal = new MeshPrimitiveGpmLocal({
+ ppeTextures: ppeTextures,
});
+ this._meshPrimitiveGpmLocal = meshPrimitiveGpmLocal;
+
+ const structuralMetadata = GltfMeshPrimitiveGpmLoader.convertToStructuralMetadata(
+ meshPrimitiveGpmLocal,
+ textures
+ );
this._structuralMetadata = structuralMetadata;
this._state = ResourceLoaderState.READY;
return true;
};
-function unloadTextures(meshPrimitiveGpmLoader) {
- const textureLoaders = meshPrimitiveGpmLoader._textureLoaders;
+GltfMeshPrimitiveGpmLoader.prototype.unloadTextures = function () {
+ const textureLoaders = this._textureLoaders;
const textureLoadersLength = textureLoaders.length;
for (let i = 0; i < textureLoadersLength; ++i) {
ResourceCache.unload(textureLoaders[i]);
}
- meshPrimitiveGpmLoader._textureLoaders.length = 0;
- meshPrimitiveGpmLoader._textureIds.length = 0;
-}
+ this._textureLoaders.length = 0;
+ this._textureIds.length = 0;
+};
/**
* Unloads the resource.
* @private
*/
GltfMeshPrimitiveGpmLoader.prototype.unload = function () {
- unloadTextures(this);
+ this.unloadTextures();
this._gltf = undefined;
this._extension = undefined;
this._structuralMetadata = undefined;
From 5110166947f0cd46c7be4fdb37fb9a006bdfd95f Mon Sep 17 00:00:00 2001
From: Marco Hutter
Date: Tue, 24 Sep 2024 22:36:46 +0200
Subject: [PATCH 12/38] Remove debug logs
---
packages/engine/Source/Scene/GltfLoader.js | 11 +++++++----
.../Extensions/Gpm/GltfMeshPrimitiveGpmLoader.js | 9 ---------
2 files changed, 7 insertions(+), 13 deletions(-)
diff --git a/packages/engine/Source/Scene/GltfLoader.js b/packages/engine/Source/Scene/GltfLoader.js
index d7abc6d3fee0..c6733177bdd0 100644
--- a/packages/engine/Source/Scene/GltfLoader.js
+++ b/packages/engine/Source/Scene/GltfLoader.js
@@ -31,6 +31,7 @@ import SupportedImageFormats from "./SupportedImageFormats.js";
import VertexAttributeSemantic from "./VertexAttributeSemantic.js";
import GltfGpmLoader from "./Model/Extensions/Gpm/GltfGpmLoader.js";
import GltfMeshPrimitiveGpmLoader from "./Model/Extensions/Gpm/GltfMeshPrimitiveGpmLoader.js";
+import oneTimeWarning from "../Core/oneTimeWarning.js";
const {
Attribute,
@@ -478,8 +479,12 @@ function processLoaders(loader, frameState) {
if (defined(meshPrimitiveGpmLoader)) {
const metadataReady = meshPrimitiveGpmLoader.process(frameState);
if (metadataReady) {
- // XXX
- console.log("Overwriting structural metadata with GPM data");
+ if (defined(loader._components.structuralMetadata)) {
+ oneTimeWarning(
+ "structural-metadata-gpm",
+ "Structural metadata is replaced with GPM data"
+ );
+ }
loader._components.structuralMetadata =
meshPrimitiveGpmLoader.structuralMetadata;
}
@@ -2728,8 +2733,6 @@ function parse(loader, frameState) {
const primitiveExtensions = primitive.extensions;
const meshPrimitiveGpmExtension = primitiveExtensions.NGA_gpm_local;
if (defined(meshPrimitiveGpmExtension)) {
- // XXX
- console.log("Loading GPM from mesh primitive");
const promise = loadMeshPrimitiveGpm(
loader,
gltf,
diff --git a/packages/engine/Source/Scene/Model/Extensions/Gpm/GltfMeshPrimitiveGpmLoader.js b/packages/engine/Source/Scene/Model/Extensions/Gpm/GltfMeshPrimitiveGpmLoader.js
index e8d52cb72997..c207a962a70c 100644
--- a/packages/engine/Source/Scene/Model/Extensions/Gpm/GltfMeshPrimitiveGpmLoader.js
+++ b/packages/engine/Source/Scene/Model/Extensions/Gpm/GltfMeshPrimitiveGpmLoader.js
@@ -293,13 +293,9 @@ GltfMeshPrimitiveGpmLoader.obtainPpeTexturesMetadataSchema = function (
key
);
if (defined(ppeTexturesMetadataSchema)) {
- // XXX_UNCERTAINTY Debug log
- //console.log(`Using cached schema for GPM PPE textures with key ${key}`);
return ppeTexturesMetadataSchema;
}
- // XXX_UNCERTAINTY Debug log - if caching works, this should be printed only ONCE!
- console.log(`Creating schema for GPM PPE textures with key ${key}`);
const schemaId = `PPE_TEXTURE_SCHEMA_${GltfMeshPrimitiveGpmLoader.ppeTexturesMetadataSchemaCache.size}`;
const ppeTexturesMetadataSchemaJson = {
id: schemaId,
@@ -391,11 +387,6 @@ GltfMeshPrimitiveGpmLoader.convertToStructuralMetadata = function (
const ppePropertyName = traits.source;
const metadataClass = ppeTexturesMetadataSchema.classes[classId];
- // XXX_UNCERTAINTY Debug log
- //console.log(
- // `Creating property texture with class ${classId} and property ${ppePropertyName}`
- //);
-
// The class property has been declared as `normalized`, so
// that `offset` and `scale` can be applied. The normalization
// factor has to be cancelled out here, by multiplying the
From 323f4f01afff34610875ea433c5705668ec1061e Mon Sep 17 00:00:00 2001
From: Marco Hutter
Date: Tue, 24 Sep 2024 22:36:59 +0200
Subject: [PATCH 13/38] Additional type checks
---
.../Scene/Model/Extensions/Gpm/GltfGpmLoader.js | 16 ++++++++++++++++
1 file changed, 16 insertions(+)
diff --git a/packages/engine/Source/Scene/Model/Extensions/Gpm/GltfGpmLoader.js b/packages/engine/Source/Scene/Model/Extensions/Gpm/GltfGpmLoader.js
index c3a1a0c0bbb9..730aca143994 100644
--- a/packages/engine/Source/Scene/Model/Extensions/Gpm/GltfGpmLoader.js
+++ b/packages/engine/Source/Scene/Model/Extensions/Gpm/GltfGpmLoader.js
@@ -172,6 +172,14 @@ GltfGpmLoader.load = function (gltfGpmLocalJson) {
GltfGpmLoader.loadDirect = function (gltfGpmLocalJson) {
//>>includeStart('debug', pragmas.debug);
Check.typeOf.object("gltfGpmLocalJson", gltfGpmLocalJson);
+ Check.typeOf.object(
+ "gltfGpmLocalJson.anchorPointsDirect",
+ gltfGpmLocalJson.anchorPointsDirect
+ );
+ Check.typeOf.object(
+ "gltfGpmLocalJson.covarianceDirectUpperTriangle",
+ gltfGpmLocalJson.covarianceDirectUpperTriangle
+ );
//>>includeEnd('debug');
const anchorPointsDirect = [];
@@ -205,6 +213,14 @@ GltfGpmLoader.loadDirect = function (gltfGpmLocalJson) {
GltfGpmLoader.loadIndirect = function (gltfGpmLocalJson) {
//>>includeStart('debug', pragmas.debug);
Check.typeOf.object("gltfGpmLocalJson", gltfGpmLocalJson);
+ Check.typeOf.object(
+ "gltfGpmLocalJson.anchorPointsIndirect",
+ gltfGpmLocalJson.anchorPointsIndirect
+ );
+ Check.typeOf.object(
+ "gltfGpmLocalJson.intraTileCorrelationGroups",
+ gltfGpmLocalJson.intraTileCorrelationGroups
+ );
//>>includeEnd('debug');
const anchorPointsIndirect = [];
From 6326d40261ce405f5f6c4fe6a0bd901f1d3b988f Mon Sep 17 00:00:00 2001
From: Marco Hutter
Date: Tue, 24 Sep 2024 22:37:18 +0200
Subject: [PATCH 14/38] Added JSDoc comment
---
packages/engine/Source/Scene/Model/Model.js | 13 ++++++++++++-
1 file changed, 12 insertions(+), 1 deletion(-)
diff --git a/packages/engine/Source/Scene/Model/Model.js b/packages/engine/Source/Scene/Model/Model.js
index 7d481d2088d1..c68109d60937 100644
--- a/packages/engine/Source/Scene/Model/Model.js
+++ b/packages/engine/Source/Scene/Model/Model.js
@@ -1773,9 +1773,20 @@ Model.prototype.applyArticulations = function () {
};
/**
- * XXX_TODO_COMMENT
+ * Returns the object that was created for the given extension.
*
+ * The given name may be the name of a glTF extension, like `"EXT_example_extension"`.
+ * If the specified extension was present in the root of the underlying glTF asset,
+ * and a loder for the specified extension has processed the extension data, then
+ * this will return the model representation of the extension.
+ *
+ * @param {string} extensionName The name of the extension
+ * @returns The object, or `undefined`
* @exception {DeveloperError} The model is not loaded. Use Model.readyEvent or wait for Model.ready to be true.
+ *
+ * @private
+ * @experimental This feature is subject to change without Cesium's standard deprecation policy.
+ *
*/
Model.prototype.getExtension = function (extensionName) {
//>>includeStart('debug', pragmas.debug);
From eaf6112903870dc7553275d9ec9574a7b65303e4 Mon Sep 17 00:00:00 2001
From: Marco Hutter
Date: Tue, 24 Sep 2024 22:37:56 +0200
Subject: [PATCH 15/38] First part of GPM loading specs
---
.../Model/Extensions/Gpm/GltfGpmLoaderSpec.js | 201 ++++++++++++++++++
1 file changed, 201 insertions(+)
create mode 100644 packages/engine/Specs/Scene/Model/Extensions/Gpm/GltfGpmLoaderSpec.js
diff --git a/packages/engine/Specs/Scene/Model/Extensions/Gpm/GltfGpmLoaderSpec.js b/packages/engine/Specs/Scene/Model/Extensions/Gpm/GltfGpmLoaderSpec.js
new file mode 100644
index 000000000000..c8920fb2831b
--- /dev/null
+++ b/packages/engine/Specs/Scene/Model/Extensions/Gpm/GltfGpmLoaderSpec.js
@@ -0,0 +1,201 @@
+import Cartesian3 from "../../../../../Source/Core/Cartesian3.js";
+import CesiumMath from "../../../../../Source/Core/Math.js";
+import Matrix3 from "../../../../../Source/Core/Matrix3.js";
+import GltfGpmLoader from "../../../../../Source/Scene/Model/Extensions/Gpm/GltfGpmLoader.js";
+
+describe("Scene/Model/Extensions/Gpm/GltfGpmLoader", function () {
+ it("load throws with invalid storageType", async function () {
+ const gltfGpmLocalJson = {
+ storageType: "INVALID",
+ };
+ expect(function () {
+ GltfGpmLoader.load(gltfGpmLocalJson);
+ }).toThrowDeveloperError();
+ });
+
+ it("load throws for storageType Direct without anchorPointsDirect", async function () {
+ const gltfGpmLocalJson = {
+ storageType: "Direct",
+ };
+ expect(function () {
+ GltfGpmLoader.load(gltfGpmLocalJson);
+ }).toThrowDeveloperError();
+ });
+
+ it("load throws for storageType Direct without covarianceDirectUpperTriangle", async function () {
+ const gltfGpmLocalJson = {
+ storageType: "Direct",
+ anchorPointsDirect: [
+ {
+ position: [1.0, 2.0, 3.0],
+ adjustmentParams: [0.1, 0.2, 0.3],
+ },
+ ],
+ };
+ expect(function () {
+ GltfGpmLoader.load(gltfGpmLocalJson);
+ }).toThrowDeveloperError();
+ });
+
+ it("load returns result for valid JSON for storageType Direct", async function () {
+ const gltfGpmLocalJson = {
+ storageType: "Direct",
+ anchorPointsDirect: [
+ {
+ position: [1.0, 2.0, 3.0],
+ adjustmentParams: [0.1, 0.2, 0.3],
+ },
+ ],
+ covarianceDirectUpperTriangle: [0.1, 0.2, 0.3, 0.4, 0.5, 0.6],
+ };
+
+ const result = GltfGpmLoader.load(gltfGpmLocalJson);
+ expect(result).toBeDefined();
+ expect(result.anchorPointsDirect.length).toBe(1);
+
+ const actualAnchorPoint = result.anchorPointsDirect[0];
+
+ const expectedPosition = new Cartesian3(1.0, 2.0, 3.0);
+ expect(
+ Cartesian3.equalsEpsilon(
+ actualAnchorPoint.position,
+ expectedPosition,
+ CesiumMath.EPSILON6
+ )
+ ).toBeTrue();
+
+ const expectedAdjustmentParams = new Cartesian3(0.1, 0.2, 0.3);
+ expect(
+ Cartesian3.equalsEpsilon(
+ actualAnchorPoint.adjustmentParams,
+ expectedAdjustmentParams,
+ CesiumMath.EPSILON6
+ )
+ ).toBeTrue();
+
+ const expectedCovarianceDirect = Matrix3.fromArray(
+ [0.1, 0.2, 0.4, 0.2, 0.3, 0.5, 0.4, 0.5, 0.6],
+ 0,
+ new Matrix3()
+ );
+ expect(
+ Matrix3.equalsEpsilon(
+ result.covarianceDirect,
+ expectedCovarianceDirect,
+ CesiumMath.EPSILON6
+ )
+ ).toBeTrue();
+ });
+
+ it("load throws for storageType Indirect without anchorPointsIndirect", async function () {
+ const gltfGpmLocalJson = {
+ storageType: "Indirect",
+ };
+ expect(function () {
+ GltfGpmLoader.load(gltfGpmLocalJson);
+ }).toThrowDeveloperError();
+ });
+
+ it("load throws for storageType Indirect without intraTileCorrelationGroups", async function () {
+ const gltfGpmLocalJson = {
+ storageType: "Indirect",
+ anchorPointsIndirect: [
+ {
+ position: [1.0, 2.0, 3.0],
+ adjustmentParams: [0.1, 0.2, 0.3],
+ },
+ ],
+ };
+ expect(function () {
+ GltfGpmLoader.load(gltfGpmLocalJson);
+ }).toThrowDeveloperError();
+ });
+
+ it("load returns result for valid JSON for storageType Indirect", async function () {
+ const gltfGpmLocalJson = {
+ storageType: "Indirect",
+ anchorPointsIndirect: [
+ {
+ position: [1.0, 2.0, 3.0],
+ adjustmentParams: [0.1, 0.2, 0.3],
+ covarianceMatrix: [0.1, 0.2, 0.3, 0.4, 0.5, 0.6],
+ },
+ ],
+ intraTileCorrelationGroups: [
+ {
+ groupFlags: [true, true, true],
+ rotationThetas: [0.1, 0.2, 0.3],
+ params: [
+ {
+ A: 0.1,
+ alpha: 0.2,
+ beta: 0.3,
+ T: 0.4,
+ },
+ ],
+ },
+ ],
+ };
+
+ const result = GltfGpmLoader.load(gltfGpmLocalJson);
+ expect(result).toBeDefined();
+
+ expect(result.anchorPointsIndirect.length).toBe(1);
+
+ const actualAnchorPoint = result.anchorPointsIndirect[0];
+
+ const expectedPosition = new Cartesian3(1.0, 2.0, 3.0);
+ expect(
+ Cartesian3.equalsEpsilon(
+ actualAnchorPoint.position,
+ expectedPosition,
+ CesiumMath.EPSILON6
+ )
+ ).toBeTrue();
+
+ const expectedAdjustmentParams = new Cartesian3(0.1, 0.2, 0.3);
+ expect(
+ Cartesian3.equalsEpsilon(
+ actualAnchorPoint.adjustmentParams,
+ expectedAdjustmentParams,
+ CesiumMath.EPSILON6
+ )
+ ).toBeTrue();
+
+ const expectedCovarianceMatrix = Matrix3.fromArray(
+ [0.1, 0.2, 0.4, 0.2, 0.3, 0.5, 0.4, 0.5, 0.6],
+ 0,
+ new Matrix3()
+ );
+ expect(
+ Matrix3.equalsEpsilon(
+ actualAnchorPoint.covarianceMatrix,
+ expectedCovarianceMatrix,
+ CesiumMath.EPSILON6
+ )
+ ).toBeTrue();
+
+ expect(result.intraTileCorrelationGroups.length).toBe(1);
+
+ const correlationGroup = result.intraTileCorrelationGroups[0];
+ const groupFlags = correlationGroup.groupFlags;
+ expect(groupFlags).toEqual([true, true, true]);
+
+ const expectedRotationThetas = new Cartesian3(0.1, 0.2, 0.3);
+ expect(
+ Cartesian3.equalsEpsilon(
+ correlationGroup.rotationThetas,
+ expectedRotationThetas,
+ CesiumMath.EPSILON6
+ )
+ ).toBeTrue();
+
+ const params = correlationGroup.params;
+ expect(params.length).toBe(1);
+ const param = params[0];
+ expect(param.A).toBe(0.1);
+ expect(param.alpha).toBe(0.2);
+ expect(param.beta).toBe(0.3);
+ expect(param.T).toBe(0.4);
+ });
+});
From 9dbd45ec7029e62cbe01e829c2070ab6c0b1ed3c Mon Sep 17 00:00:00 2001
From: Marco Hutter
Date: Wed, 25 Sep 2024 15:02:46 +0200
Subject: [PATCH 16/38] Proper imports for spec file
---
.../Scene/Model/Extensions/Gpm/GltfGpmLoaderSpec.js | 10 ++++++----
1 file changed, 6 insertions(+), 4 deletions(-)
diff --git a/packages/engine/Specs/Scene/Model/Extensions/Gpm/GltfGpmLoaderSpec.js b/packages/engine/Specs/Scene/Model/Extensions/Gpm/GltfGpmLoaderSpec.js
index c8920fb2831b..b4381d5aad9a 100644
--- a/packages/engine/Specs/Scene/Model/Extensions/Gpm/GltfGpmLoaderSpec.js
+++ b/packages/engine/Specs/Scene/Model/Extensions/Gpm/GltfGpmLoaderSpec.js
@@ -1,7 +1,9 @@
-import Cartesian3 from "../../../../../Source/Core/Cartesian3.js";
-import CesiumMath from "../../../../../Source/Core/Math.js";
-import Matrix3 from "../../../../../Source/Core/Matrix3.js";
-import GltfGpmLoader from "../../../../../Source/Scene/Model/Extensions/Gpm/GltfGpmLoader.js";
+import {
+ Matrix3,
+ Math as CesiumMath,
+ Cartesian3,
+ GltfGpmLoader,
+} from "../../../../../index.js";
describe("Scene/Model/Extensions/Gpm/GltfGpmLoader", function () {
it("load throws with invalid storageType", async function () {
From 6e290c8a84101a83982ee6d4ada11664f2ed3e8e Mon Sep 17 00:00:00 2001
From: Marco Hutter
Date: Wed, 25 Sep 2024 15:03:07 +0200
Subject: [PATCH 17/38] Explicitly offer parsed GPM data
---
.../Gpm/GltfMeshPrimitiveGpmLoader.js | 21 ++++++++++++++++++-
1 file changed, 20 insertions(+), 1 deletion(-)
diff --git a/packages/engine/Source/Scene/Model/Extensions/Gpm/GltfMeshPrimitiveGpmLoader.js b/packages/engine/Source/Scene/Model/Extensions/Gpm/GltfMeshPrimitiveGpmLoader.js
index c207a962a70c..9fc96bb907e7 100644
--- a/packages/engine/Source/Scene/Model/Extensions/Gpm/GltfMeshPrimitiveGpmLoader.js
+++ b/packages/engine/Source/Scene/Model/Extensions/Gpm/GltfMeshPrimitiveGpmLoader.js
@@ -97,8 +97,27 @@ Object.defineProperties(GltfMeshPrimitiveGpmLoader.prototype, {
return this._cacheKey;
},
},
+
/**
- * The parsed structural metadata
+ * The parsed GPM extension information from the mesh primitive
+ *
+ * @memberof GltfMeshPrimitiveGpmLoader.prototype
+ *
+ * @type {MeshPrimitiveGpmLocal}
+ * @readonly
+ * @private
+ */
+ meshPrimitiveGpmLocal: {
+ get: function () {
+ return this._meshPrimitiveGpmLocal;
+ },
+ },
+
+ /**
+ * Returns the result of converting the parsed 'MeshPrimitiveGpmLocal'
+ * into a 'StructuralMetadata'.
+ *
+ * Some details about the translation are intentionally not specified here.
*
* @memberof GltfMeshPrimitiveGpmLoader.prototype
*
From 6e41b365ba55fb875da9e5c5e1563ec6db22b8ac Mon Sep 17 00:00:00 2001
From: Marco Hutter
Date: Wed, 25 Sep 2024 15:03:26 +0200
Subject: [PATCH 18/38] Add specs for mesh primitive GPM data
---
.../Gpm/GltfMeshPrimitiveGpmLoaderSpec.js | 429 ++++++++++++++++++
1 file changed, 429 insertions(+)
create mode 100644 packages/engine/Specs/Scene/Model/Extensions/Gpm/GltfMeshPrimitiveGpmLoaderSpec.js
diff --git a/packages/engine/Specs/Scene/Model/Extensions/Gpm/GltfMeshPrimitiveGpmLoaderSpec.js b/packages/engine/Specs/Scene/Model/Extensions/Gpm/GltfMeshPrimitiveGpmLoaderSpec.js
new file mode 100644
index 000000000000..81591889fbdc
--- /dev/null
+++ b/packages/engine/Specs/Scene/Model/Extensions/Gpm/GltfMeshPrimitiveGpmLoaderSpec.js
@@ -0,0 +1,429 @@
+import {
+ GltfTextureLoader,
+ Resource,
+ ResourceCache,
+ SupportedImageFormats,
+ defined,
+ GltfMeshPrimitiveGpmLoader,
+} from "../../../../../index.js";
+import createScene from "../../../../../../../Specs/createScene.js";
+import loaderProcess from "../../../../../../../Specs/loaderProcess.js";
+import waitForLoaderProcess from "../../../../../../../Specs/waitForLoaderProcess.js";
+
+/**
+ * The JSON representation of the NGA_gpm_local extension object
+ * that will be inserted into the mesh primitive
+ */
+const ngaGpmLocalExtension = {
+ ppeTextures: [
+ {
+ traits: {
+ source: "SIGZ",
+ min: 0.0,
+ max: 16.0,
+ },
+ index: 0,
+ noData: 255,
+ offset: 0.0,
+ scale: 0.06274509803921569,
+ texCoord: 0,
+ },
+ ],
+};
+
+/**
+ * Creates an embedded glTF with a single mesh primitive.
+ *
+ * The mesh primitive is a single unit square with normals and texture
+ * coordinates in [(0,0)-(1.1)]. The glTF defines a single texture,
+ * with an image that is stored as an embdedded 16x16 PNG file
+ * where the red channel contains values in [0,256] (and the alpha
+ * channels contains 255).
+ *
+ * If the given 'gpmExtension' object is defined, then it will be
+ * inserted as the "NGA_gpm_local" extension in the mesh primitive,
+ * and "NGA_gpm_local" will be added to the 'extensionsUsed'.
+ *
+ * @param {object} gpmExtension The NGA_gpm_local extension JSON object
+ * @returns The glTF
+ */
+function createEmbeddedGltf(gpmExtension) {
+ let meshPrimitiveExtensions;
+ let extensionsUsed;
+
+ if (defined(gpmExtension)) {
+ meshPrimitiveExtensions = {
+ NGA_gpm_local: gpmExtension,
+ };
+ extensionsUsed = ["NGA_gpm_local"];
+ }
+
+ const gltf = {
+ accessors: [
+ {
+ bufferView: 0,
+ byteOffset: 0,
+ componentType: 5123,
+ count: 6,
+ type: "SCALAR",
+ max: [3],
+ min: [0],
+ },
+ {
+ bufferView: 1,
+ byteOffset: 0,
+ componentType: 5126,
+ count: 4,
+ type: "VEC3",
+ max: [1.0, 1.0, 0.0],
+ min: [0.0, 0.0, 0.0],
+ },
+ {
+ bufferView: 1,
+ byteOffset: 48,
+ componentType: 5126,
+ count: 4,
+ type: "VEC3",
+ max: [0.0, 0.0, 1.0],
+ min: [0.0, 0.0, 1.0],
+ },
+ {
+ bufferView: 1,
+ byteOffset: 96,
+ componentType: 5126,
+ count: 4,
+ type: "VEC2",
+ max: [1.0, 1.0],
+ min: [0.0, 0.0],
+ },
+ ],
+ asset: {
+ generator: "JglTF from https://github.com/javagl/JglTF",
+ version: "2.0",
+ },
+ buffers: [
+ {
+ uri:
+ "data:application/gltf-buffer;base64,AAABAAIAAQADAAIAAAAAAAAAAAAAAAAAAACAPwAAAAAAAAAAAAAAAAAAgD8AAAAAAACAPwAAgD8AAAAAAAAAAAAAAAAAAIA/AAAAAAAAAAAAAIA/AAAAAAAAAAAAAIA/AAAAAAAAAAAAAIA/AAAAAAAAgD8AAAAAAACAPwAAgD8AAAAAAAAAAAAAAAAAAAAAAACAPwAAAAAAAAAA",
+ byteLength: 156,
+ },
+ ],
+ bufferViews: [
+ {
+ buffer: 0,
+ byteOffset: 0,
+ byteLength: 12,
+ target: 34963,
+ },
+ {
+ buffer: 0,
+ byteOffset: 12,
+ byteLength: 144,
+ byteStride: 12,
+ target: 34962,
+ },
+ ],
+ images: [
+ {
+ uri:
+ "",
+ mimeType: "image/png",
+ },
+ ],
+ meshes: [
+ {
+ primitives: [
+ {
+ attributes: {
+ POSITION: 1,
+ NORMAL: 2,
+ TEXCOORD_0: 3,
+ },
+ indices: 0,
+ mode: 4,
+ extensions: meshPrimitiveExtensions,
+ },
+ ],
+ },
+ ],
+ nodes: [
+ {
+ mesh: 0,
+ },
+ ],
+ samplers: [
+ {
+ magFilter: 9728,
+ minFilter: 9728,
+ wrapS: 33071,
+ wrapT: 33071,
+ },
+ ],
+ scene: 0,
+ scenes: [
+ {
+ nodes: [0],
+ },
+ ],
+ textures: [
+ {
+ sampler: 0,
+ source: 0,
+ },
+ ],
+ extensionsUsed: extensionsUsed,
+ };
+ return gltf;
+}
+
+// NOTE: Much of this was taken from 'GltfSructuralMetadataLoaderSpec.js'.
+// I don't know what things like the 'mockFrameState' are.
+// I don't know how much of the 'Resource'-specific testing belongs here.
+// I don't know what 'resolveAfterDestroy' is actually testing.
+// Otherwise, I would have added comments...
+describe(
+ "Scene/Model/Extensions/Gpm/GltfMeshPrimitiveGpmLoader",
+ function () {
+ const gltfUri = "https://example.com/model.glb";
+ const gltfResource = new Resource({
+ url: gltfUri,
+ });
+
+ const mockFrameState = {
+ context: {
+ id: "01234",
+ },
+ };
+
+ let scene;
+
+ beforeAll(function () {
+ scene = createScene();
+ });
+
+ afterAll(function () {
+ scene.destroyForSpecs();
+ });
+
+ afterEach(function () {
+ ResourceCache.clearForSpecs();
+ });
+
+ it("throws if gltf is undefined", function () {
+ const extension = undefined;
+ const gltf = undefined;
+
+ expect(function () {
+ return new GltfMeshPrimitiveGpmLoader({
+ gltf: gltf,
+ extension: extension,
+ gltfResource: gltfResource,
+ baseResource: gltfResource,
+ supportedImageFormats: new SupportedImageFormats(),
+ frameState: mockFrameState,
+ });
+ }).toThrowDeveloperError();
+ });
+
+ it("throws if gltfResource is undefined", function () {
+ const extension = undefined;
+ const gltf = undefined;
+
+ expect(function () {
+ return new GltfMeshPrimitiveGpmLoader({
+ gltf: gltf,
+ extension: extension,
+ gltfResource: undefined,
+ baseResource: gltfResource,
+ supportedImageFormats: new SupportedImageFormats(),
+ frameState: mockFrameState,
+ });
+ }).toThrowDeveloperError();
+ });
+
+ it("throws if baseResource is undefined", function () {
+ const extension = undefined;
+ const gltf = undefined;
+
+ expect(function () {
+ return new GltfMeshPrimitiveGpmLoader({
+ gltf: gltf,
+ extension: extension,
+ gltfResource: gltfResource,
+ baseResource: undefined,
+ supportedImageFormats: new SupportedImageFormats(),
+ frameState: mockFrameState,
+ });
+ }).toThrowDeveloperError();
+ });
+
+ it("throws if supportedImageFormats is undefined", function () {
+ const extension = undefined;
+ const gltf = undefined;
+
+ expect(function () {
+ return new GltfMeshPrimitiveGpmLoader({
+ gltf: gltf,
+ extension: extension,
+ gltfResource: gltfResource,
+ baseResource: gltfResource,
+ supportedImageFormats: undefined,
+ frameState: mockFrameState,
+ });
+ }).toThrowDeveloperError();
+ });
+
+ it("throws if frameState is undefined", function () {
+ const extension = undefined;
+ const gltf = undefined;
+
+ expect(function () {
+ return new GltfMeshPrimitiveGpmLoader({
+ gltf: gltf,
+ extension: extension,
+ gltfResource: gltfResource,
+ baseResource: gltfResource,
+ supportedImageFormats: new SupportedImageFormats(),
+ frameState: undefined,
+ });
+ }).toThrowDeveloperError();
+ });
+
+ it("loads mesh primitive GPM extension data", async function () {
+ const extension = ngaGpmLocalExtension;
+ const gltf = createEmbeddedGltf(extension);
+
+ const loader = new GltfMeshPrimitiveGpmLoader({
+ gltf: gltf,
+ extension: extension,
+ gltfResource: gltfResource,
+ baseResource: gltfResource,
+ supportedImageFormats: new SupportedImageFormats(),
+ frameState: mockFrameState,
+ });
+
+ await loader.load();
+ await waitForLoaderProcess(loader, scene);
+ expect(() => loaderProcess(loader, scene)).not.toThrow();
+
+ const gpmData = loader.meshPrimitiveGpmLocal;
+ expect(gpmData).toBeDefined();
+
+ const ppeTextures = gpmData.ppeTextures;
+ expect(ppeTextures).toBeDefined();
+ expect(ppeTextures.length).toBe(1);
+
+ const ppeTexture = ppeTextures[0];
+ expect(ppeTexture.index).toBe(0);
+ expect(ppeTexture.texCoord).toBe(0);
+ expect(ppeTexture.noData).toBe(255);
+ expect(ppeTexture.offset).toBe(0.0);
+ expect(ppeTexture.scale).toBe(0.06274509803921569);
+
+ expect(ppeTexture.traits).toBeDefined();
+
+ const traits = ppeTexture.traits;
+ expect(traits.min).toBe(0);
+ expect(traits.max).toBe(16);
+ expect(traits.source).toBe("SIGZ");
+ });
+
+ it("converts mesh primitive GPM extension data into structural metadata", async function () {
+ const extension = ngaGpmLocalExtension;
+ const gltf = createEmbeddedGltf(extension);
+
+ const loader = new GltfMeshPrimitiveGpmLoader({
+ gltf: gltf,
+ extension: extension,
+ gltfResource: gltfResource,
+ baseResource: gltfResource,
+ supportedImageFormats: new SupportedImageFormats(),
+ frameState: mockFrameState,
+ });
+
+ await loader.load();
+ await waitForLoaderProcess(loader, scene);
+ expect(() => loaderProcess(loader, scene)).not.toThrow();
+
+ const structuralMetadata = loader.structuralMetadata;
+ expect(structuralMetadata).toBeDefined();
+
+ const ppePropertyTexture = structuralMetadata.getPropertyTexture(0);
+ expect(ppePropertyTexture.id).toBe(0);
+
+ const sigzProperty = ppePropertyTexture.getProperty("SIGZ");
+
+ expect(sigzProperty.textureReader.texture.width).toBe(16);
+ expect(sigzProperty.textureReader.texture.height).toBe(16);
+ });
+
+ it("destroys structural metadata", async function () {
+ const extension = ngaGpmLocalExtension;
+ const gltf = createEmbeddedGltf(extension);
+
+ const destroyTexture = spyOn(
+ GltfTextureLoader.prototype,
+ "destroy"
+ ).and.callThrough();
+
+ const loader = new GltfMeshPrimitiveGpmLoader({
+ gltf: gltf,
+ extension: extension,
+ gltfResource: gltfResource,
+ baseResource: gltfResource,
+ supportedImageFormats: new SupportedImageFormats(),
+ frameState: mockFrameState,
+ });
+
+ await loader.load();
+
+ await waitForLoaderProcess(loader, scene);
+ expect(loader.structuralMetadata).toBeDefined();
+ expect(loader.isDestroyed()).toBe(false);
+
+ loader.destroy();
+
+ expect(loader.structuralMetadata).not.toBeDefined();
+ expect(loader.isDestroyed()).toBe(true);
+
+ expect(destroyTexture.calls.count()).toBe(1);
+ });
+
+ async function resolveAfterDestroy(rejectPromise) {
+ const extension = ngaGpmLocalExtension;
+ const gltf = createEmbeddedGltf(extension);
+
+ const destroyTexture = spyOn(
+ GltfTextureLoader.prototype,
+ "destroy"
+ ).and.callThrough();
+
+ const loader = new GltfMeshPrimitiveGpmLoader({
+ gltf: gltf,
+ extension: extension,
+ gltfResource: gltfResource,
+ baseResource: gltfResource,
+ supportedImageFormats: new SupportedImageFormats(),
+ frameState: mockFrameState,
+ });
+ expect(loader.structuralMetadata).not.toBeDefined();
+ const promise = loader.load();
+ loader.destroy();
+
+ expect(loader.structuralMetadata).not.toBeDefined();
+ expect(loader.isDestroyed()).toBe(true);
+
+ expect(destroyTexture.calls.count()).toBe(1);
+ await expectAsync(promise).toBeResolved();
+ }
+
+ it("handles resolving resources after destroy", function () {
+ return resolveAfterDestroy(false);
+ });
+
+ it("handles rejecting resources after destroy", function () {
+ return resolveAfterDestroy(true);
+ });
+ },
+ "WebGL"
+);
From 616171f1fd26d38e9330e31daac0eb6f4765f04a Mon Sep 17 00:00:00 2001
From: Marco Hutter
Date: Wed, 25 Sep 2024 15:18:03 +0200
Subject: [PATCH 19/38] Remove log. Check for definedness.
---
packages/engine/Source/Scene/GltfLoader.js | 21 +++++++++++----------
1 file changed, 11 insertions(+), 10 deletions(-)
diff --git a/packages/engine/Source/Scene/GltfLoader.js b/packages/engine/Source/Scene/GltfLoader.js
index c6733177bdd0..ea4fa5303526 100644
--- a/packages/engine/Source/Scene/GltfLoader.js
+++ b/packages/engine/Source/Scene/GltfLoader.js
@@ -2718,7 +2718,6 @@ function parse(loader, frameState) {
// Load NGA_gpm_local from root object
const gpmExtension = extensions.NGA_gpm_local;
if (defined(gpmExtension)) {
- console.log("Loading GPM from root");
const gltfGpmLocal = GltfGpmLoader.load(gpmExtension);
loader._components.extensions["NGA_gpm_local"] = gltfGpmLocal;
}
@@ -2731,15 +2730,17 @@ function parse(loader, frameState) {
if (defined(primitives)) {
for (const primitive of primitives) {
const primitiveExtensions = primitive.extensions;
- const meshPrimitiveGpmExtension = primitiveExtensions.NGA_gpm_local;
- if (defined(meshPrimitiveGpmExtension)) {
- const promise = loadMeshPrimitiveGpm(
- loader,
- gltf,
- meshPrimitiveGpmExtension,
- frameState
- );
- loader._loaderPromises.push(promise);
+ if (defined(primitiveExtensions)) {
+ const meshPrimitiveGpmExtension = primitiveExtensions.NGA_gpm_local;
+ if (defined(meshPrimitiveGpmExtension)) {
+ const promise = loadMeshPrimitiveGpm(
+ loader,
+ gltf,
+ meshPrimitiveGpmExtension,
+ frameState
+ );
+ loader._loaderPromises.push(promise);
+ }
}
}
}
From 8c0f9263480991dded66326aa9be42c8ea1057bd Mon Sep 17 00:00:00 2001
From: Marco Hutter
Date: Fri, 27 Sep 2024 20:44:48 +0200
Subject: [PATCH 20/38] Remove experimental tag for getExtensions
---
packages/engine/Source/Scene/Model/Model.js | 2 --
1 file changed, 2 deletions(-)
diff --git a/packages/engine/Source/Scene/Model/Model.js b/packages/engine/Source/Scene/Model/Model.js
index 942cbf6a3d3c..7e81d46aa9c6 100644
--- a/packages/engine/Source/Scene/Model/Model.js
+++ b/packages/engine/Source/Scene/Model/Model.js
@@ -1784,8 +1784,6 @@ Model.prototype.applyArticulations = function () {
* @exception {DeveloperError} The model is not loaded. Use Model.readyEvent or wait for Model.ready to be true.
*
* @private
- * @experimental This feature is subject to change without Cesium's standard deprecation policy.
- *
*/
Model.prototype.getExtension = function (extensionName) {
//>>includeStart('debug', pragmas.debug);
From 5723c8b9499e904968e358dab2221cf019c2ab68 Mon Sep 17 00:00:00 2001
From: Marco Hutter
Date: Fri, 27 Sep 2024 20:46:21 +0200
Subject: [PATCH 21/38] Add return type to JSDoc
---
packages/engine/Source/Scene/Model/Model.js | 2 +-
1 file changed, 1 insertion(+), 1 deletion(-)
diff --git a/packages/engine/Source/Scene/Model/Model.js b/packages/engine/Source/Scene/Model/Model.js
index 7e81d46aa9c6..4c79779b66cd 100644
--- a/packages/engine/Source/Scene/Model/Model.js
+++ b/packages/engine/Source/Scene/Model/Model.js
@@ -1780,7 +1780,7 @@ Model.prototype.applyArticulations = function () {
* this will return the model representation of the extension.
*
* @param {string} extensionName The name of the extension
- * @returns The object, or `undefined`
+ * @returns {object|undefined} The object, or `undefined`
* @exception {DeveloperError} The model is not loaded. Use Model.readyEvent or wait for Model.ready to be true.
*
* @private
From bb74ec4b4e3c7ff517eef97141d9f3fa90cef2ad Mon Sep 17 00:00:00 2001
From: Marco Hutter
Date: Fri, 27 Sep 2024 20:48:46 +0200
Subject: [PATCH 22/38] Add JSDoc for getExtension
---
.../engine/Source/Scene/Model/Model3DTileContent.js | 13 +++++++++++++
1 file changed, 13 insertions(+)
diff --git a/packages/engine/Source/Scene/Model/Model3DTileContent.js b/packages/engine/Source/Scene/Model/Model3DTileContent.js
index 15b0e10f49e7..bdc0b3831a81 100644
--- a/packages/engine/Source/Scene/Model/Model3DTileContent.js
+++ b/packages/engine/Source/Scene/Model/Model3DTileContent.js
@@ -152,6 +152,19 @@ Object.defineProperties(Model3DTileContent.prototype, {
},
});
+/**
+ * Returns the object that was created for the given extension.
+ *
+ * The given name may be the name of a glTF extension, like `"EXT_example_extension"`.
+ * If the specified extension was present in the root of the underlying glTF asset,
+ * and a loder for the specified extension has processed the extension data, then
+ * this will return the model representation of the extension.
+ *
+ * @param {string} extensionName The name of the extension
+ * @returns {object|undefined} The object, or `undefined`
+ *
+ * @private
+ */
Model3DTileContent.prototype.getExtension = function (extensionName) {
const model = this._model;
const extension = model.getExtension(extensionName);
From 962a9864790a9bc224eb35fba9e3a9751c133fdc Mon Sep 17 00:00:00 2001
From: Marco Hutter
Date: Fri, 27 Sep 2024 20:50:00 +0200
Subject: [PATCH 23/38] Add type tag to property
---
packages/engine/Source/Scene/ModelComponents.js | 3 +++
1 file changed, 3 insertions(+)
diff --git a/packages/engine/Source/Scene/ModelComponents.js b/packages/engine/Source/Scene/ModelComponents.js
index fa49d4111c2d..fc72bf5baac6 100644
--- a/packages/engine/Source/Scene/ModelComponents.js
+++ b/packages/engine/Source/Scene/ModelComponents.js
@@ -1145,6 +1145,9 @@ function Components() {
/**
* A mapping from extension names like `"EXT_example_extension"` to
* the object that was created from the extension input
+ *
+ * @type {object}
+ * @private
*/
this.extensions = {};
}
From 804fa0a68bfb38298b951fdba7549c13a2a29f95 Mon Sep 17 00:00:00 2001
From: Marco Hutter
Date: Fri, 27 Sep 2024 20:57:31 +0200
Subject: [PATCH 24/38] More elaborate warning text
---
packages/engine/Source/Scene/GltfLoader.js | 5 ++++-
1 file changed, 4 insertions(+), 1 deletion(-)
diff --git a/packages/engine/Source/Scene/GltfLoader.js b/packages/engine/Source/Scene/GltfLoader.js
index 739f129d0f95..d853d90d73dc 100644
--- a/packages/engine/Source/Scene/GltfLoader.js
+++ b/packages/engine/Source/Scene/GltfLoader.js
@@ -482,7 +482,10 @@ function processLoaders(loader, frameState) {
if (defined(loader._components.structuralMetadata)) {
oneTimeWarning(
"structural-metadata-gpm",
- "Structural metadata is replaced with GPM data",
+ "The model defines both the 'EXT_structural_metadata' extension and the " +
+ "'NGA_gpm_local' extension. The data from the 'EXT_structural_metadata' " +
+ "extension will be replaced with the data from the 'NGA_gpm_local' extension, " +
+ "and will no longer be available for styling and picking.",
);
}
loader._components.structuralMetadata =
From 905925a3df64452622a5cd54674fd4b46f6ff38c Mon Sep 17 00:00:00 2001
From: Marco Hutter
Date: Fri, 27 Sep 2024 21:07:54 +0200
Subject: [PATCH 25/38] Removed obsolete part of JSDoc
---
.../engine/Source/Scene/Model/Extensions/Gpm/GltfGpmLoader.js | 2 --
1 file changed, 2 deletions(-)
diff --git a/packages/engine/Source/Scene/Model/Extensions/Gpm/GltfGpmLoader.js b/packages/engine/Source/Scene/Model/Extensions/Gpm/GltfGpmLoader.js
index b872a53c7c57..ab9322002a43 100644
--- a/packages/engine/Source/Scene/Model/Extensions/Gpm/GltfGpmLoader.js
+++ b/packages/engine/Source/Scene/Model/Extensions/Gpm/GltfGpmLoader.js
@@ -11,8 +11,6 @@ import StorageType from "./StorageType.js";
/**
* Loads glTF NGA_gpm_local from the root of a glTF object
- *
- * Implementation note: This is an experimental implementation.
*
* @alias GltfGpmLoader
* @constructor
From e887e538b87174b5bca84ace878bce6c11200136 Mon Sep 17 00:00:00 2001
From: Marco Hutter
Date: Fri, 27 Sep 2024 21:14:25 +0200
Subject: [PATCH 26/38] Add JSDoc return type info
---
.../Source/Scene/Model/Extensions/Gpm/GltfGpmLoader.js | 8 ++++----
1 file changed, 4 insertions(+), 4 deletions(-)
diff --git a/packages/engine/Source/Scene/Model/Extensions/Gpm/GltfGpmLoader.js b/packages/engine/Source/Scene/Model/Extensions/Gpm/GltfGpmLoader.js
index ab9322002a43..4d5a95cc965a 100644
--- a/packages/engine/Source/Scene/Model/Extensions/Gpm/GltfGpmLoader.js
+++ b/packages/engine/Source/Scene/Model/Extensions/Gpm/GltfGpmLoader.js
@@ -30,7 +30,7 @@ function GltfGpmLoader() {}
* column-major order.
*
* @param {number[]} array The input array
- * @returns The Matrix3
+ * @returns {Matrix3} The Matrix3
*/
function createCovarianceMatrixFromUpperTriangle(array) {
const covarianceMatrix = new Matrix3(
@@ -51,7 +51,7 @@ function createCovarianceMatrixFromUpperTriangle(array) {
* Creates an `AnchorPointDirect` from the given JSON representation
*
* @param {object} anchorPointDirectJson The input JSON
- * @returns The `AnchorPointDirect`
+ * @returns {AnchorPointDirect} The direct anchor point
*/
function createAnchorPointDirect(anchorPointDirectJson) {
const position = Cartesian3.fromArray(
@@ -75,7 +75,7 @@ function createAnchorPointDirect(anchorPointDirectJson) {
* Creates an `AnchorPointIndirect` from the given JSON representation
*
* @param {object} anchorPointIndirectJson The input JSON
- * @returns The `AnchorPointIndirect`
+ * @returns {AnchorPointIndirect} The indirect anchor point
*/
function createAnchorPointIndirect(anchorPointIndirectJson) {
const position = Cartesian3.fromArray(
@@ -103,7 +103,7 @@ function createAnchorPointIndirect(anchorPointIndirectJson) {
* Creates a `CorrelationGroup` from the given JSON representation
*
* @param {object} correlationGroupJson The input JSON
- * @returns The `CorrelationGroup`
+ * @returns {CorrelationGroup} The correlation group
*/
function createCorrelationGroup(correlationGroupJson) {
const groupFlags = correlationGroupJson.groupFlags;
From 13b2db09dc56712fac91bd0d73b96f8d2538c6af Mon Sep 17 00:00:00 2001
From: Marco Hutter
Date: Fri, 27 Sep 2024 21:16:24 +0200
Subject: [PATCH 27/38] Change error type for load function
---
.../Source/Scene/Model/Extensions/Gpm/GltfGpmLoader.js | 7 +++----
1 file changed, 3 insertions(+), 4 deletions(-)
diff --git a/packages/engine/Source/Scene/Model/Extensions/Gpm/GltfGpmLoader.js b/packages/engine/Source/Scene/Model/Extensions/Gpm/GltfGpmLoader.js
index 4d5a95cc965a..25143d00ba25 100644
--- a/packages/engine/Source/Scene/Model/Extensions/Gpm/GltfGpmLoader.js
+++ b/packages/engine/Source/Scene/Model/Extensions/Gpm/GltfGpmLoader.js
@@ -1,7 +1,7 @@
import Cartesian3 from "../../../../Core/Cartesian3.js";
import Check from "../../../../Core/Check.js";
-import DeveloperError from "../../../../Core/DeveloperError.js";
import Matrix3 from "../../../../Core/Matrix3.js";
+import RuntimeError from "../../../../Core/RuntimeError.js";
import AnchorPointDirect from "./AnchorPointDirect.js";
import AnchorPointIndirect from "./AnchorPointIndirect.js";
import CorrelationGroup from "./CorrelationGroup.js";
@@ -136,9 +136,8 @@ function createCorrelationGroup(correlationGroupJson) {
*
* @param {object} gltfGpmLocalJson The extension object
* @returns {GltfGpmLocal} The parsed object
+ * @throws RuntimeError When the given object contains invalid storage types.
* @private
- * @throws DeveloperError When the given object is `undefined`
- * or contains invalid structures
*/
GltfGpmLoader.load = function (gltfGpmLocalJson) {
//>>includeStart('debug', pragmas.debug);
@@ -152,7 +151,7 @@ GltfGpmLoader.load = function (gltfGpmLocalJson) {
if (storageType === StorageType.Indirect) {
return GltfGpmLoader.loadIndirect(gltfGpmLocalJson);
}
- throw new DeveloperError(
+ throw new RuntimeError(
`Invalid storage type in NGA_gpm_local - expected 'Direct' or 'Indirect', but found ${storageType}`,
);
};
From ac78182077e17745236cd61b86ba8522c32f436b Mon Sep 17 00:00:00 2001
From: Marco Hutter
Date: Fri, 27 Sep 2024 21:29:29 +0200
Subject: [PATCH 28/38] List supported extension in model documentation
---
packages/engine/Source/Scene/Model/Model.js | 3 +++
1 file changed, 3 insertions(+)
diff --git a/packages/engine/Source/Scene/Model/Model.js b/packages/engine/Source/Scene/Model/Model.js
index 4c79779b66cd..0892388a51a6 100644
--- a/packages/engine/Source/Scene/Model/Model.js
+++ b/packages/engine/Source/Scene/Model/Model.js
@@ -105,6 +105,9 @@ import pickModel from "./pickModel.js";
*
* {@link https://github.com/KhronosGroup/glTF/blob/main/extensions/1.0/Vendor/WEB3D_quantized_attributes/README.md|WEB3D_quantized_attributes}
*
+ *
+ * {@link https://nsgreg.nga.mil/csmwg.jsp|NGA_gpm_local}
+ *
*
*
*
From e61667c10f67ace80c730ab035f10d805be0163d Mon Sep 17 00:00:00 2001
From: Marco Hutter
Date: Fri, 27 Sep 2024 21:32:06 +0200
Subject: [PATCH 29/38] Update CHANGES.md
---
CHANGES.md | 1 +
1 file changed, 1 insertion(+)
diff --git a/CHANGES.md b/CHANGES.md
index bf6d8f4a47e6..3bf60ce39bbe 100644
--- a/CHANGES.md
+++ b/CHANGES.md
@@ -8,6 +8,7 @@
- Added `enableVerticalExaggeration` option to models. Set this value to `false` to prevent model exaggeration when `Scene.verticalExaggeration` is set to a value other than `1.0`. [#12141](https://github.com/CesiumGS/cesium/pull/12141)
- Added `CallbackPositionProperty` to allow lazy entity position evaluation. [#12170](https://github.com/CesiumGS/cesium/pull/12170)
+- Added experimental support for the `NGA_gpm_local` glTF extension, for GPM 1.2 [#12204](https://github.com/CesiumGS/cesium/pull/12204)
##### Fixes :wrench:
From e94ca56b6791b5783423392d99ea6c664d10d901 Mon Sep 17 00:00:00 2001
From: Marco Hutter
Date: Sat, 28 Sep 2024 13:19:40 +0200
Subject: [PATCH 30/38] Try to clarify private and experimental
---
.../Model/Extensions/Gpm/AnchorPointDirect.js | 3 ---
.../Model/Extensions/Gpm/AnchorPointIndirect.js | 4 ----
.../Scene/Model/Extensions/Gpm/CorrelationGroup.js | 4 ----
.../Scene/Model/Extensions/Gpm/GltfGpmLoader.js | 1 -
.../Scene/Model/Extensions/Gpm/GltfGpmLocal.js | 14 ++++----------
.../Extensions/Gpm/GltfMeshPrimitiveGpmLoader.js | 1 -
.../Model/Extensions/Gpm/MeshPrimitiveGpmLocal.js | 2 --
.../Scene/Model/Extensions/Gpm/PpeMetadata.js | 4 ----
.../Scene/Model/Extensions/Gpm/PpeTexture.js | 7 -------
.../Source/Scene/Model/Extensions/Gpm/Spdcf.js | 5 -----
packages/engine/Source/Scene/Model/Model.js | 4 ++--
11 files changed, 6 insertions(+), 43 deletions(-)
diff --git a/packages/engine/Source/Scene/Model/Extensions/Gpm/AnchorPointDirect.js b/packages/engine/Source/Scene/Model/Extensions/Gpm/AnchorPointDirect.js
index 903330dada7f..05bc225f9260 100644
--- a/packages/engine/Source/Scene/Model/Extensions/Gpm/AnchorPointDirect.js
+++ b/packages/engine/Source/Scene/Model/Extensions/Gpm/AnchorPointDirect.js
@@ -3,7 +3,6 @@ import Check from "../../../../Core/Check.js";
/**
* Metadata for one stored anchor point using direct storage
*
- * @private
* @experimental This feature is not final and is subject to change without Cesium's standard deprecation policy.
*/
function AnchorPointDirect(options) {
@@ -23,7 +22,6 @@ Object.defineProperties(AnchorPointDirect.prototype, {
* @memberof AnchorPointDirect.prototype
* @type {Cartesian3}
* @readonly
- * @private
*/
position: {
get: function () {
@@ -38,7 +36,6 @@ Object.defineProperties(AnchorPointDirect.prototype, {
* @memberof AnchorPointDirect.prototype
* @type {Cartesian3}
* @readonly
- * @private
*/
adjustmentParams: {
get: function () {
diff --git a/packages/engine/Source/Scene/Model/Extensions/Gpm/AnchorPointIndirect.js b/packages/engine/Source/Scene/Model/Extensions/Gpm/AnchorPointIndirect.js
index f0102db32943..03599b48ce8d 100644
--- a/packages/engine/Source/Scene/Model/Extensions/Gpm/AnchorPointIndirect.js
+++ b/packages/engine/Source/Scene/Model/Extensions/Gpm/AnchorPointIndirect.js
@@ -3,7 +3,6 @@ import Check from "../../../../Core/Check.js";
/**
* Metadata for one stored anchor point.
*
- * @private
* @experimental This feature is not final and is subject to change without Cesium's standard deprecation policy.
*/
function AnchorPointIndirect(options) {
@@ -25,7 +24,6 @@ Object.defineProperties(AnchorPointIndirect.prototype, {
* @memberof AnchorPointIndirect.prototype
* @type {Cartesian3}
* @readonly
- * @private
*/
position: {
get: function () {
@@ -40,7 +38,6 @@ Object.defineProperties(AnchorPointIndirect.prototype, {
* @memberof AnchorPointIndirect.prototype
* @type {Cartesian3}
* @readonly
- * @private
*/
adjustmentParams: {
get: function () {
@@ -54,7 +51,6 @@ Object.defineProperties(AnchorPointIndirect.prototype, {
* @memberof AnchorPointIndirect.prototype
* @type {Matrix3}
* @readonly
- * @private
*/
covarianceMatrix: {
get: function () {
diff --git a/packages/engine/Source/Scene/Model/Extensions/Gpm/CorrelationGroup.js b/packages/engine/Source/Scene/Model/Extensions/Gpm/CorrelationGroup.js
index cc82812aa608..23812c03871b 100644
--- a/packages/engine/Source/Scene/Model/Extensions/Gpm/CorrelationGroup.js
+++ b/packages/engine/Source/Scene/Model/Extensions/Gpm/CorrelationGroup.js
@@ -4,7 +4,6 @@ import Check from "../../../../Core/Check.js";
* Metadata identifying parameters using same correlation modeling and
* associated correlation parameters.
*
- * @private
* @experimental This feature is not final and is subject to change without Cesium's standard deprecation policy.
*/
function CorrelationGroup(options) {
@@ -27,7 +26,6 @@ Object.defineProperties(CorrelationGroup.prototype, {
* @memberof CorrelationGroup.prototype
* @type {boolean[]}
* @readonly
- * @private
*/
groupFlags: {
get: function () {
@@ -41,7 +39,6 @@ Object.defineProperties(CorrelationGroup.prototype, {
* @memberof CorrelationGroup.prototype
* @type {Cartesian3}
* @readonly
- * @private
*/
rotationThetas: {
get: function () {
@@ -55,7 +52,6 @@ Object.defineProperties(CorrelationGroup.prototype, {
* @memberof CorrelationGroup.prototype
* @type {Spdcf[]}
* @readonly
- * @private
*/
params: {
get: function () {
diff --git a/packages/engine/Source/Scene/Model/Extensions/Gpm/GltfGpmLoader.js b/packages/engine/Source/Scene/Model/Extensions/Gpm/GltfGpmLoader.js
index 25143d00ba25..f3cd3e2dd2d7 100644
--- a/packages/engine/Source/Scene/Model/Extensions/Gpm/GltfGpmLoader.js
+++ b/packages/engine/Source/Scene/Model/Extensions/Gpm/GltfGpmLoader.js
@@ -20,7 +20,6 @@ import StorageType from "./StorageType.js";
* @param {string} [options.extension] The NGA_gpm_local
extension object.
*
* @private
- * @experimental This feature is subject to change without Cesium's standard deprecation policy.
*/
function GltfGpmLoader() {}
diff --git a/packages/engine/Source/Scene/Model/Extensions/Gpm/GltfGpmLocal.js b/packages/engine/Source/Scene/Model/Extensions/Gpm/GltfGpmLocal.js
index 76878a9c4c78..73fc5789de13 100644
--- a/packages/engine/Source/Scene/Model/Extensions/Gpm/GltfGpmLocal.js
+++ b/packages/engine/Source/Scene/Model/Extensions/Gpm/GltfGpmLocal.js
@@ -7,7 +7,6 @@ import StorageType from "./StorageType.js";
* The GPM metadata for a Ground-Space Indirect implementation stored
* locally (i.e. a tile and/or leaf node).
*
- * @private
* @experimental This feature is not final and is subject to change without Cesium's standard deprecation policy.
*/
function GltfGpmLocal(options) {
@@ -76,7 +75,6 @@ Object.defineProperties(GltfGpmLocal.prototype, {
* @memberof GltfGpmLocal.prototype
* @type {StorageType}
* @readonly
- * @private
*/
storageType: {
get: function () {
@@ -88,9 +86,8 @@ Object.defineProperties(GltfGpmLocal.prototype, {
* Array of stored indirect anchor points
*
* @memberof GltfGpmLocal.prototype
- * @type {AnchorPointIndirect[]}
+ * @type {AnchorPointIndirect[]|undefined}
* @readonly
- * @private
*/
anchorPointsIndirect: {
get: function () {
@@ -102,9 +99,8 @@ Object.defineProperties(GltfGpmLocal.prototype, {
* Array of stored direct anchor points
*
* @memberof GltfGpmLocal.prototype
- * @type {AnchorPointDirect[]}
+ * @type {AnchorPointDirect[]|undefined}
* @readonly
- * @private
*/
anchorPointsDirect: {
get: function () {
@@ -117,9 +113,8 @@ Object.defineProperties(GltfGpmLocal.prototype, {
* associated correlation parameters
*
* @memberof GltfGpmLocal.prototype
- * @type {CorrelationGroup[]}
+ * @type {CorrelationGroup[]|undefined}
* @readonly
- * @private
*/
intraTileCorrelationGroups: {
get: function () {
@@ -131,9 +126,8 @@ Object.defineProperties(GltfGpmLocal.prototype, {
* The full covariance of anchor point parameters
*
* @memberof GltfGpmLocal.prototype
- * @type {Matrix3}
+ * @type {Matrix3|undefined}
* @readonly
- * @private
*/
covarianceDirect: {
get: function () {
diff --git a/packages/engine/Source/Scene/Model/Extensions/Gpm/GltfMeshPrimitiveGpmLoader.js b/packages/engine/Source/Scene/Model/Extensions/Gpm/GltfMeshPrimitiveGpmLoader.js
index 76c860fe7ceb..b718ff49e504 100644
--- a/packages/engine/Source/Scene/Model/Extensions/Gpm/GltfMeshPrimitiveGpmLoader.js
+++ b/packages/engine/Source/Scene/Model/Extensions/Gpm/GltfMeshPrimitiveGpmLoader.js
@@ -37,7 +37,6 @@ import MeshPrimitiveGpmLocal from "./MeshPrimitiveGpmLocal.js";
* @param {boolean} [options.asynchronous=true] Determines if WebGL resource creation will be spread out over several frames or block until all WebGL resources are created.
*
* @private
- * @experimental This feature is subject to change without Cesium's standard deprecation policy.
*/
function GltfMeshPrimitiveGpmLoader(options) {
options = defaultValue(options, defaultValue.EMPTY_OBJECT);
diff --git a/packages/engine/Source/Scene/Model/Extensions/Gpm/MeshPrimitiveGpmLocal.js b/packages/engine/Source/Scene/Model/Extensions/Gpm/MeshPrimitiveGpmLocal.js
index 1ca122137744..5cdf62f37943 100644
--- a/packages/engine/Source/Scene/Model/Extensions/Gpm/MeshPrimitiveGpmLocal.js
+++ b/packages/engine/Source/Scene/Model/Extensions/Gpm/MeshPrimitiveGpmLocal.js
@@ -1,7 +1,6 @@
/**
* Local Generic Point-cloud Model information about a glTF primitive.
*
- * @private
* @experimental This feature is not final and is subject to change without Cesium's standard deprecation policy.
*/
function MeshPrimitiveGpmLocal(options) {
@@ -15,7 +14,6 @@ Object.defineProperties(MeshPrimitiveGpmLocal.prototype, {
* @memberof MeshPrimitiveGpmLocal.prototype
* @type {PpeTexture[]|undefined}
* @readonly
- * @private
*/
ppeTextures: {
get: function () {
diff --git a/packages/engine/Source/Scene/Model/Extensions/Gpm/PpeMetadata.js b/packages/engine/Source/Scene/Model/Extensions/Gpm/PpeMetadata.js
index 66d38ef40cb8..457779cf4b40 100644
--- a/packages/engine/Source/Scene/Model/Extensions/Gpm/PpeMetadata.js
+++ b/packages/engine/Source/Scene/Model/Extensions/Gpm/PpeMetadata.js
@@ -3,7 +3,6 @@ import Check from "../../../../Core/Check.js";
/**
* Metadata related to the stored PPE data.
*
- * @private
* @experimental This feature is not final and is subject to change without Cesium's standard deprecation policy.
*/
function PpeMetadata(options) {
@@ -25,7 +24,6 @@ Object.defineProperties(PpeMetadata.prototype, {
* @memberof PpeMetadata.prototype
* @type {number|undefined}
* @readonly
- * @private
*/
min: {
get: function () {
@@ -41,7 +39,6 @@ Object.defineProperties(PpeMetadata.prototype, {
* @memberof PpeMetadata.prototype
* @type {number|undefined}
* @readonly
- * @private
*/
max: {
get: function () {
@@ -55,7 +52,6 @@ Object.defineProperties(PpeMetadata.prototype, {
* @memberof PpeMetadata.prototype
* @type {PpeSource}
* @readonly
- * @private
*/
source: {
get: function () {
diff --git a/packages/engine/Source/Scene/Model/Extensions/Gpm/PpeTexture.js b/packages/engine/Source/Scene/Model/Extensions/Gpm/PpeTexture.js
index 8b352956758d..f8b8109f773d 100644
--- a/packages/engine/Source/Scene/Model/Extensions/Gpm/PpeTexture.js
+++ b/packages/engine/Source/Scene/Model/Extensions/Gpm/PpeTexture.js
@@ -7,7 +7,6 @@ import Check from "../../../../Core/Check.js";
* and an optional `texCoord)`, with additional properties that
* describe the structure of the metdata that is stored in the texture.
*
- * @private
* @experimental This feature is not final and is subject to change without Cesium's standard deprecation policy.
*/
function PpeTexture(options) {
@@ -32,7 +31,6 @@ Object.defineProperties(PpeTexture.prototype, {
* @memberof PpeTexture.prototype
* @type {PpeMetadata}
* @readonly
- * @private
*/
traits: {
get: function () {
@@ -47,7 +45,6 @@ Object.defineProperties(PpeTexture.prototype, {
* @memberof PpeTexture.prototype
* @type {number|undefined}
* @readonly
- * @private
*/
noData: {
get: function () {
@@ -61,7 +58,6 @@ Object.defineProperties(PpeTexture.prototype, {
* @memberof PpeTexture.prototype
* @type {number|undefined}
* @readonly
- * @private
*/
offset: {
get: function () {
@@ -75,7 +71,6 @@ Object.defineProperties(PpeTexture.prototype, {
* @memberof PpeTexture.prototype
* @type {number|undefined}
* @readonly
- * @private
*/
scale: {
get: function () {
@@ -89,7 +84,6 @@ Object.defineProperties(PpeTexture.prototype, {
* @memberof PpeTexture.prototype
* @type {number}
* @readonly
- * @private
*/
index: {
get: function () {
@@ -103,7 +97,6 @@ Object.defineProperties(PpeTexture.prototype, {
* @memberof PpeTexture.prototype
* @type {number|undefined}
* @readonly
- * @private
*/
texCoord: {
get: function () {
diff --git a/packages/engine/Source/Scene/Model/Extensions/Gpm/Spdcf.js b/packages/engine/Source/Scene/Model/Extensions/Gpm/Spdcf.js
index 7687ed32ed8d..237b34f6ac4e 100644
--- a/packages/engine/Source/Scene/Model/Extensions/Gpm/Spdcf.js
+++ b/packages/engine/Source/Scene/Model/Extensions/Gpm/Spdcf.js
@@ -6,7 +6,6 @@ import Check from "../../../../Core/Check.js";
* Parameters (A, alpha, beta, T) used to describe the correlation decrease
* between points as a function of delta time.
*
- * @private
* @experimental This feature is not final and is subject to change without Cesium's standard deprecation policy.
*/
function Spdcf(options) {
@@ -33,7 +32,6 @@ Object.defineProperties(Spdcf.prototype, {
* @memberof Spdcf.prototype
* @type {number}
* @readonly
- * @private
*/
A: {
get: function () {
@@ -47,7 +45,6 @@ Object.defineProperties(Spdcf.prototype, {
* @memberof Spdcf.prototype
* @type {number}
* @readonly
- * @private
*/
alpha: {
get: function () {
@@ -61,7 +58,6 @@ Object.defineProperties(Spdcf.prototype, {
* @memberof Spdcf.prototype
* @type {number}
* @readonly
- * @private
*/
beta: {
get: function () {
@@ -75,7 +71,6 @@ Object.defineProperties(Spdcf.prototype, {
* @memberof Spdcf.prototype
* @type {number}
* @readonly
- * @private
*/
T: {
get: function () {
diff --git a/packages/engine/Source/Scene/Model/Model.js b/packages/engine/Source/Scene/Model/Model.js
index 0892388a51a6..15474e5df2ea 100644
--- a/packages/engine/Source/Scene/Model/Model.js
+++ b/packages/engine/Source/Scene/Model/Model.js
@@ -106,7 +106,7 @@ import pickModel from "./pickModel.js";
* {@link https://github.com/KhronosGroup/glTF/blob/main/extensions/1.0/Vendor/WEB3D_quantized_attributes/README.md|WEB3D_quantized_attributes}
*
*
- * {@link https://nsgreg.nga.mil/csmwg.jsp|NGA_gpm_local}
+ * {@link https://nsgreg.nga.mil/csmwg.jsp|NGA_gpm_local (experimental)}
*
*
*
@@ -1786,7 +1786,7 @@ Model.prototype.applyArticulations = function () {
* @returns {object|undefined} The object, or `undefined`
* @exception {DeveloperError} The model is not loaded. Use Model.readyEvent or wait for Model.ready to be true.
*
- * @private
+ * @experimental This feature is not final and is subject to change without Cesium's standard deprecation policy.
*/
Model.prototype.getExtension = function (extensionName) {
//>>includeStart('debug', pragmas.debug);
From 72414c50ffc0ce737e96d61a763460ffbd83547a Mon Sep 17 00:00:00 2001
From: Marco Hutter
Date: Sat, 28 Sep 2024 13:52:13 +0200
Subject: [PATCH 31/38] Add links to GPM spec. Updated comments.
---
.../Model/Extensions/Gpm/AnchorPointDirect.js | 5 ++++-
.../Extensions/Gpm/AnchorPointIndirect.js | 3 +++
.../Model/Extensions/Gpm/CorrelationGroup.js | 3 +++
.../Model/Extensions/Gpm/GltfGpmLocal.js | 20 +++++++++++++++++++
.../Gpm/GltfMeshPrimitiveGpmLoader.js | 4 ++--
.../Scene/Model/Extensions/Gpm/PpeMetadata.js | 5 ++++-
.../Scene/Model/Extensions/Gpm/PpeSource.js | 5 ++++-
.../Scene/Model/Extensions/Gpm/PpeTexture.js | 3 +++
.../Scene/Model/Extensions/Gpm/Spdcf.js | 12 +++++++++--
.../Scene/Model/Extensions/Gpm/StorageType.js | 5 ++++-
10 files changed, 57 insertions(+), 8 deletions(-)
diff --git a/packages/engine/Source/Scene/Model/Extensions/Gpm/AnchorPointDirect.js b/packages/engine/Source/Scene/Model/Extensions/Gpm/AnchorPointDirect.js
index 05bc225f9260..430240fee98a 100644
--- a/packages/engine/Source/Scene/Model/Extensions/Gpm/AnchorPointDirect.js
+++ b/packages/engine/Source/Scene/Model/Extensions/Gpm/AnchorPointDirect.js
@@ -1,7 +1,10 @@
import Check from "../../../../Core/Check.js";
/**
- * Metadata for one stored anchor point using direct storage
+ * Metadata for one stored anchor point using direct storage.
+ *
+ * This reflects the `anchronPointDirect` definition of the
+ * {@link https://nsgreg.nga.mil/csmwg.jsp|NGA_gpm_local} glTF extension.
*
* @experimental This feature is not final and is subject to change without Cesium's standard deprecation policy.
*/
diff --git a/packages/engine/Source/Scene/Model/Extensions/Gpm/AnchorPointIndirect.js b/packages/engine/Source/Scene/Model/Extensions/Gpm/AnchorPointIndirect.js
index 03599b48ce8d..7abe2db94046 100644
--- a/packages/engine/Source/Scene/Model/Extensions/Gpm/AnchorPointIndirect.js
+++ b/packages/engine/Source/Scene/Model/Extensions/Gpm/AnchorPointIndirect.js
@@ -3,6 +3,9 @@ import Check from "../../../../Core/Check.js";
/**
* Metadata for one stored anchor point.
*
+ * This reflects the `anchronPointIndirect` definition of the
+ * {@link https://nsgreg.nga.mil/csmwg.jsp|NGA_gpm_local} glTF extension.
+ *
* @experimental This feature is not final and is subject to change without Cesium's standard deprecation policy.
*/
function AnchorPointIndirect(options) {
diff --git a/packages/engine/Source/Scene/Model/Extensions/Gpm/CorrelationGroup.js b/packages/engine/Source/Scene/Model/Extensions/Gpm/CorrelationGroup.js
index 23812c03871b..abf0a4e1262d 100644
--- a/packages/engine/Source/Scene/Model/Extensions/Gpm/CorrelationGroup.js
+++ b/packages/engine/Source/Scene/Model/Extensions/Gpm/CorrelationGroup.js
@@ -4,6 +4,9 @@ import Check from "../../../../Core/Check.js";
* Metadata identifying parameters using same correlation modeling and
* associated correlation parameters.
*
+ * This reflects the `correlationGroup` definition of the
+ * {@link https://nsgreg.nga.mil/csmwg.jsp|NGA_gpm_local} glTF extension.
+ *
* @experimental This feature is not final and is subject to change without Cesium's standard deprecation policy.
*/
function CorrelationGroup(options) {
diff --git a/packages/engine/Source/Scene/Model/Extensions/Gpm/GltfGpmLocal.js b/packages/engine/Source/Scene/Model/Extensions/Gpm/GltfGpmLocal.js
index 73fc5789de13..84ae342bf56b 100644
--- a/packages/engine/Source/Scene/Model/Extensions/Gpm/GltfGpmLocal.js
+++ b/packages/engine/Source/Scene/Model/Extensions/Gpm/GltfGpmLocal.js
@@ -7,6 +7,26 @@ import StorageType from "./StorageType.js";
* The GPM metadata for a Ground-Space Indirect implementation stored
* locally (i.e. a tile and/or leaf node).
*
+ * This reflects the root extension object of the {@link https://nsgreg.nga.mil/csmwg.jsp|NGA_gpm_local}
+ * glTF extension. When a model that contains this extension was loaded,
+ * then an object of this type can be obtained by calling
+ * ```
+ * const gltfGpmLocal = model.getExtension("NGA_gpm_local");
+ * ```
+ *
+ * The storage type determines the presence of the optional properties:
+ *
+ * -
+ * When the storage type is `StorageType.Indirect`, then the
+ * `anchorPointsIndirect` and `intraTileCorrelationGroups`
+ * are present.
+ *
+ * -
+ * When the storage type is `StorageType.Direct`, then the
+ * `anchorPointsDirect` and `covarianceDirect` are present.
+ *
+ *
+ *
* @experimental This feature is not final and is subject to change without Cesium's standard deprecation policy.
*/
function GltfGpmLocal(options) {
diff --git a/packages/engine/Source/Scene/Model/Extensions/Gpm/GltfMeshPrimitiveGpmLoader.js b/packages/engine/Source/Scene/Model/Extensions/Gpm/GltfMeshPrimitiveGpmLoader.js
index b718ff49e504..e16672669756 100644
--- a/packages/engine/Source/Scene/Model/Extensions/Gpm/GltfMeshPrimitiveGpmLoader.js
+++ b/packages/engine/Source/Scene/Model/Extensions/Gpm/GltfMeshPrimitiveGpmLoader.js
@@ -232,7 +232,7 @@ GltfMeshPrimitiveGpmLoader.ppeTexturesMetadataSchemaCache = new Map();
/**
* Create the JSON description of a metadata class that treats
- * the given PPE texture as a property texture property(!).
+ * the given PPE texture as a property texture property.
*
* @param {any} ppeTexture - The PPE texture
* @param {number} index - The index of the texture in the extension
@@ -346,7 +346,7 @@ GltfMeshPrimitiveGpmLoader.obtainPpeTexturesMetadataSchema = function (
*
* Each glTF may define multiple `ppeTexture` objects within the
* `NGA_gpm_local` extensions. Each of these textures corresponds
- * to one 'property texture property(!)' in a metadata schema.
+ * to one 'property texture property' in a metadata schema.
*
* This method will create an array where each element is a (JSON)
* string representation of the parts of a GPM PPE texture definition
diff --git a/packages/engine/Source/Scene/Model/Extensions/Gpm/PpeMetadata.js b/packages/engine/Source/Scene/Model/Extensions/Gpm/PpeMetadata.js
index 457779cf4b40..46949992a3dc 100644
--- a/packages/engine/Source/Scene/Model/Extensions/Gpm/PpeMetadata.js
+++ b/packages/engine/Source/Scene/Model/Extensions/Gpm/PpeMetadata.js
@@ -1,7 +1,10 @@
import Check from "../../../../Core/Check.js";
/**
- * Metadata related to the stored PPE data.
+ * Metadata related to the stored PPE (Per-Point Error) data.
+ *
+ * This reflects the `ppeMetadata` definition of the
+ * {@link https://nsgreg.nga.mil/csmwg.jsp|NGA_gpm_local} glTF extension.
*
* @experimental This feature is not final and is subject to change without Cesium's standard deprecation policy.
*/
diff --git a/packages/engine/Source/Scene/Model/Extensions/Gpm/PpeSource.js b/packages/engine/Source/Scene/Model/Extensions/Gpm/PpeSource.js
index d44acabe4e1c..db0f3676f278 100644
--- a/packages/engine/Source/Scene/Model/Extensions/Gpm/PpeSource.js
+++ b/packages/engine/Source/Scene/Model/Extensions/Gpm/PpeSource.js
@@ -1,5 +1,8 @@
/**
- * An enum of per-point error sources
+ * An enum of per-point error sources.
+ *
+ * This reflects the `ppeMetadata.source` definition of the
+ * {@link https://nsgreg.nga.mil/csmwg.jsp|NGA_gpm_local} glTF extension.
*
* @enum {string}
* @experimental This feature is not final and is subject to change without Cesium's standard deprecation policy.
diff --git a/packages/engine/Source/Scene/Model/Extensions/Gpm/PpeTexture.js b/packages/engine/Source/Scene/Model/Extensions/Gpm/PpeTexture.js
index f8b8109f773d..216764da2999 100644
--- a/packages/engine/Source/Scene/Model/Extensions/Gpm/PpeTexture.js
+++ b/packages/engine/Source/Scene/Model/Extensions/Gpm/PpeTexture.js
@@ -3,6 +3,9 @@ import Check from "../../../../Core/Check.js";
/**
* PPE (Per-Point Error) texture in `NGA_gpm_local`.
*
+ * This reflects the `ppeTexture` definition of the
+ * {@link https://nsgreg.nga.mil/csmwg.jsp|NGA_gpm_local} glTF extension.
+ *
* This is a valid glTF `TextureInfo` object (with a required `index`
* and an optional `texCoord)`, with additional properties that
* describe the structure of the metdata that is stored in the texture.
diff --git a/packages/engine/Source/Scene/Model/Extensions/Gpm/Spdcf.js b/packages/engine/Source/Scene/Model/Extensions/Gpm/Spdcf.js
index 237b34f6ac4e..74ae2c607c80 100644
--- a/packages/engine/Source/Scene/Model/Extensions/Gpm/Spdcf.js
+++ b/packages/engine/Source/Scene/Model/Extensions/Gpm/Spdcf.js
@@ -3,8 +3,16 @@ import Check from "../../../../Core/Check.js";
/**
* Variables for a Strictly Positive-Definite Correlation Function.
*
- * Parameters (A, alpha, beta, T) used to describe the correlation decrease
- * between points as a function of delta time.
+ * This reflects the `spdcf` definition of the
+ * {@link https://nsgreg.nga.mil/csmwg.jsp|NGA_gpm_local} glTF extension.
+ * Instances of this type are stored as the parameters within a
+ * `CorrelationGroup`.
+ *
+ * Parameters (A, alpha, beta, T) describe the correlation decrease
+ * between points as a function of delta time:
+ * ```
+ * spdcf(delta_t) = A_t * (alpha_t + ((1 - alpha_t)(1 + beta_t)) / (beta_t + e^(delta_t/T_t)))
+ * ```
*
* @experimental This feature is not final and is subject to change without Cesium's standard deprecation policy.
*/
diff --git a/packages/engine/Source/Scene/Model/Extensions/Gpm/StorageType.js b/packages/engine/Source/Scene/Model/Extensions/Gpm/StorageType.js
index f706c32b63e1..94a864b6be38 100644
--- a/packages/engine/Source/Scene/Model/Extensions/Gpm/StorageType.js
+++ b/packages/engine/Source/Scene/Model/Extensions/Gpm/StorageType.js
@@ -1,5 +1,8 @@
/**
- * An enum of storage types for covariance information
+ * An enum of storage types for covariance information.
+ *
+ * This reflects the `gltfGpmLocal.storageType` definition of the
+ * {@link https://nsgreg.nga.mil/csmwg.jsp|NGA_gpm_local} glTF extension.
*
* @enum {string}
* @experimental This feature is not final and is subject to change without Cesium's standard deprecation policy.
From 55df6c9b8fe2d145f66061bd0546e3b9931a914b Mon Sep 17 00:00:00 2001
From: Marco Hutter
Date: Sat, 28 Sep 2024 14:29:38 +0200
Subject: [PATCH 32/38] Document constructor options
---
.../Model/Extensions/Gpm/GltfGpmLocal.js | 19 +++++++++++++++++++
.../Scene/Model/Extensions/Gpm/PpeMetadata.js | 12 ++++++++++++
.../Scene/Model/Extensions/Gpm/PpeTexture.js | 15 +++++++++++++++
3 files changed, 46 insertions(+)
diff --git a/packages/engine/Source/Scene/Model/Extensions/Gpm/GltfGpmLocal.js b/packages/engine/Source/Scene/Model/Extensions/Gpm/GltfGpmLocal.js
index 84ae342bf56b..f58aa27fd361 100644
--- a/packages/engine/Source/Scene/Model/Extensions/Gpm/GltfGpmLocal.js
+++ b/packages/engine/Source/Scene/Model/Extensions/Gpm/GltfGpmLocal.js
@@ -3,6 +3,23 @@ import Check from "../../../../Core/Check.js";
import RuntimeError from "../../../../Core/RuntimeError.js";
import StorageType from "./StorageType.js";
+/**
+ * @typedef {object} GltfGpmLocal.ConstructorOptions
+ *
+ * Initialization options for the GltfGpmLocal constructor
+ *
+ * @property {string} storageType The storage type.
+ * This must be one of the `StorageType` constants, i.e. `Direct` or `Indirect`.
+ * @property {AnchorPointIndirect[]|undefined} [anchorPointsIndirect] The indirect anchor points.
+ * This must be present if and only if the storage type is `Indirect`.
+ * @property {CorrelationGroup[]|undefined} [intraTileCorrelationGroups] The intra-tile correlation groups.
+ * This must be present if and only if the storage type is `Indirect`.
+ * @property {AnchorPointDirect[]|undefined} [anchorPointsDirect] The direct anchor points.
+ * This must be present if and only if the storage type is `Direct`.
+ * @property {Matrix3|undefined} [covarianceDirect] The covariance of anchor point parameters.
+ * This must be present if and only if the storage type is `Direct`.
+ */
+
/**
* The GPM metadata for a Ground-Space Indirect implementation stored
* locally (i.e. a tile and/or leaf node).
@@ -27,6 +44,8 @@ import StorageType from "./StorageType.js";
*
*
*
+ * @param {GltfGpmLocal.ConstructorOptions} options An object describing initialization options
+ *
* @experimental This feature is not final and is subject to change without Cesium's standard deprecation policy.
*/
function GltfGpmLocal(options) {
diff --git a/packages/engine/Source/Scene/Model/Extensions/Gpm/PpeMetadata.js b/packages/engine/Source/Scene/Model/Extensions/Gpm/PpeMetadata.js
index 46949992a3dc..88bcb70ebee9 100644
--- a/packages/engine/Source/Scene/Model/Extensions/Gpm/PpeMetadata.js
+++ b/packages/engine/Source/Scene/Model/Extensions/Gpm/PpeMetadata.js
@@ -1,11 +1,23 @@
import Check from "../../../../Core/Check.js";
+/**
+ * @typedef {object} PpeMetadata.ConstructorOptions
+ *
+ * Initialization options for the PpeMetadata constructor
+ *
+ * @property {PpeSource} source The source of the error data
+ * @property {number|undefined} [min] Minimum allowed value for the property.
+ * @property {number|undefined} [max] Maximum allowed value for the property.
+ */
+
/**
* Metadata related to the stored PPE (Per-Point Error) data.
*
* This reflects the `ppeMetadata` definition of the
* {@link https://nsgreg.nga.mil/csmwg.jsp|NGA_gpm_local} glTF extension.
*
+ * @param {PpeMetadata.ConstructorOptions} options An object describing initialization options
+ *
* @experimental This feature is not final and is subject to change without Cesium's standard deprecation policy.
*/
function PpeMetadata(options) {
diff --git a/packages/engine/Source/Scene/Model/Extensions/Gpm/PpeTexture.js b/packages/engine/Source/Scene/Model/Extensions/Gpm/PpeTexture.js
index 216764da2999..7588fd2a4ed0 100644
--- a/packages/engine/Source/Scene/Model/Extensions/Gpm/PpeTexture.js
+++ b/packages/engine/Source/Scene/Model/Extensions/Gpm/PpeTexture.js
@@ -1,5 +1,18 @@
import Check from "../../../../Core/Check.js";
+/**
+ * @typedef {object} PpeTexture.ConstructorOptions
+ *
+ * Initialization options for the PpeTexture constructor
+ *
+ * @property {PpeMetadata} traits The traits that indicate which data is stored in this texture
+ * @property {number} index The index of the texture inside the glTF textures array
+ * @property {number|undefined} [texCoord] The optional set index for the TEXCOORD attribute
+ * @property {number|undefined} [noData] The value to represent missing data
+ * @property {number|undefined} [offset] An offset to apply to property values.
+ * @property {number|undefined} [scale] A scale to apply to property values.
+ */
+
/**
* PPE (Per-Point Error) texture in `NGA_gpm_local`.
*
@@ -10,6 +23,8 @@ import Check from "../../../../Core/Check.js";
* and an optional `texCoord)`, with additional properties that
* describe the structure of the metdata that is stored in the texture.
*
+ * @param {PpeTexture.ConstructorOptions} options An object describing initialization options
+ *
* @experimental This feature is not final and is subject to change without Cesium's standard deprecation policy.
*/
function PpeTexture(options) {
From b8b5445903bb9fb9b3a9d4c3a55f58dfa38bb47c Mon Sep 17 00:00:00 2001
From: Marco Hutter
Date: Sat, 28 Sep 2024 14:30:02 +0200
Subject: [PATCH 33/38] Avoid unnecessary constructor options
---
.../Scene/Model/Extensions/Gpm/MeshPrimitiveGpmLocal.js | 8 +++++---
1 file changed, 5 insertions(+), 3 deletions(-)
diff --git a/packages/engine/Source/Scene/Model/Extensions/Gpm/MeshPrimitiveGpmLocal.js b/packages/engine/Source/Scene/Model/Extensions/Gpm/MeshPrimitiveGpmLocal.js
index 5cdf62f37943..c8b76f25b0de 100644
--- a/packages/engine/Source/Scene/Model/Extensions/Gpm/MeshPrimitiveGpmLocal.js
+++ b/packages/engine/Source/Scene/Model/Extensions/Gpm/MeshPrimitiveGpmLocal.js
@@ -1,10 +1,12 @@
/**
* Local Generic Point-cloud Model information about a glTF primitive.
*
- * @experimental This feature is not final and is subject to change without Cesium's standard deprecation policy.
+ * @param {PpeTexture[]} ppeTextures The Per-Point Error textures
+ *
+ * @private
*/
-function MeshPrimitiveGpmLocal(options) {
- this._ppeTextures = options.ppeTextures;
+function MeshPrimitiveGpmLocal(ppeTextures) {
+ this._ppeTextures = ppeTextures;
}
Object.defineProperties(MeshPrimitiveGpmLocal.prototype, {
From e4efa515554dcb2b203bf7bd00e9114711218ab2 Mon Sep 17 00:00:00 2001
From: Marco Hutter
Date: Sat, 28 Sep 2024 14:30:23 +0200
Subject: [PATCH 34/38] Add underscore to internal functions
---
.../Gpm/GltfMeshPrimitiveGpmLoader.js | 38 +++++++++----------
1 file changed, 18 insertions(+), 20 deletions(-)
diff --git a/packages/engine/Source/Scene/Model/Extensions/Gpm/GltfMeshPrimitiveGpmLoader.js b/packages/engine/Source/Scene/Model/Extensions/Gpm/GltfMeshPrimitiveGpmLoader.js
index e16672669756..7be022144311 100644
--- a/packages/engine/Source/Scene/Model/Extensions/Gpm/GltfMeshPrimitiveGpmLoader.js
+++ b/packages/engine/Source/Scene/Model/Extensions/Gpm/GltfMeshPrimitiveGpmLoader.js
@@ -131,9 +131,9 @@ Object.defineProperties(GltfMeshPrimitiveGpmLoader.prototype, {
},
});
-GltfMeshPrimitiveGpmLoader.prototype.loadResources = async function () {
+GltfMeshPrimitiveGpmLoader.prototype._loadResources = async function () {
try {
- const texturesPromise = this.loadTextures();
+ const texturesPromise = this._loadTextures();
await texturesPromise;
if (this.isDestroyed()) {
@@ -167,7 +167,7 @@ GltfMeshPrimitiveGpmLoader.prototype.load = function () {
}
this._state = ResourceLoaderState.LOADING;
- this._promise = this.loadResources(this);
+ this._promise = this._loadResources(this);
return this._promise;
};
@@ -185,7 +185,7 @@ function gatherUsedTextureIds(gpmExtension) {
return textureIds;
}
-GltfMeshPrimitiveGpmLoader.prototype.loadTextures = function () {
+GltfMeshPrimitiveGpmLoader.prototype._loadTextures = function () {
let textureIds;
if (defined(this._extension)) {
textureIds = gatherUsedTextureIds(this._extension);
@@ -224,7 +224,7 @@ GltfMeshPrimitiveGpmLoader.prototype.loadTextures = function () {
* A static mapping from PPE texture property identifier keys
* to `MetadataSchema` instances. This is used to create each
* schema (with a certain structure) only ONCE in
- * obtainPpeTexturesMetadataSchema
+ * _obtainPpeTexturesMetadataSchema
*
* @private
*/
@@ -238,7 +238,7 @@ GltfMeshPrimitiveGpmLoader.ppeTexturesMetadataSchemaCache = new Map();
* @param {number} index - The index of the texture in the extension
* @returns The class JSON
*/
-GltfMeshPrimitiveGpmLoader.createPpeTextureClassJson = function (
+GltfMeshPrimitiveGpmLoader._createPpeTextureClassJson = function (
ppeTexture,
index,
) {
@@ -295,16 +295,16 @@ GltfMeshPrimitiveGpmLoader.createPpeTextureClassJson = function (
* schema that reflects the structure of the PPE textures in the
* given instance, creating and caching it if necessary.
*
- * For details on the cache key, see `collectPpeTexturePropertyIdentifiers`
+ * For details on the cache key, see `_collectPpeTexturePropertyIdentifiers`
*
* @param {MeshPrimitiveGpmLocal} meshPrimitiveGpmLocal The extension object
* @returns The `MetadataSchema`
*/
-GltfMeshPrimitiveGpmLoader.obtainPpeTexturesMetadataSchema = function (
+GltfMeshPrimitiveGpmLoader._obtainPpeTexturesMetadataSchema = function (
meshPrimitiveGpmLocal,
) {
const ppeTexturePropertyIdentifiers =
- GltfMeshPrimitiveGpmLoader.collectPpeTexturePropertyIdentifiers(
+ GltfMeshPrimitiveGpmLoader._collectPpeTexturePropertyIdentifiers(
meshPrimitiveGpmLocal,
);
const key = ppeTexturePropertyIdentifiers.toString();
@@ -324,7 +324,7 @@ GltfMeshPrimitiveGpmLoader.obtainPpeTexturesMetadataSchema = function (
for (let i = 0; i < ppeTextures.length; i++) {
const ppeTexture = ppeTextures[i];
const classId = `ppeTexture_${i}`;
- const classJson = GltfMeshPrimitiveGpmLoader.createPpeTextureClassJson(
+ const classJson = GltfMeshPrimitiveGpmLoader._createPpeTextureClassJson(
ppeTexture,
i,
);
@@ -356,7 +356,7 @@ GltfMeshPrimitiveGpmLoader.obtainPpeTexturesMetadataSchema = function (
* @param {MeshPrimitiveGpmLocal} meshPrimitiveGpmLocal The extension object
* @returns The identifiers
*/
-GltfMeshPrimitiveGpmLoader.collectPpeTexturePropertyIdentifiers = function (
+GltfMeshPrimitiveGpmLoader._collectPpeTexturePropertyIdentifiers = function (
meshPrimitiveGpmLocal,
) {
const ppeTexturePropertyIdentifiers = [];
@@ -367,7 +367,7 @@ GltfMeshPrimitiveGpmLoader.collectPpeTexturePropertyIdentifiers = function (
// to define two PPE textures as "representing the same
// property texture property" within a structural metadata
// schema.
- const classJson = GltfMeshPrimitiveGpmLoader.createPpeTextureClassJson(
+ const classJson = GltfMeshPrimitiveGpmLoader._createPpeTextureClassJson(
ppeTexture,
i,
);
@@ -389,13 +389,13 @@ GltfMeshPrimitiveGpmLoader.collectPpeTexturePropertyIdentifiers = function (
* @param {object} textures The mapping from texture ID to texture objects
* @returns The `StructuralMetadata` object
*/
-GltfMeshPrimitiveGpmLoader.convertToStructuralMetadata = function (
+GltfMeshPrimitiveGpmLoader._convertToStructuralMetadata = function (
meshPrimitiveGpmLocal,
textures,
) {
const propertyTextures = [];
const ppeTexturesMetadataSchema =
- GltfMeshPrimitiveGpmLoader.obtainPpeTexturesMetadataSchema(
+ GltfMeshPrimitiveGpmLoader._obtainPpeTexturesMetadataSchema(
meshPrimitiveGpmLocal,
);
const ppeTextures = meshPrimitiveGpmLocal.ppeTextures;
@@ -510,13 +510,11 @@ GltfMeshPrimitiveGpmLoader.prototype.process = function (frameState) {
ppeTextures.push(ppeTexture);
}
}
- const meshPrimitiveGpmLocal = new MeshPrimitiveGpmLocal({
- ppeTextures: ppeTextures,
- });
+ const meshPrimitiveGpmLocal = new MeshPrimitiveGpmLocal(ppeTextures);
this._meshPrimitiveGpmLocal = meshPrimitiveGpmLocal;
const structuralMetadata =
- GltfMeshPrimitiveGpmLoader.convertToStructuralMetadata(
+ GltfMeshPrimitiveGpmLoader._convertToStructuralMetadata(
meshPrimitiveGpmLocal,
textures,
);
@@ -526,7 +524,7 @@ GltfMeshPrimitiveGpmLoader.prototype.process = function (frameState) {
return true;
};
-GltfMeshPrimitiveGpmLoader.prototype.unloadTextures = function () {
+GltfMeshPrimitiveGpmLoader.prototype._unloadTextures = function () {
const textureLoaders = this._textureLoaders;
const textureLoadersLength = textureLoaders.length;
for (let i = 0; i < textureLoadersLength; ++i) {
@@ -541,7 +539,7 @@ GltfMeshPrimitiveGpmLoader.prototype.unloadTextures = function () {
* @private
*/
GltfMeshPrimitiveGpmLoader.prototype.unload = function () {
- this.unloadTextures();
+ this._unloadTextures();
this._gltf = undefined;
this._extension = undefined;
this._structuralMetadata = undefined;
From 860b05b25271607755173654600974b3ee9065f6 Mon Sep 17 00:00:00 2001
From: Marco Hutter
Date: Sat, 28 Sep 2024 14:39:53 +0200
Subject: [PATCH 35/38] Further clarification for private
---
.../engine/Source/Scene/Model/Extensions/Gpm/PpeMetadata.js | 2 +-
packages/engine/Source/Scene/Model/Extensions/Gpm/PpeSource.js | 2 +-
packages/engine/Source/Scene/Model/Extensions/Gpm/PpeTexture.js | 2 +-
3 files changed, 3 insertions(+), 3 deletions(-)
diff --git a/packages/engine/Source/Scene/Model/Extensions/Gpm/PpeMetadata.js b/packages/engine/Source/Scene/Model/Extensions/Gpm/PpeMetadata.js
index 88bcb70ebee9..e5289c2ef94f 100644
--- a/packages/engine/Source/Scene/Model/Extensions/Gpm/PpeMetadata.js
+++ b/packages/engine/Source/Scene/Model/Extensions/Gpm/PpeMetadata.js
@@ -18,7 +18,7 @@ import Check from "../../../../Core/Check.js";
*
* @param {PpeMetadata.ConstructorOptions} options An object describing initialization options
*
- * @experimental This feature is not final and is subject to change without Cesium's standard deprecation policy.
+ * @private
*/
function PpeMetadata(options) {
//>>includeStart('debug', pragmas.debug);
diff --git a/packages/engine/Source/Scene/Model/Extensions/Gpm/PpeSource.js b/packages/engine/Source/Scene/Model/Extensions/Gpm/PpeSource.js
index db0f3676f278..92d1bbedd815 100644
--- a/packages/engine/Source/Scene/Model/Extensions/Gpm/PpeSource.js
+++ b/packages/engine/Source/Scene/Model/Extensions/Gpm/PpeSource.js
@@ -5,7 +5,7 @@
* {@link https://nsgreg.nga.mil/csmwg.jsp|NGA_gpm_local} glTF extension.
*
* @enum {string}
- * @experimental This feature is not final and is subject to change without Cesium's standard deprecation policy.
+ * @private
*/
const PpeSource = {
/**
diff --git a/packages/engine/Source/Scene/Model/Extensions/Gpm/PpeTexture.js b/packages/engine/Source/Scene/Model/Extensions/Gpm/PpeTexture.js
index 7588fd2a4ed0..24fac262afdf 100644
--- a/packages/engine/Source/Scene/Model/Extensions/Gpm/PpeTexture.js
+++ b/packages/engine/Source/Scene/Model/Extensions/Gpm/PpeTexture.js
@@ -25,7 +25,7 @@ import Check from "../../../../Core/Check.js";
*
* @param {PpeTexture.ConstructorOptions} options An object describing initialization options
*
- * @experimental This feature is not final and is subject to change without Cesium's standard deprecation policy.
+ * @private
*/
function PpeTexture(options) {
//>>includeStart('debug', pragmas.debug);
From 9bfe61afc1827a6ad9032138f5ffdcc8cbe77d38 Mon Sep 17 00:00:00 2001
From: Marco Hutter
Date: Sat, 28 Sep 2024 15:13:16 +0200
Subject: [PATCH 36/38] Update spec for changed error type
---
.../Specs/Scene/Model/Extensions/Gpm/GltfGpmLoaderSpec.js | 6 +++---
1 file changed, 3 insertions(+), 3 deletions(-)
diff --git a/packages/engine/Specs/Scene/Model/Extensions/Gpm/GltfGpmLoaderSpec.js b/packages/engine/Specs/Scene/Model/Extensions/Gpm/GltfGpmLoaderSpec.js
index e6f35b16e09e..ddaf8ddfb390 100644
--- a/packages/engine/Specs/Scene/Model/Extensions/Gpm/GltfGpmLoaderSpec.js
+++ b/packages/engine/Specs/Scene/Model/Extensions/Gpm/GltfGpmLoaderSpec.js
@@ -12,7 +12,7 @@ describe("Scene/Model/Extensions/Gpm/GltfGpmLoader", function () {
};
expect(function () {
GltfGpmLoader.load(gltfGpmLocalJson);
- }).toThrowDeveloperError();
+ }).toThrowError();
});
it("load throws for storageType Direct without anchorPointsDirect", async function () {
@@ -21,7 +21,7 @@ describe("Scene/Model/Extensions/Gpm/GltfGpmLoader", function () {
};
expect(function () {
GltfGpmLoader.load(gltfGpmLocalJson);
- }).toThrowDeveloperError();
+ }).toThrowError();
});
it("load throws for storageType Direct without covarianceDirectUpperTriangle", async function () {
@@ -36,7 +36,7 @@ describe("Scene/Model/Extensions/Gpm/GltfGpmLoader", function () {
};
expect(function () {
GltfGpmLoader.load(gltfGpmLocalJson);
- }).toThrowDeveloperError();
+ }).toThrowError();
});
it("load returns result for valid JSON for storageType Direct", async function () {
From 9805652f03e8ad24895a78929713e1bfd8b8710b Mon Sep 17 00:00:00 2001
From: Marco Hutter
Date: Sat, 28 Sep 2024 15:13:37 +0200
Subject: [PATCH 37/38] Further options- and constructor documentation
---
.../Model/Extensions/Gpm/AnchorPointDirect.js | 11 +++++++++++
.../Model/Extensions/Gpm/AnchorPointIndirect.js | 12 ++++++++++++
.../Model/Extensions/Gpm/CorrelationGroup.js | 15 +++++++++++++++
.../Scene/Model/Extensions/Gpm/GltfGpmLocal.js | 1 +
.../Model/Extensions/Gpm/MeshPrimitiveGpmLocal.js | 1 +
.../Scene/Model/Extensions/Gpm/PpeMetadata.js | 1 +
.../Scene/Model/Extensions/Gpm/PpeTexture.js | 1 +
.../Source/Scene/Model/Extensions/Gpm/Spdcf.js | 13 +++++++++++++
8 files changed, 55 insertions(+)
diff --git a/packages/engine/Source/Scene/Model/Extensions/Gpm/AnchorPointDirect.js b/packages/engine/Source/Scene/Model/Extensions/Gpm/AnchorPointDirect.js
index 430240fee98a..7f20273f7a90 100644
--- a/packages/engine/Source/Scene/Model/Extensions/Gpm/AnchorPointDirect.js
+++ b/packages/engine/Source/Scene/Model/Extensions/Gpm/AnchorPointDirect.js
@@ -1,11 +1,22 @@
import Check from "../../../../Core/Check.js";
+/**
+ * @typedef {object} AnchorPointDirect.ConstructorOptions
+ *
+ * Initialization options for the AnchorPointDirect constructor
+ *
+ * @property {Cartesian3} position Anchor point geographic coordinates
+ * @property {Cartesian3} adjustmentParams The adjustment values in meters
+ */
+
/**
* Metadata for one stored anchor point using direct storage.
*
* This reflects the `anchronPointDirect` definition of the
* {@link https://nsgreg.nga.mil/csmwg.jsp|NGA_gpm_local} glTF extension.
*
+ * @constructor
+ * @param {AnchorPointDirect.ConstructorOptions} options An object describing initialization options
* @experimental This feature is not final and is subject to change without Cesium's standard deprecation policy.
*/
function AnchorPointDirect(options) {
diff --git a/packages/engine/Source/Scene/Model/Extensions/Gpm/AnchorPointIndirect.js b/packages/engine/Source/Scene/Model/Extensions/Gpm/AnchorPointIndirect.js
index 7abe2db94046..b7f368b6d859 100644
--- a/packages/engine/Source/Scene/Model/Extensions/Gpm/AnchorPointIndirect.js
+++ b/packages/engine/Source/Scene/Model/Extensions/Gpm/AnchorPointIndirect.js
@@ -1,11 +1,23 @@
import Check from "../../../../Core/Check.js";
+/**
+ * @typedef {object} AnchorPointIndirect.ConstructorOptions
+ *
+ * Initialization options for the AnchorPointIndirect constructor
+ *
+ * @property {Cartesian3} position Anchor point geographic coordinates
+ * @property {Cartesian3} adjustmentParams The adjustment values in meters
+ * @property {Matrix3} covarianceMatrix The 3x3 covariance matrix
+ */
+
/**
* Metadata for one stored anchor point.
*
* This reflects the `anchronPointIndirect` definition of the
* {@link https://nsgreg.nga.mil/csmwg.jsp|NGA_gpm_local} glTF extension.
*
+ * @constructor
+ * @param {AnchorPointIndirect.ConstructorOptions} options An object describing initialization options
* @experimental This feature is not final and is subject to change without Cesium's standard deprecation policy.
*/
function AnchorPointIndirect(options) {
diff --git a/packages/engine/Source/Scene/Model/Extensions/Gpm/CorrelationGroup.js b/packages/engine/Source/Scene/Model/Extensions/Gpm/CorrelationGroup.js
index abf0a4e1262d..919ef9ed9991 100644
--- a/packages/engine/Source/Scene/Model/Extensions/Gpm/CorrelationGroup.js
+++ b/packages/engine/Source/Scene/Model/Extensions/Gpm/CorrelationGroup.js
@@ -1,5 +1,18 @@
import Check from "../../../../Core/Check.js";
+/**
+ * @typedef {object} CorrelationGroup.ConstructorOptions
+ *
+ * Initialization options for the CorrelationGroup constructor
+ *
+ * @property {boolean[]} groupFlags Array of 3 booleans indicating if
+ * parameters delta-x delta-y delta-z used in the correlation group
+ * @property {Cartesian3} rotationThetas Rotations in milliradians
+ * about X, Y, Z axes, respectively
+ * @property {Spdcf[]} params Array of `Spdcf` (Strictly Positive-Definite
+ * Correlation Function) parameters, for the U, V, W directions, respectively
+ */
+
/**
* Metadata identifying parameters using same correlation modeling and
* associated correlation parameters.
@@ -7,6 +20,8 @@ import Check from "../../../../Core/Check.js";
* This reflects the `correlationGroup` definition of the
* {@link https://nsgreg.nga.mil/csmwg.jsp|NGA_gpm_local} glTF extension.
*
+ * @constructor
+ * @param {CorrelationGroup.ConstructorOptions} options An object describing initialization options
* @experimental This feature is not final and is subject to change without Cesium's standard deprecation policy.
*/
function CorrelationGroup(options) {
diff --git a/packages/engine/Source/Scene/Model/Extensions/Gpm/GltfGpmLocal.js b/packages/engine/Source/Scene/Model/Extensions/Gpm/GltfGpmLocal.js
index f58aa27fd361..80339c2247e6 100644
--- a/packages/engine/Source/Scene/Model/Extensions/Gpm/GltfGpmLocal.js
+++ b/packages/engine/Source/Scene/Model/Extensions/Gpm/GltfGpmLocal.js
@@ -44,6 +44,7 @@ import StorageType from "./StorageType.js";
*
*
*
+ * @constructor
* @param {GltfGpmLocal.ConstructorOptions} options An object describing initialization options
*
* @experimental This feature is not final and is subject to change without Cesium's standard deprecation policy.
diff --git a/packages/engine/Source/Scene/Model/Extensions/Gpm/MeshPrimitiveGpmLocal.js b/packages/engine/Source/Scene/Model/Extensions/Gpm/MeshPrimitiveGpmLocal.js
index c8b76f25b0de..aab7130e859f 100644
--- a/packages/engine/Source/Scene/Model/Extensions/Gpm/MeshPrimitiveGpmLocal.js
+++ b/packages/engine/Source/Scene/Model/Extensions/Gpm/MeshPrimitiveGpmLocal.js
@@ -3,6 +3,7 @@
*
* @param {PpeTexture[]} ppeTextures The Per-Point Error textures
*
+ * @constructor
* @private
*/
function MeshPrimitiveGpmLocal(ppeTextures) {
diff --git a/packages/engine/Source/Scene/Model/Extensions/Gpm/PpeMetadata.js b/packages/engine/Source/Scene/Model/Extensions/Gpm/PpeMetadata.js
index e5289c2ef94f..d7c077fe603e 100644
--- a/packages/engine/Source/Scene/Model/Extensions/Gpm/PpeMetadata.js
+++ b/packages/engine/Source/Scene/Model/Extensions/Gpm/PpeMetadata.js
@@ -16,6 +16,7 @@ import Check from "../../../../Core/Check.js";
* This reflects the `ppeMetadata` definition of the
* {@link https://nsgreg.nga.mil/csmwg.jsp|NGA_gpm_local} glTF extension.
*
+ * @constructor
* @param {PpeMetadata.ConstructorOptions} options An object describing initialization options
*
* @private
diff --git a/packages/engine/Source/Scene/Model/Extensions/Gpm/PpeTexture.js b/packages/engine/Source/Scene/Model/Extensions/Gpm/PpeTexture.js
index 24fac262afdf..e81d7605448f 100644
--- a/packages/engine/Source/Scene/Model/Extensions/Gpm/PpeTexture.js
+++ b/packages/engine/Source/Scene/Model/Extensions/Gpm/PpeTexture.js
@@ -23,6 +23,7 @@ import Check from "../../../../Core/Check.js";
* and an optional `texCoord)`, with additional properties that
* describe the structure of the metdata that is stored in the texture.
*
+ * @constructor
* @param {PpeTexture.ConstructorOptions} options An object describing initialization options
*
* @private
diff --git a/packages/engine/Source/Scene/Model/Extensions/Gpm/Spdcf.js b/packages/engine/Source/Scene/Model/Extensions/Gpm/Spdcf.js
index 74ae2c607c80..1cc01a96b536 100644
--- a/packages/engine/Source/Scene/Model/Extensions/Gpm/Spdcf.js
+++ b/packages/engine/Source/Scene/Model/Extensions/Gpm/Spdcf.js
@@ -1,5 +1,16 @@
import Check from "../../../../Core/Check.js";
+/**
+ * @typedef {object} Spdcf.ConstructorOptions
+ *
+ * Initialization options for the Spdcf constructor
+ *
+ * @property {number} A The factor A, in (0, 1]
+ * @property {number} alpha The alpha value, in [0, 1)
+ * @property {number} beta The beta value, in [0, 10]
+ * @property {number} T the tau value, in (0, +inf)
+ */
+
/**
* Variables for a Strictly Positive-Definite Correlation Function.
*
@@ -14,6 +25,8 @@ import Check from "../../../../Core/Check.js";
* spdcf(delta_t) = A_t * (alpha_t + ((1 - alpha_t)(1 + beta_t)) / (beta_t + e^(delta_t/T_t)))
* ```
*
+ * @constructor
+ * @param {Spdcf.ConstructorOptions} options An object describing initialization options
* @experimental This feature is not final and is subject to change without Cesium's standard deprecation policy.
*/
function Spdcf(options) {
From c9100904dc03183ee37fe51b4d3f41cce99f76bf Mon Sep 17 00:00:00 2001
From: Marco Hutter
Date: Mon, 30 Sep 2024 16:03:43 +0200
Subject: [PATCH 38/38] Try to revert undesired formatting changes
---
.../3D Tiles Point Cloud Classification.html | 3 +-
.../gallery/3D Tiles Point Cloud Shading.html | 3 +-
.../3D Tiles Terrain Classification.html | 3 +-
.../Custom Per-Feature Post Process.html | 2 +-
.../gallery/Custom Post Process.html | 34 +++++-----
Apps/Sandcastle/gallery/Fog Post Process.html | 68 +++++++++----------
.../gallery/Google Earth Enterprise.html | 2 +-
Apps/Sandcastle/gallery/Materials.html | 22 +++---
.../Scene/ArcGisMapServerImageryProvider.js | 6 +-
.../Model/PrimitiveOutlineGeneratorSpec.js | 2 +-
.../Specs/Scene/Vector3DTilePointsSpec.js | 2 +-
11 files changed, 75 insertions(+), 72 deletions(-)
diff --git a/Apps/Sandcastle/gallery/3D Tiles Point Cloud Classification.html b/Apps/Sandcastle/gallery/3D Tiles Point Cloud Classification.html
index 13b15e133c6d..fedb3b12bb33 100644
--- a/Apps/Sandcastle/gallery/3D Tiles Point Cloud Classification.html
+++ b/Apps/Sandcastle/gallery/3D Tiles Point Cloud Classification.html
@@ -108,7 +108,8 @@
highlighted.feature = pickedFeature;
Cesium.Color.clone(pickedFeature.color, highlighted.originalColor);
pickedFeature.color = Cesium.Color.YELLOW.withAlpha(0.5);
- }, Cesium.ScreenSpaceEventType.MOUSE_MOVE); //Sandcastle_End
+ }, Cesium.ScreenSpaceEventType.MOUSE_MOVE);
+ //Sandcastle_End
};
if (typeof Cesium !== "undefined") {
window.startupCalled = true;
diff --git a/Apps/Sandcastle/gallery/3D Tiles Point Cloud Shading.html b/Apps/Sandcastle/gallery/3D Tiles Point Cloud Shading.html
index 2241bd0ea04e..3d3dc2a6c10b 100644
--- a/Apps/Sandcastle/gallery/3D Tiles Point Cloud Shading.html
+++ b/Apps/Sandcastle/gallery/3D Tiles Point Cloud Shading.html
@@ -345,7 +345,8 @@
if (Cesium.defined(viewModelTileset)) {
viewModelTileset.pointCloudShading.eyeDomeLighting = checked;
}
- }); //Sandcastle_End
+ });
+ //Sandcastle_End
};
if (typeof Cesium !== "undefined") {
window.startupCalled = true;
diff --git a/Apps/Sandcastle/gallery/3D Tiles Terrain Classification.html b/Apps/Sandcastle/gallery/3D Tiles Terrain Classification.html
index c336e6899595..581814432c70 100644
--- a/Apps/Sandcastle/gallery/3D Tiles Terrain Classification.html
+++ b/Apps/Sandcastle/gallery/3D Tiles Terrain Classification.html
@@ -80,7 +80,8 @@
highlighted.feature = pickedFeature;
Cesium.Color.clone(pickedFeature.color, highlighted.originalColor);
pickedFeature.color = Cesium.Color.YELLOW;
- }, Cesium.ScreenSpaceEventType.MOUSE_MOVE); //Sandcastle_End
+ }, Cesium.ScreenSpaceEventType.MOUSE_MOVE);
+ //Sandcastle_End
};
if (typeof Cesium !== "undefined") {
window.startupCalled = true;
diff --git a/Apps/Sandcastle/gallery/Custom Per-Feature Post Process.html b/Apps/Sandcastle/gallery/Custom Per-Feature Post Process.html
index 72cc3f2c001d..d08090d9378d 100644
--- a/Apps/Sandcastle/gallery/Custom Per-Feature Post Process.html
+++ b/Apps/Sandcastle/gallery/Custom Per-Feature Post Process.html
@@ -54,7 +54,7 @@
if (czm_selected()) {
vec3 highlighted = highlight.a * highlight.rgb + (1.0 - highlight.a) * color.rgb;
out_FragColor = vec4(highlighted, 1.0);
- } else {
+ } else {
out_FragColor = color;
}
}
diff --git a/Apps/Sandcastle/gallery/Custom Post Process.html b/Apps/Sandcastle/gallery/Custom Post Process.html
index 38eeaf856e9d..39f5450cb271 100644
--- a/Apps/Sandcastle/gallery/Custom Post Process.html
+++ b/Apps/Sandcastle/gallery/Custom Post Process.html
@@ -44,23 +44,23 @@
});
const fragmentShaderSource = `
- uniform sampler2D colorTexture;
- in vec2 v_textureCoordinates;
- const int KERNEL_WIDTH = 16;
- void main(void)
- {
- vec2 step = czm_pixelRatio / czm_viewport.zw;
- vec2 integralPos = v_textureCoordinates - mod(v_textureCoordinates, 8.0 * step);
- vec3 averageValue = vec3(0.0);
- for (int i = 0; i < KERNEL_WIDTH; i++)
- {
- for (int j = 0; j < KERNEL_WIDTH; j++)
- {
- averageValue += texture(colorTexture, integralPos + step * vec2(i, j)).rgb;
- }
- }
- averageValue /= float(KERNEL_WIDTH * KERNEL_WIDTH);
- out_FragColor = vec4(averageValue, 1.0);
+ uniform sampler2D colorTexture;
+ in vec2 v_textureCoordinates;
+ const int KERNEL_WIDTH = 16;
+ void main(void)
+ {
+ vec2 step = czm_pixelRatio / czm_viewport.zw;
+ vec2 integralPos = v_textureCoordinates - mod(v_textureCoordinates, 8.0 * step);
+ vec3 averageValue = vec3(0.0);
+ for (int i = 0; i < KERNEL_WIDTH; i++)
+ {
+ for (int j = 0; j < KERNEL_WIDTH; j++)
+ {
+ averageValue += texture(colorTexture, integralPos + step * vec2(i, j)).rgb;
+ }
+ }
+ averageValue /= float(KERNEL_WIDTH * KERNEL_WIDTH);
+ out_FragColor = vec4(averageValue, 1.0);
}
`;
viewer.scene.postProcessStages.add(
diff --git a/Apps/Sandcastle/gallery/Fog Post Process.html b/Apps/Sandcastle/gallery/Fog Post Process.html
index d3936751c118..8896ca04c059 100644
--- a/Apps/Sandcastle/gallery/Fog Post Process.html
+++ b/Apps/Sandcastle/gallery/Fog Post Process.html
@@ -48,40 +48,40 @@
}
const fragmentShaderSource = `
- float getDistance(sampler2D depthTexture, vec2 texCoords)
- {
- float depth = czm_unpackDepth(texture(depthTexture, texCoords));
- if (depth == 0.0) {
- return czm_infinity;
- }
- vec4 eyeCoordinate = czm_windowToEyeCoordinates(gl_FragCoord.xy, depth);
- return -eyeCoordinate.z / eyeCoordinate.w;
- }
- float interpolateByDistance(vec4 nearFarScalar, float distance)
- {
- float startDistance = nearFarScalar.x;
- float startValue = nearFarScalar.y;
- float endDistance = nearFarScalar.z;
- float endValue = nearFarScalar.w;
- float t = clamp((distance - startDistance) / (endDistance - startDistance), 0.0, 1.0);
- return mix(startValue, endValue, t);
- }
- vec4 alphaBlend(vec4 sourceColor, vec4 destinationColor)
- {
- return sourceColor * vec4(sourceColor.aaa, 1.0) + destinationColor * (1.0 - sourceColor.a);
- }
- uniform sampler2D colorTexture;
- uniform sampler2D depthTexture;
- uniform vec4 fogByDistance;
- uniform vec4 fogColor;
- in vec2 v_textureCoordinates;
- void main(void)
- {
- float distance = getDistance(depthTexture, v_textureCoordinates);
- vec4 sceneColor = texture(colorTexture, v_textureCoordinates);
- float blendAmount = interpolateByDistance(fogByDistance, distance);
- vec4 finalFogColor = vec4(fogColor.rgb, fogColor.a * blendAmount);
- out_FragColor = alphaBlend(finalFogColor, sceneColor);
+ float getDistance(sampler2D depthTexture, vec2 texCoords)
+ {
+ float depth = czm_unpackDepth(texture(depthTexture, texCoords));
+ if (depth == 0.0) {
+ return czm_infinity;
+ }
+ vec4 eyeCoordinate = czm_windowToEyeCoordinates(gl_FragCoord.xy, depth);
+ return -eyeCoordinate.z / eyeCoordinate.w;
+ }
+ float interpolateByDistance(vec4 nearFarScalar, float distance)
+ {
+ float startDistance = nearFarScalar.x;
+ float startValue = nearFarScalar.y;
+ float endDistance = nearFarScalar.z;
+ float endValue = nearFarScalar.w;
+ float t = clamp((distance - startDistance) / (endDistance - startDistance), 0.0, 1.0);
+ return mix(startValue, endValue, t);
+ }
+ vec4 alphaBlend(vec4 sourceColor, vec4 destinationColor)
+ {
+ return sourceColor * vec4(sourceColor.aaa, 1.0) + destinationColor * (1.0 - sourceColor.a);
+ }
+ uniform sampler2D colorTexture;
+ uniform sampler2D depthTexture;
+ uniform vec4 fogByDistance;
+ uniform vec4 fogColor;
+ in vec2 v_textureCoordinates;
+ void main(void)
+ {
+ float distance = getDistance(depthTexture, v_textureCoordinates);
+ vec4 sceneColor = texture(colorTexture, v_textureCoordinates);
+ float blendAmount = interpolateByDistance(fogByDistance, distance);
+ vec4 finalFogColor = vec4(fogColor.rgb, fogColor.a * blendAmount);
+ out_FragColor = alphaBlend(finalFogColor, sceneColor);
}
`;
diff --git a/Apps/Sandcastle/gallery/Google Earth Enterprise.html b/Apps/Sandcastle/gallery/Google Earth Enterprise.html
index 115773cdba69..651d2114543f 100644
--- a/Apps/Sandcastle/gallery/Google Earth Enterprise.html
+++ b/Apps/Sandcastle/gallery/Google Earth Enterprise.html
@@ -52,7 +52,7 @@
);
layers.add(blackMarble);
} catch (error) {
- console.log(`Failed to create Google Earth providers from metadata. Confirm GEE service is correctly configured.
+ console.log(`Failed to create Google Earth providers from metadata. Confirm GEE service is correctly configured.
${error}`);
}
diff --git a/Apps/Sandcastle/gallery/Materials.html b/Apps/Sandcastle/gallery/Materials.html
index 1640330d4f27..803d899b99a8 100644
--- a/Apps/Sandcastle/gallery/Materials.html
+++ b/Apps/Sandcastle/gallery/Materials.html
@@ -112,19 +112,19 @@
},
},
source: `
- czm_material czm_getMaterial(czm_materialInput materialInput) {
- czm_material material = czm_getDefaultMaterial(materialInput);
- vec4 color;
- float heightValue = texture(heightField, materialInput.st).r;
- color.rgb = mix(vec3(0.2, 0.6, 0.2), vec3(1.0, 0.5, 0.2), heightValue);
- color.a = (1.0 - texture(image, materialInput.st).r) * 0.7;
- color = czm_gammaCorrect(color);
- material.diffuse = color.rgb;
- material.alpha = color.a;
- material.normal = bumpMap.normal;
+ czm_material czm_getMaterial(czm_materialInput materialInput) {
+ czm_material material = czm_getDefaultMaterial(materialInput);
+ vec4 color;
+ float heightValue = texture(heightField, materialInput.st).r;
+ color.rgb = mix(vec3(0.2, 0.6, 0.2), vec3(1.0, 0.5, 0.2), heightValue);
+ color.a = (1.0 - texture(image, materialInput.st).r) * 0.7;
+ color = czm_gammaCorrect(color);
+ material.diffuse = color.rgb;
+ material.alpha = color.a;
+ material.normal = bumpMap.normal;
material.specular = step(0.1, heightValue); // Specular mountain tops
material.shininess = 8.0; // Sharpen highlight
- return material;
+ return material;
}
`,
},
diff --git a/packages/engine/Source/Scene/ArcGisMapServerImageryProvider.js b/packages/engine/Source/Scene/ArcGisMapServerImageryProvider.js
index f2768fbc57f6..d1b97d8f6365 100644
--- a/packages/engine/Source/Scene/ArcGisMapServerImageryProvider.js
+++ b/packages/engine/Source/Scene/ArcGisMapServerImageryProvider.js
@@ -260,9 +260,9 @@ async function requestMetadata(resource, imageryProviderBuilder) {
*
* Provides tiled imagery hosted by an ArcGIS MapServer. By default, the server's pre-cached tiles are
* used, if available.
- *
+ *
*
- *
+ *
* An {@link https://developers.arcgis.com/documentation/mapping-apis-and-services/security| ArcGIS Access Token } is required to authenticate requests to an ArcGIS Image Tile service.
* To access secure ArcGIS resources, it's required to create an ArcGIS developer
* account or an ArcGIS online account, then implement an authentication method to obtain an access token.
@@ -278,7 +278,7 @@ async function requestMetadata(resource, imageryProviderBuilder) {
* @example
* // Set the default access token for accessing ArcGIS Image Tile service
* Cesium.ArcGisMapService.defaultAccessToken = "";
- *
+ *
* // Add a base layer from a default ArcGIS basemap
* const viewer = new Cesium.Viewer("cesiumContainer", {
* baseLayer: Cesium.ImageryLayer.fromProviderAsync(
diff --git a/packages/engine/Specs/Scene/Model/PrimitiveOutlineGeneratorSpec.js b/packages/engine/Specs/Scene/Model/PrimitiveOutlineGeneratorSpec.js
index eb73de212900..1e3a0ace79e9 100644
--- a/packages/engine/Specs/Scene/Model/PrimitiveOutlineGeneratorSpec.js
+++ b/packages/engine/Specs/Scene/Model/PrimitiveOutlineGeneratorSpec.js
@@ -263,7 +263,7 @@ describe(
4, 8, 11
]);
// prettier-ignore
- let expectedOutlineCoordinates = new Float32Array([
+ let expectedOutlineCoordinates = new Float32Array([
0, 1, 0,
0, 1, 0,
0, 1, 0,
diff --git a/packages/engine/Specs/Scene/Vector3DTilePointsSpec.js b/packages/engine/Specs/Scene/Vector3DTilePointsSpec.js
index ee9cf1d6d196..b171cc95684b 100644
--- a/packages/engine/Specs/Scene/Vector3DTilePointsSpec.js
+++ b/packages/engine/Specs/Scene/Vector3DTilePointsSpec.js
@@ -431,7 +431,7 @@ describe(
/*[
Cesium3DTileStyle option,
Cesium3DTileStyle default value,
- Cesium3DTileFeature property,
+ Cesium3DTileFeature property,
expected Cesium3DTileFeature value,
expected undefined Cesium3DTileFeature value
]*/