import Cartesian3 from "../Core/Cartesian3.js"; import Check from "../Core/Check.js"; import clone from "../Core/clone.js"; import combine from "../Core/combine.js"; import defaultValue from "../Core/defaultValue.js"; import defined from "../Core/defined.js"; import destroyObject from "../Core/destroyObject.js"; import CesiumMath from "../Core/Math.js"; import HilbertOrder from "../Core/HilbertOrder.js"; import Matrix3 from "../Core/Matrix3.js"; import Rectangle from "../Core/Rectangle.js"; import S2Cell from "../Core/S2Cell.js"; import when from "../ThirdParty/when.js"; import ImplicitSubtree from "./ImplicitSubtree.js"; import ImplicitTileMetadata from "./ImplicitTileMetadata.js"; import hasExtension from "./hasExtension.js"; import MetadataSemantic from "./MetadataSemantic.js"; import parseBoundingVolumeSemantics from "./parseBoundingVolumeSemantics.js"; /** * A specialized {@link Cesium3DTileContent} that lazily evaluates an implicit * tileset. It is somewhat similar in operation to a * {@link Tileset3DTileContent} in that once the content is constructed, it * updates the tileset tree with more tiles. However, unlike external tilesets, * child subtrees are represented as additional placeholder nodes with * Implicit3DTileContent. *
* Implements the {@link Cesium3DTileContent} interface. *
* * @alias Implicit3DTileContent * @constructor * * @param {Cesium3DTileset} tileset The tileset this content belongs to * @param {Cesium3DTile} tile The tile this content belongs to. * @param {Resource} resource The resource for the tileset * @param {ArrayBuffer} arrayBuffer The array buffer that stores the content payload * @param {Number} [byteOffset=0] The offset into the array buffer * @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. */ export default function Implicit3DTileContent( tileset, tile, resource, arrayBuffer, byteOffset ) { //>>includeStart('debug', pragmas.debug); Check.defined("tile.implicitTileset", tile.implicitTileset); Check.defined("tile.implicitCoordinates", tile.implicitCoordinates); //>>includeEnd('debug'); var implicitTileset = tile.implicitTileset; var implicitCoordinates = tile.implicitCoordinates; this._implicitTileset = implicitTileset; this._implicitCoordinates = implicitCoordinates; this._implicitSubtree = undefined; this._tileset = tileset; this._tile = tile; this._resource = resource; this._readyPromise = when.defer(); this.featurePropertiesDirty = false; this._groupMetadata = undefined; var templateValues = implicitCoordinates.getTemplateValues(); var subtreeResource = implicitTileset.subtreeUriTemplate.getDerivedResource({ templateValues: templateValues, }); this._url = subtreeResource.getUrlComponent(true); initialize(this, arrayBuffer, byteOffset); } Object.defineProperties(Implicit3DTileContent.prototype, { featuresLength: { get: function () { return 0; }, }, pointsLength: { get: function () { return 0; }, }, trianglesLength: { get: function () { return 0; }, }, geometryByteLength: { get: function () { return 0; }, }, texturesByteLength: { get: function () { return 0; }, }, batchTableByteLength: { get: function () { return 0; }, }, innerContents: { get: function () { return undefined; }, }, readyPromise: { get: function () { return this._readyPromise.promise; }, }, tileset: { get: function () { return this._tileset; }, }, tile: { get: function () { return this._tile; }, }, url: { get: function () { return this._url; }, }, batchTable: { get: function () { return undefined; }, }, groupMetadata: { get: function () { return this._groupMetadata; }, set: function (value) { this._groupMetadata = value; }, }, }); /** * Initialize the implicit content by parsing the subtree resource and setting * up a promise chain to expand the immediate subtree. * * @param {Implicit3DTileContent} content The implicit content * @param {ArrayBuffer} arrayBuffer The ArrayBuffer containing a subtree binary * @param {Number} [byteOffset=0] The byte offset into the arrayBuffer * @private */ function initialize(content, arrayBuffer, byteOffset) { // Parse the subtree file byteOffset = defaultValue(byteOffset, 0); var uint8Array = new Uint8Array(arrayBuffer, byteOffset); var subtree = new ImplicitSubtree( content._resource, uint8Array, content._implicitTileset, content._implicitCoordinates ); content._implicitSubtree = subtree; subtree.readyPromise .then(function () { expandSubtree(content, subtree); content._readyPromise.resolve(); }) .otherwise(function (error) { content._readyPromise.reject(error); }); } /** * Expand a single subtree placeholder tile. This transcodes the subtree into * a tree of {@link Cesium3DTile}. The root of this tree is stored in * the placeholder tile's children array. This method also creates placeholder * tiles for the child subtrees to be lazily expanded as needed. * * @param {Implicit3DTileContent} content The content * @param {ImplicitSubtree} subtree The parsed subtree * @private */ function expandSubtree(content, subtree) { var placeholderTile = content._tile; // Parse the tiles inside this immediate subtree var childIndex = content._implicitCoordinates.childIndex; var results = transcodeSubtreeTiles( content, subtree, placeholderTile, childIndex ); // Link the new subtree to the existing placeholder tile. placeholderTile.children.push(results.rootTile); // for each child subtree, make new placeholder tiles var childSubtrees = listChildSubtrees(content, subtree, results.bottomRow); for (var i = 0; i < childSubtrees.length; i++) { var subtreeLocator = childSubtrees[i]; var leafTile = subtreeLocator.tile; var implicitChildTile = makePlaceholderChildSubtree( content, leafTile, subtreeLocator.childIndex ); leafTile.children.push(implicitChildTile); } } /** * A pair of (tile, childIndex) used for finding child subtrees. * * @typedef {Object} ChildSubtreeLocator * @property {Cesium3DTile} tile One of the tiles in the bottommost row of the subtree. * @property {Number} childIndex The morton index of the child tile relative to its parent * @private */ /** * Determine what child subtrees exist and return a list of information * * @param {Implicit3DTileContent} content The implicit content * @param {ImplicitSubtree} subtree The subtree for looking up availability * @param {Array* This creates a real tile for rendering, not a placeholder tile like some of * the other methods of ImplicitTileset. *
* * @param {Implicit3DTileContent} implicitContent The implicit content * @param {ImplicitSubtree} subtree The subtree the child tile belongs to * @param {Cesium3DTile} parentTile The parent of the new child tile * @param {Number} childIndex The morton index of the child tile relative to its parent * @param {Number} childBitIndex The index of the child tile within the tile's availability information. * @param {Boolean} [parentIsPlaceholderTile=false] True if parentTile is a placeholder tile. This is true for the root of each subtree. * @returns {Cesium3DTile} The new child tile. * @private */ function deriveChildTile( implicitContent, subtree, parentTile, childIndex, childBitIndex, parentIsPlaceholderTile ) { var implicitTileset = implicitContent._implicitTileset; var implicitCoordinates; if (defaultValue(parentIsPlaceholderTile, false)) { implicitCoordinates = parentTile.implicitCoordinates; } else { implicitCoordinates = parentTile.implicitCoordinates.getChildCoordinates( childIndex ); } // Parse metadata and bounding volume semantics at the beginning // as the bounding volumes are needed below. var tileMetadata; var tileBounds; var contentBounds; if (defined(subtree.metadataExtension)) { var metadataTable = subtree.metadataTable; tileMetadata = new ImplicitTileMetadata({ class: metadataTable.class, implicitCoordinates: implicitCoordinates, implicitSubtree: subtree, }); var boundingVolumeSemantics = parseBoundingVolumeSemantics(tileMetadata); tileBounds = boundingVolumeSemantics.tile; contentBounds = boundingVolumeSemantics.content; } var boundingVolume = getTileBoundingVolume( implicitTileset, implicitCoordinates, childIndex, parentIsPlaceholderTile, parentTile, tileBounds ); var contentJsons = []; for (var i = 0; i < implicitTileset.contentCount; i++) { if (!subtree.contentIsAvailableAtIndex(childBitIndex, i)) { continue; } var childContentTemplate = implicitTileset.contentUriTemplates[i]; var childContentUri = childContentTemplate.getDerivedResource({ templateValues: implicitCoordinates.getTemplateValues(), }).url; var contentJson = { uri: childContentUri, }; var contentBoundingVolume = getContentBoundingVolume( boundingVolume, contentBounds ); if (defined(contentBoundingVolume)) { contentJson.boundingVolume = contentBoundingVolume; } // combine() is used to pass through any additional properties the // user specified such as extras or extensions contentJsons.push(combine(contentJson, implicitTileset.contentHeaders[i])); } var childGeometricError = getGeometricError( tileMetadata, implicitTileset, implicitCoordinates ); var tileJson = { boundingVolume: boundingVolume, geometricError: childGeometricError, refine: implicitTileset.refine, }; if (contentJsons.length === 1) { tileJson.content = contentJsons[0]; } else if (contentJsons.length > 1) { tileJson.extensions = { "3DTILES_multiple_contents": { content: contentJsons, }, }; } // combine() is used to pass through any additional properties the // user specified such as extras or extensions. var deep = true; var rootHeader = clone(implicitTileset.tileHeader, deep); delete rootHeader.boundingVolume; var combinedTileJson = combine(tileJson, rootHeader, deep); var childTile = makeTile( implicitContent, implicitTileset.baseResource, combinedTileJson, parentTile ); childTile.implicitCoordinates = implicitCoordinates; childTile.implicitSubtree = subtree; childTile.metadata = tileMetadata; return childTile; } /** * Checks whether the bounding volume's heights can be updated. * Returns true if the minimumHeight/maximumHeight parameter * is defined and the bounding volume is a region or S2 cell. * * @param {Object} [boundingVolume] The bounding voume * @param {Object} [tileBounds] The tile bounds * @param {Number} [tileBounds.minimumHeight] The minimum height * @param {Number} [tileBounds.maximumHeight] The maximum height * @returns {Boolean} Whether the bounding volume's heights can be updated * @private */ function canUpdateHeights(boundingVolume, tileBounds) { return ( defined(boundingVolume) && defined(tileBounds) && (defined(tileBounds.minimumHeight) || defined(tileBounds.maximumHeight)) && (hasExtension(boundingVolume, "3DTILES_bounding_volume_S2") || defined(boundingVolume.region)) ); } /** * Update the minimum and maximum height of the bounding volume. * This is typically used to tighten a bounding volume using the *TILE_MINIMUM_HEIGHT and TILE_MAXIMUM_HEIGHT
* semantics. Heights are only updated if the respective
* minimumHeight/maximumHeight parameter is defined and the
* bounding volume is a region or S2 cell.
*
* @param {Object} boundingVolume The bounding voume
* @param {Object} [tileBounds] The tile bounds
* @param {Number} [tileBounds.minimumHeight] The new minimum height
* @param {Number} [tileBounds.maximumHeight] The new maximum height
* @private
*/
function updateHeights(boundingVolume, tileBounds) {
if (
hasExtension(boundingVolume, "3DTILES_bounding_volume_S2") &&
defined(tileBounds)
) {
updateS2CellHeights(
boundingVolume.extensions["3DTILES_bounding_volume_S2"],
tileBounds.minimumHeight,
tileBounds.maximumHeight
);
} else if (defined(boundingVolume.region) && defined(tileBounds)) {
updateRegionHeights(
boundingVolume.region,
tileBounds.minimumHeight,
tileBounds.maximumHeight
);
}
}
/**
* For a bounding region, update the minimum and maximum height. This
* is typically used to tighten a bounding volume using the
* TILE_MINIMUM_HEIGHT and TILE_MAXIMUM_HEIGHT
* semantics. Heights are only updated if the respective
* minimumHeight/maximumHeight parameter is defined.
*
* @param {Array} region A 6-element array describing the bounding region
* @param {Number} [minimumHeight] The new minimum height
* @param {Number} [maximumHeight] The new maximum height
* @private
*/
function updateRegionHeights(region, minimumHeight, maximumHeight) {
if (defined(minimumHeight)) {
region[4] = minimumHeight;
}
if (defined(maximumHeight)) {
region[5] = maximumHeight;
}
}
/**
* For a bounding S2 cell, update the minimum and maximum height. This
* is typically used to tighten a bounding volume using the
* TILE_MINIMUM_HEIGHT and TILE_MAXIMUM_HEIGHT
* semantics. Heights are only updated if the respective
* minimumHeight/maximumHeight parameter is defined.
*
* @param {Object} s2CellVolume An object describing the S2 cell
* @param {Number} [minimumHeight] The new minimum height
* @param {Number} [maximumHeight] The new maximum height
* @private
*/
function updateS2CellHeights(s2CellVolume, minimumHeight, maximumHeight) {
if (defined(minimumHeight)) {
s2CellVolume.minimumHeight = minimumHeight;
}
if (defined(maximumHeight)) {
s2CellVolume.maximumHeight = maximumHeight;
}
}
/**
* Gets the tile's bounding volume, which may be specified via
* metadata semantics such as TILE_BOUNDING_BOX or implicitly
* derived from the implicit root tile's bounding volume.
* * Priority of bounding volume types: *
* Priority of bounding volume types: *
undefined if there is no bounding volume
* @private
*/
function getContentBoundingVolume(tileBoundingVolume, contentBounds) {
// content bounding volumes can only be specified via
// metadata semantics such as CONTENT_BOUNDING_BOX
var contentBoundingVolume;
if (defined(contentBounds)) {
contentBoundingVolume = contentBounds.boundingVolume;
}
// The CONTENT_MINIMUM_HEIGHT and CONTENT_MAXIMUM_HEIGHT metadata semantics
// can be used to tighten the bounding volume
if (canUpdateHeights(contentBoundingVolume, contentBounds)) {
updateHeights(contentBoundingVolume, contentBounds);
} else if (canUpdateHeights(tileBoundingVolume, contentBounds)) {
contentBoundingVolume = clone(tileBoundingVolume, true);
updateHeights(contentBoundingVolume, contentBounds);
}
return contentBoundingVolume;
}
/**
* Given the coordinates of a tile, derive its bounding volume from the root.
*
* @param {ImplicitTileset} implicitTileset The implicit tileset struct which holds the root bounding volume
* @param {ImplicitTileCoordinates} implicitCoordinates The coordinates of the child tile
* @param {Number} childIndex The morton index of the child tile relative to its parent
* @param {Boolean} parentIsPlaceholderTile True if parentTile is a placeholder tile. This is true for the root of each subtree.
* @param {Cesium3DTile} parentTile The parent of the new child tile
* @returns {Object} An object containing the JSON for a bounding volume
* @private
*/
function deriveBoundingVolume(
implicitTileset,
implicitCoordinates,
childIndex,
parentIsPlaceholderTile,
parentTile
) {
var rootBoundingVolume = implicitTileset.boundingVolume;
if (hasExtension(rootBoundingVolume, "3DTILES_bounding_volume_S2")) {
return deriveBoundingVolumeS2(
parentIsPlaceholderTile,
parentTile,
childIndex,
implicitCoordinates.level,
implicitCoordinates.x,
implicitCoordinates.y,
implicitCoordinates.z
);
}
if (defined(rootBoundingVolume.region)) {
var childRegion = deriveBoundingRegion(
rootBoundingVolume.region,
implicitCoordinates.level,
implicitCoordinates.x,
implicitCoordinates.y,
implicitCoordinates.z
);
return {
region: childRegion,
};
}
var childBox = deriveBoundingBox(
rootBoundingVolume.box,
implicitCoordinates.level,
implicitCoordinates.x,
implicitCoordinates.y,
implicitCoordinates.z
);
return {
box: childBox,
};
}
/**
* Derive a bounding volume for a descendant tile (child, grandchild, etc.),
* assuming a quadtree or octree implicit tiling scheme. The (level, x, y, [z])
* coordinates are given to select the descendant tile and compute its position
* and dimensions.
* * If z is present, octree subdivision is used. Otherwise, quadtree subdivision * is used. Quadtrees are always divided at the midpoint of the the horizontal * dimensions, i.e. (x, y), leaving the z axis unchanged. *
* * @param {Boolean} parentIsPlaceholderTile True if parentTile is a placeholder tile. This is true for the root of each subtree. * @param {Cesium3DTile} parentTile The parent of the new child tile * @param {Number} childIndex The morton index of the child tile relative to its parent * @param {Number} level The level of the descendant tile relative to the root implicit tile * @param {Number} x The x coordinate of the descendant tile * @param {Number} y The y coordinate of the descendant tile * @param {Number} [z] The z coordinate of the descendant tile (octree only) * @returns {Object} An object with the 3DTILES_bounding_volume_S2 extension. * @private */ function deriveBoundingVolumeS2( parentIsPlaceholderTile, parentTile, childIndex, level, x, y, z ) { //>>includeStart('debug', pragmas.debug); Check.typeOf.bool("parentIsPlaceholderTile", parentIsPlaceholderTile); Check.typeOf.object("parentTile", parentTile); Check.typeOf.number("childIndex", childIndex); Check.typeOf.number("level", level); Check.typeOf.number("x", x); Check.typeOf.number("y", y); if (defined(z)) { Check.typeOf.number("z", z); } //>>includeEnd('debug'); var boundingVolumeS2 = parentTile._boundingVolume; // Handle the placeholder tile case, where we just duplicate the placeholder's bounding volume. if (parentIsPlaceholderTile) { return { extensions: { "3DTILES_bounding_volume_S2": { token: S2Cell.getTokenFromId(boundingVolumeS2.s2Cell._cellId), minimumHeight: boundingVolumeS2.minimumHeight, maximumHeight: boundingVolumeS2.maximumHeight, }, }, }; } // Extract the first 3 face bits from the 64-bit S2 cell ID. // eslint-disable-next-line var face = Number(parentTile._boundingVolume.s2Cell._cellId >> BigInt(61)); // The Hilbert curve is rotated for the "odd" faces on the S2 Earthcube. // See http://s2geometry.io/devguide/img/s2cell_global.jpg var position = face % 2 === 0 ? HilbertOrder.encode2D(level, x, y) : HilbertOrder.encode2D(level, y, x); // eslint-disable-next-line var cell = S2Cell.fromFacePositionLevel(face, BigInt(position), level); var minHeight, maxHeight; if (defined(z)) { var midpointHeight = (boundingVolumeS2.maximumHeight + boundingVolumeS2.minimumHeight) / 2; minHeight = childIndex < 4 ? boundingVolumeS2.minimumHeight : midpointHeight; maxHeight = childIndex < 4 ? midpointHeight : boundingVolumeS2.maximumHeight; } else { minHeight = boundingVolumeS2.minimumHeight; maxHeight = boundingVolumeS2.maximumHeight; } return { extensions: { "3DTILES_bounding_volume_S2": { token: S2Cell.getTokenFromId(cell._cellId), minimumHeight: minHeight, maximumHeight: maxHeight, }, }, }; } var scratchScaleFactors = new Cartesian3(); var scratchRootCenter = new Cartesian3(); var scratchCenter = new Cartesian3(); var scratchHalfAxes = new Matrix3(); /** * Derive a bounding volume for a descendant tile (child, grandchild, etc.), * assuming a quadtree or octree implicit tiling scheme. The (level, x, y, [z]) * coordinates are given to select the descendant tile and compute its position * and dimensions. ** If z is present, octree subdivision is used. Otherwise, quadtree subdivision * is used. Quadtrees are always divided at the midpoint of the the horizontal * dimensions, i.e. (x, y), leaving the z axis unchanged. *
** This computes the child volume directly from the root bounding volume rather * than recursively subdividing to minimize floating point error. *
* * @param {Number[]} rootBox An array of 12 numbers representing the bounding box of the root tile * @param {Number} level The level of the descendant tile relative to the root implicit tile * @param {Number} x The x coordinate of the descendant tile * @param {Number} y The y coordinate of the descendant tile * @param {Number} [z] The z coordinate of the descendant tile (octree only) * @returns {Number[]} An array of 12 numbers representing the bounding box of the descendant tile. * @private */ function deriveBoundingBox(rootBox, level, x, y, z) { //>>includeStart('debug', pragmas.debug); Check.typeOf.object("rootBox", rootBox); Check.typeOf.number("level", level); Check.typeOf.number("x", x); Check.typeOf.number("y", y); if (defined(z)) { Check.typeOf.number("z", z); } //>>includeEnd('debug'); if (level === 0) { return rootBox; } var rootCenter = Cartesian3.unpack(rootBox, 0, scratchRootCenter); var rootHalfAxes = Matrix3.unpack(rootBox, 3, scratchHalfAxes); var tileScale = Math.pow(2, -level); var modelSpaceX = -1 + (2 * x + 1) * tileScale; var modelSpaceY = -1 + (2 * y + 1) * tileScale; var modelSpaceZ = 0; var scaleFactors = Cartesian3.fromElements( tileScale, tileScale, 1, scratchScaleFactors ); if (defined(z)) { modelSpaceZ = -1 + (2 * z + 1) * tileScale; scaleFactors.z = tileScale; } var center = Cartesian3.fromElements( modelSpaceX, modelSpaceY, modelSpaceZ, scratchCenter ); center = Matrix3.multiplyByVector(rootHalfAxes, center, scratchCenter); center = Cartesian3.add(center, rootCenter, scratchCenter); var halfAxes = Matrix3.clone(rootHalfAxes); halfAxes = Matrix3.multiplyByScale(halfAxes, scaleFactors, halfAxes); var childBox = new Array(12); Cartesian3.pack(center, childBox); Matrix3.pack(halfAxes, childBox, 3); return childBox; } var scratchRectangle = new Rectangle(); /** * Derive a bounding volume for a descendant tile (child, grandchild, etc.), * assuming a quadtree or octree implicit tiling scheme. The (level, x, y, [z]) * coordinates are given to select the descendant tile and compute its position * and dimensions. ** If z is present, octree subdivision is used. Otherwise, quadtree subdivision * is used. Quadtrees are always divided at the midpoint of the the horizontal * dimensions, i.e. (mid_longitude, mid_latitude), leaving the height values * unchanged. *
** This computes the child volume directly from the root bounding volume rather * than recursively subdividing to minimize floating point error. *
* @param {Number[]} rootRegion An array of 6 numbers representing the root implicit tile * @param {Number} level The level of the descendant tile relative to the root implicit tile * @param {Number} x The x coordinate of the descendant tile * @param {Number} y The x coordinate of the descendant tile * @param {Number} [z] The z coordinate of the descendant tile (octree only) * @returns {Number[]} An array of 6 numbers representing the bounding region of the descendant tile * @private */ function deriveBoundingRegion(rootRegion, level, x, y, z) { //>>includeStart('debug', pragmas.debug); Check.typeOf.object("rootRegion", rootRegion); Check.typeOf.number("level", level); Check.typeOf.number("x", x); Check.typeOf.number("y", y); if (defined(z)) { Check.typeOf.number("z", z); } //>>includeEnd('debug'); if (level === 0) { return rootRegion.slice(); } var rectangle = Rectangle.unpack(rootRegion, 0, scratchRectangle); var rootMinimumHeight = rootRegion[4]; var rootMaximumHeight = rootRegion[5]; var tileScale = Math.pow(2, -level); var childWidth = tileScale * rectangle.width; var west = CesiumMath.negativePiToPi(rectangle.west + x * childWidth); var east = CesiumMath.negativePiToPi(west + childWidth); var childHeight = tileScale * rectangle.height; var south = CesiumMath.negativePiToPi(rectangle.south + y * childHeight); var north = CesiumMath.negativePiToPi(south + childHeight); // Height is only subdivided for octrees; It remains constant for quadtrees. var minimumHeight = rootMinimumHeight; var maximumHeight = rootMaximumHeight; if (defined(z)) { var childThickness = tileScale * (rootMaximumHeight - rootMinimumHeight); minimumHeight += z * childThickness; maximumHeight = minimumHeight + childThickness; } return [west, south, east, north, minimumHeight, maximumHeight]; } /** * Create a placeholder 3D Tile whose content will be an Implicit3DTileContent * for lazy evaluation of a child subtree. * * @param {Implicit3DTileContent} content The content object. * @param {Cesium3DTile} parentTile The parent of the new child subtree. * @param {Number} childIndex The morton index of the child tile relative to its parent * @returns {Cesium3DTile} The new placeholder tile * @private */ function makePlaceholderChildSubtree(content, parentTile, childIndex) { var implicitTileset = content._implicitTileset; var implicitCoordinates = parentTile.implicitCoordinates.getChildCoordinates( childIndex ); var childBoundingVolume = deriveBoundingVolume( implicitTileset, implicitCoordinates, childIndex, false, parentTile ); // Ignore tile metadata when computing geometric error for the placeholder tile // since the child subtree's metadata hasn't been loaded yet. // The actual geometric error will be computed in deriveChildTile. var childGeometricError = getGeometricError( undefined, implicitTileset, implicitCoordinates ); var childContentUri = implicitTileset.subtreeUriTemplate.getDerivedResource({ templateValues: implicitCoordinates.getTemplateValues(), }).url; var tileJson = { boundingVolume: childBoundingVolume, geometricError: childGeometricError, refine: implicitTileset.refine, content: { uri: childContentUri, }, }; var tile = makeTile( content, implicitTileset.baseResource, tileJson, parentTile ); tile.implicitTileset = implicitTileset; tile.implicitCoordinates = implicitCoordinates; return tile; } /** * Make a {@link Cesium3DTile}. This uses the content's tile's constructor instead * of importing Cesium3DTile. This is to avoid a circular dependency between * this file and Cesium3DTile.js * @param {Implicit3DTileContent} content The implicit content * @param {Resource} baseResource The base resource for the tileset * @param {Object} tileJson The JSON header for the tile * @param {Cesium3DTile} parentTile The parent of the new tile * @returns {Cesium3DTile} The newly created tile. * @private */ function makeTile(content, baseResource, tileJson, parentTile) { var Cesium3DTile = content._tile.constructor; return new Cesium3DTile(content._tileset, baseResource, tileJson, parentTile); } /** * Part of the {@link Cesium3DTileContent} interface.Implicit3DTileContent
* always returns false since a tile of this type does not have any features.
* @private
*/
Implicit3DTileContent.prototype.hasProperty = function (batchId, name) {
return false;
};
/**
* Part of the {@link Cesium3DTileContent} interface. Implicit3DTileContent
* always returns undefined since a tile of this type does not have any features.
* @private
*/
Implicit3DTileContent.prototype.getFeature = function (batchId) {
return undefined;
};
Implicit3DTileContent.prototype.applyDebugSettings = function (
enabled,
color
) {};
Implicit3DTileContent.prototype.applyStyle = function (style) {};
Implicit3DTileContent.prototype.update = function (tileset, frameState) {};
Implicit3DTileContent.prototype.isDestroyed = function () {
return false;
};
Implicit3DTileContent.prototype.destroy = function () {
this._implicitSubtree =
this._implicitSubtree && this._implicitSubtree.destroy();
return destroyObject(this);
};
// Exposed for testing
Implicit3DTileContent._deriveBoundingBox = deriveBoundingBox;
Implicit3DTileContent._deriveBoundingRegion = deriveBoundingRegion;
Implicit3DTileContent._deriveBoundingVolumeS2 = deriveBoundingVolumeS2;