/*
* Copyright 2003-2006, 2009, 2017, 2020 United States Government, as represented
* by the Administrator of the National Aeronautics and Space Administration.
* All rights reserved.
*
* The NASAWorldWind/WebWorldWind platform is licensed under the Apache License,
* Version 2.0 (the "License"); you may not use this file except in compliance
* with the License. You may obtain a copy of the License
* at http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software distributed
* under the License is distributed on an "AS IS" BASIS, WITHOUT WARRANTIES OR
* CONDITIONS OF ANY KIND, either express or implied. See the License for the
* specific language governing permissions and limitations under the License.
*
* NASAWorldWind/WebWorldWind also contains the following 3rd party Open Source
* software:
*
* ES6-Promise – under MIT License
* libtess.js – SGI Free Software License B
* Proj4 – under MIT License
* JSZip – under MIT License
*
* A complete listing of 3rd Party software notices and licenses included in
* WebWorldWind can be found in the WebWorldWind 3rd-party notices and licenses
* PDF found in code directory.
*/
/**
* @exports AreaMeasurer
*/
define([
'../../geom/Angle',
'../../error/ArgumentError',
'../../geom/Location',
'../Logger',
'./MeasurerUtils',
'../../geom/Sector',
'../../geom/Vec3',
'../libtess'
],
function (Angle,
ArgumentError,
Location,
Logger,
MeasurerUtils,
Sector,
Vec3,
libtessDummy) {
'use strict';
/**
* Utility class to compute approximations of projected and surface (terrain following) area on a globe.
*
* <p>To properly compute surface area the measurer must be provided with a list of positions that describe a
* closed path - one which last position is equal to the first.</p>
*
* <p>Segments which are longer then the current maxSegmentLength will be subdivided along lines following the
* current pathType - WorldWind.LINEAR, WorldWind.RHUMB_LINE or WorldWind.GREAT_CIRCLE.</p>
*
* <p>Projected or non terrain following area is computed in a sinusoidal projection which is equivalent or
* equal area.
* Surface or terrain following area is approximated by sampling the path bounding sector with square cells
* along a grid. Cells which center is inside the path have their area estimated and summed according to the
* overall slope at the cell south-west corner.</p>
*
* @alias AreaMeasurer
* @constructor
* @param {WorldWindow} wwd The WorldWindow associated with AreaMeasurer.
* @throws {ArgumentError} If the specified WorldWindow is null or undefined.
*/
var AreaMeasurer = function (wwd) {
if (!wwd) {
throw new ArgumentError(
Logger.logMessage(Logger.LEVEL_SEVERE, "AreaMeasurer", "constructor", "missingWorldWindow"));
}
this.wwd = wwd;
// Private. Sampling grid max rows or cols
this.DEFAULT_AREA_SAMPLING_STEPS = 32;
// Private. Documentation is with the defined property below.
this._areaTerrainSamplingSteps = this.DEFAULT_AREA_SAMPLING_STEPS;
// Private. Documentation is with the defined property below.
this._maxSegmentLength = 100e3;
// Private. A list of positions with no segment longer then maxLength and elevations following terrain or not.
this.subdividedPositions = null;
// Private.
this.vecZ = new Vec3(0, 0, 1);
// Private. Reusable Location.
this.scratchLocation = new Location(0, 0);
};
Object.defineProperties(AreaMeasurer.prototype, {
/**
* The sampling grid maximum number of rows or columns for terrain following surface area approximation.
* @type {Number}
* @memberof AreaMeasurer.prototype
*/
areaTerrainSamplingSteps: {
get: function () {
return this._areaTerrainSamplingSteps;
},
set: function (value) {
this._areaTerrainSamplingSteps = value;
}
},
/**
* The maximum length a segment can have before being subdivided along a line following the current pathType.
* @type {Number}
* @memberof AreaMeasurer.prototype
*/
maxSegmentLength: {
get: function () {
return this._maxSegmentLength;
},
set: function (value) {
this._maxSegmentLength = value;
}
}
});
/**
* Get the sampling grid maximum number of rows or columns for terrain following surface area approximation.
*
* @param {Position[]} positions A list of positions describing a polygon
* @param {Boolean} followTerrain If true, the computed length will account for terrain deformations as if
* someone was walking along that path
* @param {String} pathType One of WorldWind.LINEAR, WorldWind.RHUMB_LINE or WorldWind.GREAT_CIRCLE
*
* @return {Number} area in square meters
*/
AreaMeasurer.prototype.getArea = function (positions, followTerrain, pathType) {
var globe = this.wwd.globe;
if (followTerrain) {
return this.computeSurfaceAreaSampling(globe, positions, pathType);
}
return this.computeProjectedAreaGeometry(globe, positions, pathType);
};
/**
* Sample the path bounding sector with square cells which area are approximated according to the surface normal
* at the cell south-west corner.
*
* @param {Globe} globe
* @param {Position[]} positions
* @param {String} pathType One of WorldWind.LINEAR, WorldWind.RHUMB_LINE or WorldWind.GREAT_CIRCLE
*
* @return {Number} area in square meters
*/
AreaMeasurer.prototype.computeSurfaceAreaSampling = function (globe, positions, pathType) {
var sector = new Sector(0, 0, 0, 0);
sector.setToBoundingSector(positions);
// Subdivide long segments if needed
this.subdividedPositions = MeasurerUtils.subdividePositions(globe, positions, true, pathType,
this._maxSegmentLength);
// Sample the bounding sector with cells about the same length in side - squares
var steps = Math.max(this.DEFAULT_AREA_SAMPLING_STEPS, this._areaTerrainSamplingSteps);
var deltaLatRadians = sector.deltaLatitude() * Angle.DEGREES_TO_RADIANS;
var deltaLonRadians = sector.deltaLongitude() * Angle.DEGREES_TO_RADIANS;
var stepsRadians = Math.max(deltaLatRadians / steps, deltaLonRadians / steps);
var latSteps = Math.round(deltaLatRadians / stepsRadians);
var lonSteps = Math.round(deltaLonRadians / stepsRadians *
Math.cos(sector.centroidLatitude() * Angle.DEGREES_TO_RADIANS));
var latStepRadians = deltaLatRadians / latSteps;
var lonStepRadians = deltaLonRadians / lonSteps;
var area = 0;
for (var i = 0; i < latSteps; i++) {
var lat = sector.minLatitude * Angle.DEGREES_TO_RADIANS + latStepRadians * i;
// Compute this latitude row cells area
var radius = globe.radiusAt((lat + latStepRadians / 2) * Angle.RADIANS_TO_DEGREES, sector.centroidLongitude());
var cellWidth = lonStepRadians * radius * Math.cos(lat + latStepRadians / 2);
var cellHeight = latStepRadians * radius;
var cellArea = cellWidth * cellHeight;
for (var j = 0; j < lonSteps; j++) {
var lon = sector.minLongitude * Angle.DEGREES_TO_RADIANS + lonStepRadians * j;
var minLat = lat * Angle.RADIANS_TO_DEGREES;
var maxLat = (lat + latStepRadians) * Angle.RADIANS_TO_DEGREES;
var minLon = lon * Angle.RADIANS_TO_DEGREES;
var maxLon = (lon + lonStepRadians) * Angle.RADIANS_TO_DEGREES;
var cellSector = new Sector(minLat, maxLat, minLon, maxLon);
var isLocationInside = MeasurerUtils.isLocationInside(cellSector.centroid(this.scratchLocation),
this.subdividedPositions);
if (isLocationInside) {
// Compute suface area using terrain normal in SW corner
// Corners elevation
var eleSW = globe.elevationAtLocation(minLat, minLon);
var eleSE = globe.elevationAtLocation(minLat, maxLon);
var eleNW = globe.elevationAtLocation(maxLat, minLon);
// Compute normal
var vx = new Vec3(cellWidth, 0, eleSE - eleSW);
var vy = new Vec3(0, cellHeight, eleNW - eleSW);
vx.normalize();
vy.normalize();
var normalSW = vx.cross(vy).normalize(); // point toward positive Z
// Compute slope factor
var tan = Math.tan(MeasurerUtils.angleBetweenVectors(this.vecZ, normalSW));
var slopeFactor = Math.sqrt(1 + tan * tan);
// Add cell area
area += (cellArea * slopeFactor);
}
}
}
return area;
};
/**
* Tessellate the path in lat-lon space, then sum each triangle area.
*
* @param {Globe} globe
* @param {Position[]} positions
* @param {String} pathType One of WorldWind.LINEAR, WorldWind.RHUMB_LINE or WorldWind.GREAT_CIRCLE
*
* @return {Number} area in square meters
*/
AreaMeasurer.prototype.computeProjectedAreaGeometry = function (globe, positions, pathType) {
// Subdivide long segments if needed
this.subdividedPositions = MeasurerUtils.subdividePositions(globe, positions, false, pathType,
this._maxSegmentLength);
// First: tessellate polygon
var verticesCount = this.subdividedPositions.length;
var firstPos = this.subdividedPositions[0];
var lastPos = this.subdividedPositions[verticesCount - 1];
if (firstPos.equals(lastPos)) {
verticesCount--;
}
var verts = [];
var idx = 0;
for (var i = 0; i < verticesCount; i++) {
var pos = this.subdividedPositions[i];
verts[idx++] = pos.longitude * Angle.DEGREES_TO_RADIANS;
verts[idx++] = pos.latitude * Angle.DEGREES_TO_RADIANS;
verts[idx++] = 0;
}
var triangles = this.tessellatePolygon(verticesCount, verts);
// Second: sum triangles area
var area = 0;
var triangleCount = triangles.length / 9;
for (i = 0; i < triangleCount; i++) {
idx = i * 9;
var triangle = [
triangles[idx + 0], triangles[idx + 1], triangles[idx + 2],
triangles[idx + 3], triangles[idx + 4], triangles[idx + 5],
triangles[idx + 6], triangles[idx + 7], triangles[idx + 8]
];
area += this.computeTriangleProjectedArea(globe, triangle);
}
return area;
};
/**
* Compute triangle area in a sinusoidal projection centered at the triangle center.
* Note sinusoidal projection is equivalent or equal area.
*
* @param {Globe} globe
* @param {Number[]} verts A list of 9 positions in radians describing a triangle
*
* @return {Number} area in square meters
*/
AreaMeasurer.prototype.computeTriangleProjectedArea = function (globe, verts) {
// http://www.mathopenref.com/coordtrianglearea.html
var ax = verts[0];
var ay = verts[1];
var bx = verts[3];
var by = verts[4];
var cx = verts[6];
var cy = verts[7];
var area = Math.abs(
ax * (by - cy) +
bx * (cy - ay) +
cx * (ay - by)
);
area /= 2;
var centerLon = (ax + bx + cx) / 3;
var centerLat = (ay + by + cy) / 3;
// Apply globe radius at triangle center and scale down area according to center latitude cosine
var radius = globe.radiusAt(centerLat * Angle.RADIANS_TO_DEGREES, centerLon * Angle.RADIANS_TO_DEGREES);
area *= Math.cos(centerLat) * radius * radius; // Square meter
return area;
};
/**
* Tessellate a Polygon
*
* @param {Number} count the number of vertices
* @param {Number[]} vertices A list of positions in radians
*
* @return {Number[]} a list of tessellated vertices
*/
AreaMeasurer.prototype.tessellatePolygon = function (count, vertices) {
var tess = new libtess.GluTesselator();
var triangles = [];
var coords;
tess.gluTessCallback(libtess.gluEnum.GLU_TESS_BEGIN, function (prim) {
if (prim !== libtess.primitiveType.GL_TRIANGLES) {
Logger.logMessage(Logger.LEVEL_WARNING, "AreaMeasurer", "tessellatePolygon",
"Tessellation error, primitive is not TRIANGLES.");
}
});
tess.gluTessCallback(libtess.gluEnum.GLU_TESS_VERTEX_DATA, function (data, tris) {
tris.push(data[0]);
tris.push(data[1]);
tris.push(data[2]);
});
//prevents triangle fans and strips
tess.gluTessCallback(libtess.gluEnum.GLU_TESS_EDGE_FLAG, function () {
});
tess.gluTessCallback(libtess.gluEnum.GLU_TESS_ERROR, function (errno) {
Logger.logMessage(Logger.LEVEL_WARNING, "AreaMeasurer", "tessellatePolygon",
"Tessellation error " + errno + ".");
});
// Tessellate the polygon.
tess.gluTessBeginPolygon(triangles);
tess.gluTessBeginContour();
for (var i = 0; i < count; i++) {
coords = vertices.slice(3 * i, 3 * i + 3);
tess.gluTessVertex(coords, coords);
}
tess.gluTessEndContour();
tess.gluTessEndPolygon();
return triangles;
};
return AreaMeasurer;
});