123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549 |
- import AxisAlignedBoundingBox from "../Core/AxisAlignedBoundingBox.js";
- import Cartesian2 from "../Core/Cartesian2.js";
- import Cartesian3 from "../Core/Cartesian3.js";
- import Cartographic from "../Core/Cartographic.js";
- import defined from "../Core/defined.js";
- import Ellipsoid from "../Core/Ellipsoid.js";
- import EllipsoidalOccluder from "../Core/EllipsoidalOccluder.js";
- import IndexDatatype from "../Core/IndexDatatype.js";
- import CesiumMath from "../Core/Math.js";
- import Matrix4 from "../Core/Matrix4.js";
- import Rectangle from "../Core/Rectangle.js";
- import TerrainEncoding from "../Core/TerrainEncoding.js";
- import TerrainProvider from "../Core/TerrainProvider.js";
- import Transforms from "../Core/Transforms.js";
- import WebMercatorProjection from "../Core/WebMercatorProjection.js";
- import createTaskProcessorWorker from "./createTaskProcessorWorker.js";
- const maxShort = 32767;
- const cartesian3Scratch = new Cartesian3();
- const scratchMinimum = new Cartesian3();
- const scratchMaximum = new Cartesian3();
- const cartographicScratch = new Cartographic();
- const toPack = new Cartesian2();
- function createVerticesFromQuantizedTerrainMesh(
- parameters,
- transferableObjects
- ) {
- const quantizedVertices = parameters.quantizedVertices;
- const quantizedVertexCount = quantizedVertices.length / 3;
- const octEncodedNormals = parameters.octEncodedNormals;
- const edgeVertexCount =
- parameters.westIndices.length +
- parameters.eastIndices.length +
- parameters.southIndices.length +
- parameters.northIndices.length;
- const includeWebMercatorT = parameters.includeWebMercatorT;
- const exaggeration = parameters.exaggeration;
- const exaggerationRelativeHeight = parameters.exaggerationRelativeHeight;
- const hasExaggeration = exaggeration !== 1.0;
- const includeGeodeticSurfaceNormals = hasExaggeration;
- const rectangle = Rectangle.clone(parameters.rectangle);
- const west = rectangle.west;
- const south = rectangle.south;
- const east = rectangle.east;
- const north = rectangle.north;
- const ellipsoid = Ellipsoid.clone(parameters.ellipsoid);
- const minimumHeight = parameters.minimumHeight;
- const maximumHeight = parameters.maximumHeight;
- const center = parameters.relativeToCenter;
- const fromENU = Transforms.eastNorthUpToFixedFrame(center, ellipsoid);
- const toENU = Matrix4.inverseTransformation(fromENU, new Matrix4());
- let southMercatorY;
- let oneOverMercatorHeight;
- if (includeWebMercatorT) {
- southMercatorY = WebMercatorProjection.geodeticLatitudeToMercatorAngle(
- south
- );
- oneOverMercatorHeight =
- 1.0 /
- (WebMercatorProjection.geodeticLatitudeToMercatorAngle(north) -
- southMercatorY);
- }
- const uBuffer = quantizedVertices.subarray(0, quantizedVertexCount);
- const vBuffer = quantizedVertices.subarray(
- quantizedVertexCount,
- 2 * quantizedVertexCount
- );
- const heightBuffer = quantizedVertices.subarray(
- quantizedVertexCount * 2,
- 3 * quantizedVertexCount
- );
- const hasVertexNormals = defined(octEncodedNormals);
- const uvs = new Array(quantizedVertexCount);
- const heights = new Array(quantizedVertexCount);
- const positions = new Array(quantizedVertexCount);
- const webMercatorTs = includeWebMercatorT
- ? new Array(quantizedVertexCount)
- : [];
- const geodeticSurfaceNormals = includeGeodeticSurfaceNormals
- ? new Array(quantizedVertexCount)
- : [];
- const minimum = scratchMinimum;
- minimum.x = Number.POSITIVE_INFINITY;
- minimum.y = Number.POSITIVE_INFINITY;
- minimum.z = Number.POSITIVE_INFINITY;
- const maximum = scratchMaximum;
- maximum.x = Number.NEGATIVE_INFINITY;
- maximum.y = Number.NEGATIVE_INFINITY;
- maximum.z = Number.NEGATIVE_INFINITY;
- let minLongitude = Number.POSITIVE_INFINITY;
- let maxLongitude = Number.NEGATIVE_INFINITY;
- let minLatitude = Number.POSITIVE_INFINITY;
- let maxLatitude = Number.NEGATIVE_INFINITY;
- for (let i = 0; i < quantizedVertexCount; ++i) {
- const rawU = uBuffer[i];
- const rawV = vBuffer[i];
- const u = rawU / maxShort;
- const v = rawV / maxShort;
- const height = CesiumMath.lerp(
- minimumHeight,
- maximumHeight,
- heightBuffer[i] / maxShort
- );
- cartographicScratch.longitude = CesiumMath.lerp(west, east, u);
- cartographicScratch.latitude = CesiumMath.lerp(south, north, v);
- cartographicScratch.height = height;
- minLongitude = Math.min(cartographicScratch.longitude, minLongitude);
- maxLongitude = Math.max(cartographicScratch.longitude, maxLongitude);
- minLatitude = Math.min(cartographicScratch.latitude, minLatitude);
- maxLatitude = Math.max(cartographicScratch.latitude, maxLatitude);
- const position = ellipsoid.cartographicToCartesian(cartographicScratch);
- uvs[i] = new Cartesian2(u, v);
- heights[i] = height;
- positions[i] = position;
- if (includeWebMercatorT) {
- webMercatorTs[i] =
- (WebMercatorProjection.geodeticLatitudeToMercatorAngle(
- cartographicScratch.latitude
- ) -
- southMercatorY) *
- oneOverMercatorHeight;
- }
- if (includeGeodeticSurfaceNormals) {
- geodeticSurfaceNormals[i] = ellipsoid.geodeticSurfaceNormal(position);
- }
- Matrix4.multiplyByPoint(toENU, position, cartesian3Scratch);
- Cartesian3.minimumByComponent(cartesian3Scratch, minimum, minimum);
- Cartesian3.maximumByComponent(cartesian3Scratch, maximum, maximum);
- }
- const westIndicesSouthToNorth = copyAndSort(parameters.westIndices, function (
- a,
- b
- ) {
- return uvs[a].y - uvs[b].y;
- });
- const eastIndicesNorthToSouth = copyAndSort(parameters.eastIndices, function (
- a,
- b
- ) {
- return uvs[b].y - uvs[a].y;
- });
- const southIndicesEastToWest = copyAndSort(parameters.southIndices, function (
- a,
- b
- ) {
- return uvs[b].x - uvs[a].x;
- });
- const northIndicesWestToEast = copyAndSort(parameters.northIndices, function (
- a,
- b
- ) {
- return uvs[a].x - uvs[b].x;
- });
- let occludeePointInScaledSpace;
- if (minimumHeight < 0.0) {
- // Horizon culling point needs to be recomputed since the tile is at least partly under the ellipsoid.
- const occluder = new EllipsoidalOccluder(ellipsoid);
- occludeePointInScaledSpace = occluder.computeHorizonCullingPointPossiblyUnderEllipsoid(
- center,
- positions,
- minimumHeight
- );
- }
- let hMin = minimumHeight;
- hMin = Math.min(
- hMin,
- findMinMaxSkirts(
- parameters.westIndices,
- parameters.westSkirtHeight,
- heights,
- uvs,
- rectangle,
- ellipsoid,
- toENU,
- minimum,
- maximum
- )
- );
- hMin = Math.min(
- hMin,
- findMinMaxSkirts(
- parameters.southIndices,
- parameters.southSkirtHeight,
- heights,
- uvs,
- rectangle,
- ellipsoid,
- toENU,
- minimum,
- maximum
- )
- );
- hMin = Math.min(
- hMin,
- findMinMaxSkirts(
- parameters.eastIndices,
- parameters.eastSkirtHeight,
- heights,
- uvs,
- rectangle,
- ellipsoid,
- toENU,
- minimum,
- maximum
- )
- );
- hMin = Math.min(
- hMin,
- findMinMaxSkirts(
- parameters.northIndices,
- parameters.northSkirtHeight,
- heights,
- uvs,
- rectangle,
- ellipsoid,
- toENU,
- minimum,
- maximum
- )
- );
- const aaBox = new AxisAlignedBoundingBox(minimum, maximum, center);
- const encoding = new TerrainEncoding(
- center,
- aaBox,
- hMin,
- maximumHeight,
- fromENU,
- hasVertexNormals,
- includeWebMercatorT,
- includeGeodeticSurfaceNormals,
- exaggeration,
- exaggerationRelativeHeight
- );
- const vertexStride = encoding.stride;
- const size =
- quantizedVertexCount * vertexStride + edgeVertexCount * vertexStride;
- const vertexBuffer = new Float32Array(size);
- let bufferIndex = 0;
- for (let j = 0; j < quantizedVertexCount; ++j) {
- if (hasVertexNormals) {
- const n = j * 2.0;
- toPack.x = octEncodedNormals[n];
- toPack.y = octEncodedNormals[n + 1];
- }
- bufferIndex = encoding.encode(
- vertexBuffer,
- bufferIndex,
- positions[j],
- uvs[j],
- heights[j],
- toPack,
- webMercatorTs[j],
- geodeticSurfaceNormals[j]
- );
- }
- const edgeTriangleCount = Math.max(0, (edgeVertexCount - 4) * 2);
- const indexBufferLength = parameters.indices.length + edgeTriangleCount * 3;
- const indexBuffer = IndexDatatype.createTypedArray(
- quantizedVertexCount + edgeVertexCount,
- indexBufferLength
- );
- indexBuffer.set(parameters.indices, 0);
- const percentage = 0.0001;
- const lonOffset = (maxLongitude - minLongitude) * percentage;
- const latOffset = (maxLatitude - minLatitude) * percentage;
- const westLongitudeOffset = -lonOffset;
- const westLatitudeOffset = 0.0;
- const eastLongitudeOffset = lonOffset;
- const eastLatitudeOffset = 0.0;
- const northLongitudeOffset = 0.0;
- const northLatitudeOffset = latOffset;
- const southLongitudeOffset = 0.0;
- const southLatitudeOffset = -latOffset;
- // Add skirts.
- let vertexBufferIndex = quantizedVertexCount * vertexStride;
- addSkirt(
- vertexBuffer,
- vertexBufferIndex,
- westIndicesSouthToNorth,
- encoding,
- heights,
- uvs,
- octEncodedNormals,
- ellipsoid,
- rectangle,
- parameters.westSkirtHeight,
- southMercatorY,
- oneOverMercatorHeight,
- westLongitudeOffset,
- westLatitudeOffset
- );
- vertexBufferIndex += parameters.westIndices.length * vertexStride;
- addSkirt(
- vertexBuffer,
- vertexBufferIndex,
- southIndicesEastToWest,
- encoding,
- heights,
- uvs,
- octEncodedNormals,
- ellipsoid,
- rectangle,
- parameters.southSkirtHeight,
- southMercatorY,
- oneOverMercatorHeight,
- southLongitudeOffset,
- southLatitudeOffset
- );
- vertexBufferIndex += parameters.southIndices.length * vertexStride;
- addSkirt(
- vertexBuffer,
- vertexBufferIndex,
- eastIndicesNorthToSouth,
- encoding,
- heights,
- uvs,
- octEncodedNormals,
- ellipsoid,
- rectangle,
- parameters.eastSkirtHeight,
- southMercatorY,
- oneOverMercatorHeight,
- eastLongitudeOffset,
- eastLatitudeOffset
- );
- vertexBufferIndex += parameters.eastIndices.length * vertexStride;
- addSkirt(
- vertexBuffer,
- vertexBufferIndex,
- northIndicesWestToEast,
- encoding,
- heights,
- uvs,
- octEncodedNormals,
- ellipsoid,
- rectangle,
- parameters.northSkirtHeight,
- southMercatorY,
- oneOverMercatorHeight,
- northLongitudeOffset,
- northLatitudeOffset
- );
- TerrainProvider.addSkirtIndices(
- westIndicesSouthToNorth,
- southIndicesEastToWest,
- eastIndicesNorthToSouth,
- northIndicesWestToEast,
- quantizedVertexCount,
- indexBuffer,
- parameters.indices.length
- );
- transferableObjects.push(vertexBuffer.buffer, indexBuffer.buffer);
- return {
- vertices: vertexBuffer.buffer,
- indices: indexBuffer.buffer,
- westIndicesSouthToNorth: westIndicesSouthToNorth,
- southIndicesEastToWest: southIndicesEastToWest,
- eastIndicesNorthToSouth: eastIndicesNorthToSouth,
- northIndicesWestToEast: northIndicesWestToEast,
- vertexStride: vertexStride,
- center: center,
- minimumHeight: minimumHeight,
- maximumHeight: maximumHeight,
- occludeePointInScaledSpace: occludeePointInScaledSpace,
- encoding: encoding,
- indexCountWithoutSkirts: parameters.indices.length,
- };
- }
- function findMinMaxSkirts(
- edgeIndices,
- edgeHeight,
- heights,
- uvs,
- rectangle,
- ellipsoid,
- toENU,
- minimum,
- maximum
- ) {
- let hMin = Number.POSITIVE_INFINITY;
- const north = rectangle.north;
- const south = rectangle.south;
- let east = rectangle.east;
- const west = rectangle.west;
- if (east < west) {
- east += CesiumMath.TWO_PI;
- }
- const length = edgeIndices.length;
- for (let i = 0; i < length; ++i) {
- const index = edgeIndices[i];
- const h = heights[index];
- const uv = uvs[index];
- cartographicScratch.longitude = CesiumMath.lerp(west, east, uv.x);
- cartographicScratch.latitude = CesiumMath.lerp(south, north, uv.y);
- cartographicScratch.height = h - edgeHeight;
- const position = ellipsoid.cartographicToCartesian(
- cartographicScratch,
- cartesian3Scratch
- );
- Matrix4.multiplyByPoint(toENU, position, position);
- Cartesian3.minimumByComponent(position, minimum, minimum);
- Cartesian3.maximumByComponent(position, maximum, maximum);
- hMin = Math.min(hMin, cartographicScratch.height);
- }
- return hMin;
- }
- function addSkirt(
- vertexBuffer,
- vertexBufferIndex,
- edgeVertices,
- encoding,
- heights,
- uvs,
- octEncodedNormals,
- ellipsoid,
- rectangle,
- skirtLength,
- southMercatorY,
- oneOverMercatorHeight,
- longitudeOffset,
- latitudeOffset
- ) {
- const hasVertexNormals = defined(octEncodedNormals);
- const north = rectangle.north;
- const south = rectangle.south;
- let east = rectangle.east;
- const west = rectangle.west;
- if (east < west) {
- east += CesiumMath.TWO_PI;
- }
- const length = edgeVertices.length;
- for (let i = 0; i < length; ++i) {
- const index = edgeVertices[i];
- const h = heights[index];
- const uv = uvs[index];
- cartographicScratch.longitude =
- CesiumMath.lerp(west, east, uv.x) + longitudeOffset;
- cartographicScratch.latitude =
- CesiumMath.lerp(south, north, uv.y) + latitudeOffset;
- cartographicScratch.height = h - skirtLength;
- const position = ellipsoid.cartographicToCartesian(
- cartographicScratch,
- cartesian3Scratch
- );
- if (hasVertexNormals) {
- const n = index * 2.0;
- toPack.x = octEncodedNormals[n];
- toPack.y = octEncodedNormals[n + 1];
- }
- let webMercatorT;
- if (encoding.hasWebMercatorT) {
- webMercatorT =
- (WebMercatorProjection.geodeticLatitudeToMercatorAngle(
- cartographicScratch.latitude
- ) -
- southMercatorY) *
- oneOverMercatorHeight;
- }
- let geodeticSurfaceNormal;
- if (encoding.hasGeodeticSurfaceNormals) {
- geodeticSurfaceNormal = ellipsoid.geodeticSurfaceNormal(position);
- }
- vertexBufferIndex = encoding.encode(
- vertexBuffer,
- vertexBufferIndex,
- position,
- uv,
- cartographicScratch.height,
- toPack,
- webMercatorT,
- geodeticSurfaceNormal
- );
- }
- }
- function copyAndSort(typedArray, comparator) {
- let copy;
- if (typeof typedArray.slice === "function") {
- copy = typedArray.slice();
- if (typeof copy.sort !== "function") {
- // Sliced typed array isn't sortable, so we can't use it.
- copy = undefined;
- }
- }
- if (!defined(copy)) {
- copy = Array.prototype.slice.call(typedArray);
- }
- copy.sort(comparator);
- return copy;
- }
- export default createTaskProcessorWorker(
- createVerticesFromQuantizedTerrainMesh
- );
|