123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068106910701071107210731074107510761077107810791080108110821083108410851086108710881089109010911092109310941095109610971098109911001101110211031104 |
- import Cartesian2 from "../Core/Cartesian2.js";
- import Cartesian3 from "../Core/Cartesian3.js";
- import Cartesian4 from "../Core/Cartesian4.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 Matrix2 from "../Core/Matrix2.js";
- import Matrix3 from "../Core/Matrix3.js";
- import Matrix4 from "../Core/Matrix4.js";
- import MetadataType from "./MetadataType.js";
- import MetadataComponentType from "./MetadataComponentType.js";
- /**
- * A metadata property, as part of a {@link MetadataClass}
- *
- * @param {Object} options Object with the following properties:
- * @param {String} options.id The ID of the property.
- * @param {Object} options.property The property JSON object.
- * @param {Object.<String, MetadataEnum>} [options.enums] A dictionary of enums.
- *
- * @alias MetadataClassProperty
- * @constructor
- * @private
- * @experimental This feature is using part of the 3D Tiles spec that is not final and is subject to change without Cesium's standard deprecation policy.
- */
- function MetadataClassProperty(options) {
- options = defaultValue(options, defaultValue.EMPTY_OBJECT);
- const id = options.id;
- const property = options.property;
- //>>includeStart('debug', pragmas.debug);
- Check.typeOf.string("options.id", id);
- Check.typeOf.object("options.property", property);
- Check.typeOf.string("options.property.type", property.type);
- //>>includeEnd('debug');
- // Try to determine if this is the legacy extension. This is not
- // always possible, as there are some types that are valid in both
- // extensions.
- const isLegacyExtension = isLegacy(property);
- const parsedType = parseType(property, options.enums);
- const componentType = parsedType.componentType;
- const normalized =
- defined(componentType) &&
- MetadataComponentType.isIntegerType(componentType) &&
- defaultValue(property.normalized, false);
- // Basic information about this property
- this._id = id;
- this._name = property.name;
- this._description = property.description;
- this._semantic = property.semantic;
- this._isLegacyExtension = isLegacyExtension;
- // Details about basic types
- this._type = parsedType.type;
- this._componentType = componentType;
- this._enumType = parsedType.enumType;
- this._valueType = parsedType.valueType;
- // Details about arrays
- this._isArray = parsedType.isArray;
- this._isVariableLengthArray = parsedType.isVariableLengthArray;
- this._arrayLength = parsedType.arrayLength;
- // min and max allowed values
- this._min = property.min;
- this._max = property.max;
- // properties that adjust the range of metadata values
- this._normalized = normalized;
- let offset = property.offset;
- let scale = property.scale;
- const hasValueTransform = defined(offset) || defined(scale);
- const enableNestedArrays = true;
- if (!defined(offset)) {
- offset = this.expandConstant(0, enableNestedArrays);
- }
- if (!defined(scale)) {
- scale = this.expandConstant(1, enableNestedArrays);
- }
- this._offset = offset;
- this._scale = scale;
- this._hasValueTransform = hasValueTransform;
- // sentinel value for missing data, and a default value to use
- // in its place if needed.
- this._noData = property.noData;
- // For vector and array types, this is stored as an array of values.
- this._default = property.default;
- // EXT_feature_metadata had an optional flag, while EXT_structural_metadata
- // has a required flag. The defaults are not the same, and there's some cases
- // like {type: BOOLEAN} that are ambiguous. Coalesce this into a single
- // required flag
- let required;
- if (!defined(isLegacyExtension)) {
- // Impossible to tell which extension was used, so don't require
- // the property
- required = false;
- } else if (isLegacyExtension) {
- required = defined(property.optional) ? !property.optional : true;
- } else {
- required = defaultValue(property.required, false);
- }
- this._required = required;
- // extras and extensions
- this._extras = property.extras;
- this._extensions = property.extensions;
- }
- Object.defineProperties(MetadataClassProperty.prototype, {
- /**
- * The ID of the property.
- *
- * @memberof MetadataClassProperty.prototype
- * @type {String}
- * @readonly
- * @private
- */
- id: {
- get: function () {
- return this._id;
- },
- },
- /**
- * The name of the property.
- *
- * @memberof MetadataClassProperty.prototype
- * @type {String}
- * @readonly
- * @private
- */
- name: {
- get: function () {
- return this._name;
- },
- },
- /**
- * The description of the property.
- *
- * @memberof MetadataClassProperty.prototype
- * @type {String}
- * @readonly
- * @private
- */
- description: {
- get: function () {
- return this._description;
- },
- },
- /**
- * The type of the property such as SCALAR, VEC2, VEC3
- *
- *
- * @memberof MetadataClassProperty.prototype
- * @type {MetadataType}
- * @readonly
- * @private
- */
- type: {
- get: function () {
- return this._type;
- },
- },
- /**
- * The enum type of the property. Only defined when type is ENUM.
- *
- * @memberof MetadataClassProperty.prototype
- * @type {MetadataEnum}
- * @readonly
- * @private
- */
- enumType: {
- get: function () {
- return this._enumType;
- },
- },
- /**
- * The component type of the property. This includes integer
- * (e.g. INT8 or UINT16), and floating point (FLOAT32 and FLOAT64) values
- *
- * @memberof MetadataClassProperty.prototype
- * @type {MetadataComponentType}
- * @readonly
- * @private
- */
- componentType: {
- get: function () {
- return this._componentType;
- },
- },
- /**
- * The datatype used for storing each component of the property. This
- * is usually the same as componentType except for ENUM, where this
- * returns an integer type
- *
- * @memberof MetadataClassProperty.prototype
- * @type {MetadataComponentType}
- * @readonly
- * @private
- */
- valueType: {
- get: function () {
- return this._valueType;
- },
- },
- /**
- * True if a property is an array (either fixed length or variable length),
- * false otherwise.
- *
- * @memberof MetadataClassProperty.prototype
- * @type {Boolean}
- * @readonly
- * @private
- */
- isArray: {
- get: function () {
- return this._isArray;
- },
- },
- /**
- * True if a property is a variable length array, false otherwise.
- *
- * @memberof MetadataClassProperty.prototype
- * @type {Boolean}
- * @readonly
- * @private
- */
- isVariableLengthArray: {
- get: function () {
- return this._isVariableLengthArray;
- },
- },
- /**
- * The number of array elements. Only defined for fixed-size
- * arrays.
- *
- * @memberof MetadataClassProperty.prototype
- * @type {Number}
- * @readonly
- * @private
- */
- arrayLength: {
- get: function () {
- return this._arrayLength;
- },
- },
- /**
- * Whether the property is normalized.
- *
- * @memberof MetadataClassProperty.prototype
- * @type {Boolean}
- * @readonly
- * @private
- */
- normalized: {
- get: function () {
- return this._normalized;
- },
- },
- /**
- * A number or an array of numbers storing the maximum allowable value of this property. Only defined when type or componentType is a numeric type.
- *
- * @memberof MetadataClassProperty.prototype
- * @type {Number|Number[]}
- * @readonly
- * @private
- */
- max: {
- get: function () {
- return this._max;
- },
- },
- /**
- * A number or an array of numbers storing the minimum allowable value of this property. Only defined when type or componentType is a numeric type.
- *
- * @memberof MetadataClassProperty.prototype
- * @type {Number|Number[]}
- * @readonly
- * @private
- */
- min: {
- get: function () {
- return this._min;
- },
- },
- /**
- * The no-data sentinel value that represents null values
- *
- * @memberof MetadataClassProperty.prototype
- * @type {Boolean|Number|String|Array}
- * @readonly
- * @private
- */
- noData: {
- get: function () {
- return this._noData;
- },
- },
- /**
- * A default value to use when an entity's property value is not defined.
- *
- * @memberof MetadataClassProperty.prototype
- * @type {Boolean|Number|String|Array}
- * @readonly
- * @private
- */
- default: {
- get: function () {
- return this._default;
- },
- },
- /**
- * Whether the property is required.
- *
- * @memberof MetadataClassProperty.prototype
- * @type {Boolean}
- * @readonly
- * @private
- */
- required: {
- get: function () {
- return this._required;
- },
- },
- /**
- * An identifier that describes how this property should be interpreted.
- *
- * @memberof MetadataClassProperty.prototype
- * @type {String}
- * @readonly
- * @private
- */
- semantic: {
- get: function () {
- return this._semantic;
- },
- },
- /**
- * True if offset/scale should be applied. If both offset/scale were
- * undefined, they default to identity so this property is set false
- *
- * @memberof MetadataClassProperty.prototype
- * @type {Boolean}
- * @readonly
- * @private
- */
- hasValueTransform: {
- get: function () {
- return this._hasValueTransform;
- },
- },
- /**
- * The offset to be added to property values as part of the value transform.
- *
- * @memberof MetadataClassProperty.prototype
- * @type {Number|Number[]|Number[][]}
- * @readonly
- * @private
- */
- offset: {
- get: function () {
- return this._offset;
- },
- },
- /**
- * The scale to be multiplied to property values as part of the value transform.
- *
- * @memberof MetadataClassProperty.prototype
- * @type {Number|Number[]|Number[][]}
- * @readonly
- * @private
- */
- scale: {
- get: function () {
- return this._scale;
- },
- },
- /**
- * Extras in the JSON object.
- *
- * @memberof MetadataClassProperty.prototype
- * @type {*}
- * @readonly
- * @private
- */
- extras: {
- get: function () {
- return this._extras;
- },
- },
- /**
- * Extensions in the JSON object.
- *
- * @memberof MetadataClassProperty.prototype
- * @type {Object}
- * @readonly
- * @private
- */
- extensions: {
- get: function () {
- return this._extensions;
- },
- },
- });
- function isLegacy(property) {
- if (property.type === "ARRAY") {
- return true;
- }
- // New property types in EXT_structural_metadata
- const type = property.type;
- if (
- type === MetadataType.SCALAR ||
- MetadataType.isMatrixType(type) ||
- MetadataType.isVectorType(type)
- ) {
- return false;
- }
- // EXT_feature_metadata allowed numeric types as a type. Now they are
- // represented as {type: SINGLE, componentType: type}
- if (MetadataComponentType.isNumericType(type)) {
- return true;
- }
- // New properties in EXT_structural_metadata
- if (
- defined(property.noData) ||
- defined(property.scale) ||
- defined(property.offset) ||
- defined(property.required) ||
- defined(property.count) ||
- defined(property.array)
- ) {
- return false;
- }
- // Properties that only exist in EXT_feature_metadata
- if (defined(property.optional)) {
- return false;
- }
- // impossible to tell, give up.
- return undefined;
- }
- function parseType(property, enums) {
- const type = property.type;
- const componentType = property.componentType;
- // EXT_feature_metadata had an ARRAY type. This is now handled
- // with array + count, so some details need to be transcoded
- const isLegacyArray = type === "ARRAY";
- let isArray;
- let arrayLength;
- let isVariableLengthArray;
- if (isLegacyArray) {
- // definitely EXT_feature_metadata
- isArray = true;
- arrayLength = property.componentCount;
- isVariableLengthArray = !defined(arrayLength);
- } else if (property.array) {
- isArray = true;
- arrayLength = property.count;
- isVariableLengthArray = !defined(property.count);
- } else {
- // Could be either extension. Some cases are impossible to distinguish
- // Default to a single value
- isArray = false;
- arrayLength = undefined;
- isVariableLengthArray = false;
- }
- let enumType;
- if (defined(property.enumType)) {
- enumType = enums[property.enumType];
- }
- // In both EXT_feature_metadata and EXT_structural_metadata, ENUM appears
- // as a type.
- if (type === MetadataType.ENUM) {
- return {
- type: type,
- componentType: undefined,
- enumType: enumType,
- valueType: enumType.valueType,
- isArray: isArray,
- isVariableLengthArray: isVariableLengthArray,
- arrayLength: arrayLength,
- };
- }
- // In EXT_feature_metadata, ENUM also appears as an ARRAY componentType
- if (isLegacyArray && componentType === MetadataType.ENUM) {
- return {
- type: componentType,
- componentType: undefined,
- enumType: enumType,
- valueType: enumType.valueType,
- isArray: isArray,
- isVariableLengthArray: isVariableLengthArray,
- arrayLength: arrayLength,
- };
- }
- // EXT_structural_metadata only: SCALAR, VECN and MATN
- if (
- type === MetadataType.SCALAR ||
- MetadataType.isMatrixType(type) ||
- MetadataType.isVectorType(type)
- ) {
- return {
- type: type,
- componentType: componentType,
- enumType: undefined,
- valueType: componentType,
- isArray: isArray,
- isVariableLengthArray: isVariableLengthArray,
- arrayLength: arrayLength,
- };
- }
- // In both EXT_structural_metadata and EXT_feature_metadata,
- // BOOLEAN and STRING appear as types
- if (type === MetadataType.BOOLEAN || type === MetadataType.STRING) {
- return {
- type: type,
- componentType: undefined,
- enumType: undefined,
- valueType: undefined,
- isArray: isArray,
- isVariableLengthArray: isVariableLengthArray,
- arrayLength: arrayLength,
- };
- }
- // EXT_feature_metadata also allows BOOLEAN and STRING as an ARRAY
- // componentType
- if (
- isLegacyArray &&
- (componentType === MetadataType.BOOLEAN ||
- componentType === MetadataType.STRING)
- ) {
- return {
- type: componentType,
- componentType: undefined,
- enumType: undefined,
- valueType: undefined,
- isArray: isArray,
- isVariableLengthArray: isVariableLengthArray,
- arrayLength: arrayLength,
- };
- }
- // Both EXT_feature_metadata and EXT_structural_metadata allow numeric types like
- // INT32 or FLOAT64 as a componentType.
- if (
- defined(componentType) &&
- MetadataComponentType.isNumericType(componentType)
- ) {
- return {
- type: MetadataType.SCALAR,
- componentType: componentType,
- enumType: undefined,
- valueType: componentType,
- isArray: isArray,
- isVariableLengthArray: isVariableLengthArray,
- arrayLength: arrayLength,
- };
- }
- // EXT_feature_metadata: integer and float types were allowed as types,
- // but now these are expressed as {type: SCALAR, componentType: type}
- if (MetadataComponentType.isNumericType(type)) {
- return {
- type: MetadataType.SCALAR,
- componentType: type,
- enumType: undefined,
- valueType: type,
- isArray: isArray,
- isVariableLengthArray: isVariableLengthArray,
- arrayLength: arrayLength,
- };
- }
- //>>includeStart('debug', pragmas.debug);
- throw new DeveloperError(
- `unknown metadata type {type: ${type}, componentType: ${componentType})`
- );
- //>>includeEnd('debug');
- }
- /**
- * Normalizes integer property values. If the property is not normalized
- * the value is returned unmodified.
- * <p>
- * Given the way normalization is defined in {@link https://github.com/CesiumGS/3d-tiles/tree/main/specification/Metadata#normalized-values|the 3D Metadata Specification},
- * normalize and unnormalize are almost, but not quite inverses. In particular,
- * the smallest signed integer value will be off by one after normalizing and
- * unnormalizing. See
- * {@link https://www.desmos.com/calculator/nledg1evut|this Desmos graph} for
- * an example using INT8.
- * </p>
- * <p>
- * Furthermore, for 64-bit integer types, there may be a loss of precision
- * due to conversion to Number
- * </p>
- *
- * @param {*} value The integer value or array of integer values.
- * @returns {*} The normalized value or array of normalized values.
- *
- * @private
- */
- MetadataClassProperty.prototype.normalize = function (value) {
- if (!this._normalized) {
- return value;
- }
- return normalizeInPlace(
- value,
- this._valueType,
- MetadataComponentType.normalize
- );
- };
- /**
- * Unnormalizes integer property values. If the property is not normalized
- * the value is returned unmodified.
- * <p>
- * Given the way normalization is defined in {@link https://github.com/CesiumGS/3d-tiles/tree/main/specification/Metadata#normalized-values|the 3D Metadata Specification},
- * normalize and unnormalize are almost, but not quite inverses. In particular,
- * the smallest signed integer value will be off by one after normalizing and
- * unnormalizing. See
- * {@link https://www.desmos.com/calculator/nledg1evut|this Desmos graph} for
- * an example using INT8.
- * </p>
- * <p>
- * Furthermore, for 64-bit integer types, there may be a loss of precision
- * due to conversion to Number
- * </p>
- *
- * @param {*} value The normalized value or array of normalized values.
- * @returns {*} The integer value or array of integer values.
- *
- * @private
- */
- MetadataClassProperty.prototype.unnormalize = function (value) {
- if (!this._normalized) {
- return value;
- }
- return normalizeInPlace(
- value,
- this._valueType,
- MetadataComponentType.unnormalize
- );
- };
- MetadataClassProperty.prototype.applyValueTransform = function (value) {
- // variable length arrays do not have a well-defined offset/scale so this
- // is forbidden by the spec
- if (!this._hasValueTransform || this._isVariableLengthArray) {
- return value;
- }
- return MetadataClassProperty.valueTransformInPlace(
- value,
- this._offset,
- this._scale,
- MetadataComponentType.applyValueTransform
- );
- };
- MetadataClassProperty.prototype.unapplyValueTransform = function (value) {
- // variable length arrays do not have a well-defined offset/scale so this
- // is forbidden by the spec
- if (!this._hasValueTransform || this._isVariableLengthArray) {
- return value;
- }
- return MetadataClassProperty.valueTransformInPlace(
- value,
- this._offset,
- this._scale,
- MetadataComponentType.unapplyValueTransform
- );
- };
- MetadataClassProperty.prototype.expandConstant = function (
- constant,
- enableNestedArrays
- ) {
- enableNestedArrays = defaultValue(enableNestedArrays, false);
- const isArray = this._isArray;
- const arrayLength = this._arrayLength;
- const componentCount = MetadataType.getComponentCount(this._type);
- const isNested = isArray && componentCount > 1;
- // scalar values can be returned directly
- if (!isArray && componentCount === 1) {
- return constant;
- }
- // vector and matrix values
- if (!isArray) {
- return new Array(componentCount).fill(constant);
- }
- // arrays of scalars
- if (!isNested) {
- return new Array(arrayLength).fill(constant);
- }
- // arrays of vectors/matrices: flattened
- if (!enableNestedArrays) {
- return new Array(this._arrayLength * componentCount).fill(constant);
- }
- // array of vectors/matrices: nested
- const innerConstant = new Array(componentCount).fill(constant);
- // This array fill duplicates the pointer to the inner arrays. Since this is
- // intended for use with constants, no need to clone the array.
- return new Array(this._arrayLength).fill(innerConstant);
- };
- /**
- * If the value is the noData sentinel, return undefined. Otherwise, return
- * the value.
- * @param {*} value The raw value
- * @returns {*} Either the value or undefined if the value was a no data value.
- *
- * @private
- */
- MetadataClassProperty.prototype.handleNoData = function (value) {
- const sentinel = this._noData;
- if (!defined(sentinel)) {
- return value;
- }
- if (arrayEquals(value, sentinel)) {
- return undefined;
- }
- return value;
- };
- function arrayEquals(left, right) {
- if (!Array.isArray(left)) {
- return left === right;
- }
- if (!Array.isArray(right)) {
- return false;
- }
- if (left.length !== right.length) {
- return false;
- }
- for (let i = 0; i < left.length; i++) {
- if (!arrayEquals(left[i], right[i])) {
- return false;
- }
- }
- return true;
- }
- /**
- * Unpack VECN values into {@link Cartesian2}, {@link Cartesian3}, or
- * {@link Cartesian4} and MATN values into {@link Matrix2}, {@link Matrix3}, or
- * {@link Matrix4} depending on N. All other values (including arrays of
- * other sizes) are passed through unaltered.
- *
- * @param {*} value the original, normalized values.
- * @param {Boolean} [enableNestedArrays=false] If true, arrays of vectors are represented as nested arrays. This is used for JSON encoding but not binary encoding
- * @returns {*} The appropriate vector or matrix type if the value is a vector or matrix type, respectively. If the property is an array of vectors or matrices, an array of the appropriate vector or matrix type is returned. Otherwise, the value is returned unaltered.
- * @private
- */
- MetadataClassProperty.prototype.unpackVectorAndMatrixTypes = function (
- value,
- enableNestedArrays
- ) {
- enableNestedArrays = defaultValue(enableNestedArrays, false);
- const MathType = MetadataType.getMathType(this._type);
- const isArray = this._isArray;
- const componentCount = MetadataType.getComponentCount(this._type);
- const isNested = isArray && componentCount > 1;
- if (!defined(MathType)) {
- return value;
- }
- if (enableNestedArrays && isNested) {
- return value.map(function (x) {
- return MathType.unpack(x);
- });
- }
- if (isArray) {
- return MathType.unpackArray(value);
- }
- return MathType.unpack(value);
- };
- /**
- * Pack a {@link Cartesian2}, {@link Cartesian3}, or {@link Cartesian4} into an
- * array if this property is an <code>VECN</code>.
- * Pack a {@link Matrix2}, {@link Matrix3}, or {@link Matrix4} into an
- * array if this property is an <code>MATN</code>.
- * All other values (including arrays of other sizes) are passed through unaltered.
- *
- * @param {*} value The value of this property
- * @param {Boolean} [enableNestedArrays=false] If true, arrays of vectors are represented as nested arrays. This is used for JSON encoding but not binary encoding
- * @returns {*} An array of the appropriate length if the property is a vector or matrix type. Otherwise, the value is returned unaltered.
- * @private
- */
- MetadataClassProperty.prototype.packVectorAndMatrixTypes = function (
- value,
- enableNestedArrays
- ) {
- enableNestedArrays = defaultValue(enableNestedArrays, false);
- const MathType = MetadataType.getMathType(this._type);
- const isArray = this._isArray;
- const componentCount = MetadataType.getComponentCount(this._type);
- const isNested = isArray && componentCount > 1;
- if (!defined(MathType)) {
- return value;
- }
- if (enableNestedArrays && isNested) {
- return value.map(function (x) {
- return MathType.pack(x, []);
- });
- }
- if (isArray) {
- return MathType.packArray(value, []);
- }
- return MathType.pack(value, []);
- };
- /**
- * Validates whether the given value conforms to the property.
- *
- * @param {*} value The value.
- * @returns {String|undefined} An error message if the value does not conform to the property, otherwise undefined.
- * @private
- */
- MetadataClassProperty.prototype.validate = function (value) {
- if (!defined(value) && defined(this._default)) {
- // no value, but we have a default to use.
- return undefined;
- }
- if (this._required && !defined(value)) {
- return `required property must have a value`;
- }
- if (this._isArray) {
- return validateArray(this, value);
- }
- return validateSingleValue(this, value);
- };
- function validateArray(classProperty, value) {
- if (!Array.isArray(value)) {
- return `value ${value} must be an array`;
- }
- const length = value.length;
- if (
- !classProperty._isVariableLengthArray &&
- length !== classProperty._arrayLength
- ) {
- return "Array length does not match property.arrayLength";
- }
- for (let i = 0; i < length; i++) {
- const message = validateSingleValue(classProperty, value[i]);
- if (defined(message)) {
- return message;
- }
- }
- }
- function validateSingleValue(classProperty, value) {
- const type = classProperty._type;
- const componentType = classProperty._componentType;
- const enumType = classProperty._enumType;
- const normalized = classProperty._normalized;
- if (MetadataType.isVectorType(type)) {
- return validateVector(value, type, componentType);
- } else if (MetadataType.isMatrixType(type)) {
- return validateMatrix(value, type, componentType);
- } else if (type === MetadataType.STRING) {
- return validateString(value);
- } else if (type === MetadataType.BOOLEAN) {
- return validateBoolean(value);
- } else if (type === MetadataType.ENUM) {
- return validateEnum(value, enumType);
- }
- return validateScalar(value, componentType, normalized);
- }
- function validateVector(value, type, componentType) {
- if (!MetadataComponentType.isVectorCompatible(componentType)) {
- return `componentType ${componentType} is incompatible with vector type ${type}`;
- }
- if (type === MetadataType.VEC2 && !(value instanceof Cartesian2)) {
- return `vector value ${value} must be a Cartesian2`;
- }
- if (type === MetadataType.VEC3 && !(value instanceof Cartesian3)) {
- return `vector value ${value} must be a Cartesian3`;
- }
- if (type === MetadataType.VEC4 && !(value instanceof Cartesian4)) {
- return `vector value ${value} must be a Cartesian4`;
- }
- }
- function validateMatrix(value, type, componentType) {
- if (!MetadataComponentType.isVectorCompatible(componentType)) {
- return `componentType ${componentType} is incompatible with matrix type ${type}`;
- }
- if (type === MetadataType.MAT2 && !(value instanceof Matrix2)) {
- return `matrix value ${value} must be a Matrix2`;
- }
- if (type === MetadataType.MAT3 && !(value instanceof Matrix3)) {
- return `matrix value ${value} must be a Matrix3`;
- }
- if (type === MetadataType.MAT4 && !(value instanceof Matrix4)) {
- return `matrix value ${value} must be a Matrix4`;
- }
- }
- function validateString(value) {
- if (typeof value !== "string") {
- return getTypeErrorMessage(value, MetadataType.STRING);
- }
- }
- function validateBoolean(value) {
- if (typeof value !== "boolean") {
- return getTypeErrorMessage(value, MetadataType.BOOLEAN);
- }
- }
- function validateEnum(value, enumType) {
- const javascriptType = typeof value;
- if (defined(enumType)) {
- if (javascriptType !== "string" || !defined(enumType.valuesByName[value])) {
- return `value ${value} is not a valid enum name for ${enumType.id}`;
- }
- return;
- }
- }
- function validateScalar(value, componentType, normalized) {
- const javascriptType = typeof value;
- switch (componentType) {
- case MetadataComponentType.INT8:
- case MetadataComponentType.UINT8:
- case MetadataComponentType.INT16:
- case MetadataComponentType.UINT16:
- case MetadataComponentType.INT32:
- case MetadataComponentType.UINT32:
- case MetadataComponentType.FLOAT32:
- case MetadataComponentType.FLOAT64:
- if (javascriptType !== "number") {
- return getTypeErrorMessage(value, componentType);
- }
- if (!isFinite(value)) {
- return getNonFiniteErrorMessage(value, componentType);
- }
- return checkInRange(value, componentType, normalized);
- case MetadataComponentType.INT64:
- case MetadataComponentType.UINT64:
- if (javascriptType !== "number" && javascriptType !== "bigint") {
- return getTypeErrorMessage(value, componentType);
- }
- if (javascriptType === "number" && !isFinite(value)) {
- return getNonFiniteErrorMessage(value, componentType);
- }
- return checkInRange(value, componentType, normalized);
- }
- }
- function getTypeErrorMessage(value, type) {
- return `value ${value} does not match type ${type}`;
- }
- function getOutOfRangeErrorMessage(value, type, normalized) {
- let errorMessage = `value ${value} is out of range for type ${type}`;
- if (normalized) {
- errorMessage += " (normalized)";
- }
- return errorMessage;
- }
- function checkInRange(value, componentType, normalized) {
- if (normalized) {
- const min = MetadataComponentType.isUnsignedIntegerType(componentType)
- ? 0.0
- : -1.0;
- const max = 1.0;
- if (value < min || value > max) {
- return getOutOfRangeErrorMessage(value, componentType, normalized);
- }
- return;
- }
- if (
- value < MetadataComponentType.getMinimum(componentType) ||
- value > MetadataComponentType.getMaximum(componentType)
- ) {
- return getOutOfRangeErrorMessage(value, componentType, normalized);
- }
- }
- function getNonFiniteErrorMessage(value, type) {
- return `value ${value} of type ${type} must be finite`;
- }
- function normalizeInPlace(values, valueType, normalizeFunction) {
- if (!Array.isArray(values)) {
- return normalizeFunction(values, valueType);
- }
- for (let i = 0; i < values.length; i++) {
- values[i] = normalizeInPlace(values[i], valueType, normalizeFunction);
- }
- return values;
- }
- MetadataClassProperty.valueTransformInPlace = function (
- values,
- offsets,
- scales,
- transformationFunction
- ) {
- if (!Array.isArray(values)) {
- // transform a single value
- return transformationFunction(values, offsets, scales);
- }
- for (let i = 0; i < values.length; i++) {
- // offsets and scales must be the same array shape as values.
- values[i] = MetadataClassProperty.valueTransformInPlace(
- values[i],
- offsets[i],
- scales[i],
- transformationFunction
- );
- }
- return values;
- };
- export default MetadataClassProperty;
|