You cannot select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
Cesium-Prequel/Source/Scene/Cesium3DTilesetMostDetailed...

143 lines
3.3 KiB
JavaScript

import Intersect from "../Core/Intersect.js";
import ManagedArray from "../Core/ManagedArray.js";
import Cesium3DTileRefine from "./Cesium3DTileRefine.js";
/**
* Traversal that loads all leaves that intersect the camera frustum.
* Used to determine ray-tileset intersections during a pickFromRayMostDetailed call.
*
* @private
*/
function Cesium3DTilesetMostDetailedTraversal() {}
var traversal = {
stack: new ManagedArray(),
stackMaximumLength: 0,
};
Cesium3DTilesetMostDetailedTraversal.selectTiles = function (
tileset,
frameState
) {
tileset._selectedTiles.length = 0;
tileset._requestedTiles.length = 0;
tileset._hasMixedContent = false;
var ready = true;
var root = tileset.root;
root.updateVisibility(frameState);
if (!isVisible(root)) {
return ready;
}
var stack = traversal.stack;
stack.push(tileset.root);
while (stack.length > 0) {
traversal.stackMaximumLength = Math.max(
traversal.stackMaximumLength,
stack.length
);
var tile = stack.pop();
var add = tile.refine === Cesium3DTileRefine.ADD;
var replace = tile.refine === Cesium3DTileRefine.REPLACE;
var traverse = canTraverse(tileset, tile);
if (traverse) {
updateAndPushChildren(tileset, tile, stack, frameState);
}
if (add || (replace && !traverse)) {
loadTile(tileset, tile);
touchTile(tileset, tile, frameState);
selectDesiredTile(tileset, tile, frameState);
if (!hasEmptyContent(tile) && !tile.contentAvailable) {
ready = false;
}
}
visitTile(tileset);
}
traversal.stack.trim(traversal.stackMaximumLength);
return ready;
};
function isVisible(tile) {
return tile._visible && tile._inRequestVolume;
}
function hasEmptyContent(tile) {
return tile.hasEmptyContent || tile.hasTilesetContent;
}
function hasUnloadedContent(tile) {
return !hasEmptyContent(tile) && tile.contentUnloaded;
}
function canTraverse(tileset, tile) {
if (tile.children.length === 0) {
return false;
}
if (tile.hasTilesetContent) {
// Traverse external tileset to visit its root tile
// Don't traverse if the subtree is expired because it will be destroyed
return !tile.contentExpired;
}
if (tile.hasEmptyContent) {
return true;
}
return true; // Keep traversing until a leave is hit
}
function updateAndPushChildren(tileset, tile, stack, frameState) {
var children = tile.children;
var length = children.length;
for (var i = 0; i < length; ++i) {
var child = children[i];
child.updateVisibility(frameState);
if (isVisible(child)) {
stack.push(child);
}
}
}
function loadTile(tileset, tile) {
if (hasUnloadedContent(tile) || tile.contentExpired) {
tile._priority = 0.0; // Highest priority
tileset._requestedTiles.push(tile);
}
}
function touchTile(tileset, tile, frameState) {
if (tile._touchedFrame === frameState.frameNumber) {
// Prevents another pass from touching the frame again
return;
}
tileset._cache.touch(tile);
tile._touchedFrame = frameState.frameNumber;
}
function visitTile(tileset) {
++tileset.statistics.visited;
}
function selectDesiredTile(tileset, tile, frameState) {
if (
tile.contentAvailable &&
tile.contentVisibility(frameState) !== Intersect.OUTSIDE
) {
tileset._selectedTiles.push(tile);
}
}
export default Cesium3DTilesetMostDetailedTraversal;