createCoplanarPolygonGeometry.js 19 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539
  1. /**
  2. * Cesium - https://github.com/CesiumGS/cesium
  3. *
  4. * Copyright 2011-2020 Cesium Contributors
  5. *
  6. * Licensed under the Apache License, Version 2.0 (the "License");
  7. * you may not use this file except in compliance with the License.
  8. * You may obtain a copy of the License at
  9. *
  10. * http://www.apache.org/licenses/LICENSE-2.0
  11. *
  12. * Unless required by applicable law or agreed to in writing, software
  13. * distributed under the License is distributed on an "AS IS" BASIS,
  14. * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
  15. * See the License for the specific language governing permissions and
  16. * limitations under the License.
  17. *
  18. * Columbus View (Pat. Pend.)
  19. *
  20. * Portions licensed separately.
  21. * See https://github.com/CesiumGS/cesium/blob/master/LICENSE.md for full licensing details.
  22. */
  23. define(['./when-54c2dc71', './Check-6c0211bc', './Math-1124a290', './Cartesian2-33d2657c', './Transforms-8be64844', './RuntimeError-2109023a', './WebGLConstants-76bb35d1', './ComponentDatatype-a26dd044', './GeometryAttribute-e9a8b203', './GeometryAttributes-4fcfcf40', './AttributeCompression-75249b5e', './GeometryPipeline-466ad516', './EncodedCartesian3-6c97231d', './IndexDatatype-25023891', './IntersectionTests-afc38163', './Plane-fa30fc46', './VertexFormat-4d8b817a', './GeometryInstance-8c9b4df5', './arrayRemoveDuplicates-0263f42c', './BoundingRectangle-dede91e8', './EllipsoidTangentPlane-ce6e380f', './OrientedBoundingBox-8897f474', './CoplanarPolygonGeometryLibrary-f703c864', './ArcType-dc1c5aee', './EllipsoidRhumbLine-5f1492e5', './PolygonPipeline-9f9b7763', './PolygonGeometryLibrary-98a03962'], function (when, Check, _Math, Cartesian2, Transforms, RuntimeError, WebGLConstants, ComponentDatatype, GeometryAttribute, GeometryAttributes, AttributeCompression, GeometryPipeline, EncodedCartesian3, IndexDatatype, IntersectionTests, Plane, VertexFormat, GeometryInstance, arrayRemoveDuplicates, BoundingRectangle, EllipsoidTangentPlane, OrientedBoundingBox, CoplanarPolygonGeometryLibrary, ArcType, EllipsoidRhumbLine, PolygonPipeline, PolygonGeometryLibrary) { 'use strict';
  24. var scratchPosition = new Cartesian2.Cartesian3();
  25. var scratchBR = new BoundingRectangle.BoundingRectangle();
  26. var stScratch = new Cartesian2.Cartesian2();
  27. var textureCoordinatesOrigin = new Cartesian2.Cartesian2();
  28. var scratchNormal = new Cartesian2.Cartesian3();
  29. var scratchTangent = new Cartesian2.Cartesian3();
  30. var scratchBitangent = new Cartesian2.Cartesian3();
  31. var centerScratch = new Cartesian2.Cartesian3();
  32. var axis1Scratch = new Cartesian2.Cartesian3();
  33. var axis2Scratch = new Cartesian2.Cartesian3();
  34. var quaternionScratch = new Transforms.Quaternion();
  35. var textureMatrixScratch = new Transforms.Matrix3();
  36. var tangentRotationScratch = new Transforms.Matrix3();
  37. var surfaceNormalScratch = new Cartesian2.Cartesian3();
  38. function createGeometryFromPolygon(
  39. polygon,
  40. vertexFormat,
  41. boundingRectangle,
  42. stRotation,
  43. projectPointTo2D,
  44. normal,
  45. tangent,
  46. bitangent
  47. ) {
  48. var positions = polygon.positions;
  49. var indices = PolygonPipeline.PolygonPipeline.triangulate(polygon.positions2D, polygon.holes);
  50. /* If polygon is completely unrenderable, just use the first three vertices */
  51. if (indices.length < 3) {
  52. indices = [0, 1, 2];
  53. }
  54. var newIndices = IndexDatatype.IndexDatatype.createTypedArray(
  55. positions.length,
  56. indices.length
  57. );
  58. newIndices.set(indices);
  59. var textureMatrix = textureMatrixScratch;
  60. if (stRotation !== 0.0) {
  61. var rotation = Transforms.Quaternion.fromAxisAngle(
  62. normal,
  63. stRotation,
  64. quaternionScratch
  65. );
  66. textureMatrix = Transforms.Matrix3.fromQuaternion(rotation, textureMatrix);
  67. if (vertexFormat.tangent || vertexFormat.bitangent) {
  68. rotation = Transforms.Quaternion.fromAxisAngle(
  69. normal,
  70. -stRotation,
  71. quaternionScratch
  72. );
  73. var tangentRotation = Transforms.Matrix3.fromQuaternion(
  74. rotation,
  75. tangentRotationScratch
  76. );
  77. tangent = Cartesian2.Cartesian3.normalize(
  78. Transforms.Matrix3.multiplyByVector(tangentRotation, tangent, tangent),
  79. tangent
  80. );
  81. if (vertexFormat.bitangent) {
  82. bitangent = Cartesian2.Cartesian3.normalize(
  83. Cartesian2.Cartesian3.cross(normal, tangent, bitangent),
  84. bitangent
  85. );
  86. }
  87. }
  88. } else {
  89. textureMatrix = Transforms.Matrix3.clone(Transforms.Matrix3.IDENTITY, textureMatrix);
  90. }
  91. var stOrigin = textureCoordinatesOrigin;
  92. if (vertexFormat.st) {
  93. stOrigin.x = boundingRectangle.x;
  94. stOrigin.y = boundingRectangle.y;
  95. }
  96. var length = positions.length;
  97. var size = length * 3;
  98. var flatPositions = new Float64Array(size);
  99. var normals = vertexFormat.normal ? new Float32Array(size) : undefined;
  100. var tangents = vertexFormat.tangent ? new Float32Array(size) : undefined;
  101. var bitangents = vertexFormat.bitangent ? new Float32Array(size) : undefined;
  102. var textureCoordinates = vertexFormat.st
  103. ? new Float32Array(length * 2)
  104. : undefined;
  105. var positionIndex = 0;
  106. var normalIndex = 0;
  107. var bitangentIndex = 0;
  108. var tangentIndex = 0;
  109. var stIndex = 0;
  110. for (var i = 0; i < length; i++) {
  111. var position = positions[i];
  112. flatPositions[positionIndex++] = position.x;
  113. flatPositions[positionIndex++] = position.y;
  114. flatPositions[positionIndex++] = position.z;
  115. if (vertexFormat.st) {
  116. var p = Transforms.Matrix3.multiplyByVector(
  117. textureMatrix,
  118. position,
  119. scratchPosition
  120. );
  121. var st = projectPointTo2D(p, stScratch);
  122. Cartesian2.Cartesian2.subtract(st, stOrigin, st);
  123. var stx = _Math.CesiumMath.clamp(st.x / boundingRectangle.width, 0, 1);
  124. var sty = _Math.CesiumMath.clamp(st.y / boundingRectangle.height, 0, 1);
  125. textureCoordinates[stIndex++] = stx;
  126. textureCoordinates[stIndex++] = sty;
  127. }
  128. if (vertexFormat.normal) {
  129. normals[normalIndex++] = normal.x;
  130. normals[normalIndex++] = normal.y;
  131. normals[normalIndex++] = normal.z;
  132. }
  133. if (vertexFormat.tangent) {
  134. tangents[tangentIndex++] = tangent.x;
  135. tangents[tangentIndex++] = tangent.y;
  136. tangents[tangentIndex++] = tangent.z;
  137. }
  138. if (vertexFormat.bitangent) {
  139. bitangents[bitangentIndex++] = bitangent.x;
  140. bitangents[bitangentIndex++] = bitangent.y;
  141. bitangents[bitangentIndex++] = bitangent.z;
  142. }
  143. }
  144. var attributes = new GeometryAttributes.GeometryAttributes();
  145. if (vertexFormat.position) {
  146. attributes.position = new GeometryAttribute.GeometryAttribute({
  147. componentDatatype: ComponentDatatype.ComponentDatatype.DOUBLE,
  148. componentsPerAttribute: 3,
  149. values: flatPositions,
  150. });
  151. }
  152. if (vertexFormat.normal) {
  153. attributes.normal = new GeometryAttribute.GeometryAttribute({
  154. componentDatatype: ComponentDatatype.ComponentDatatype.FLOAT,
  155. componentsPerAttribute: 3,
  156. values: normals,
  157. });
  158. }
  159. if (vertexFormat.tangent) {
  160. attributes.tangent = new GeometryAttribute.GeometryAttribute({
  161. componentDatatype: ComponentDatatype.ComponentDatatype.FLOAT,
  162. componentsPerAttribute: 3,
  163. values: tangents,
  164. });
  165. }
  166. if (vertexFormat.bitangent) {
  167. attributes.bitangent = new GeometryAttribute.GeometryAttribute({
  168. componentDatatype: ComponentDatatype.ComponentDatatype.FLOAT,
  169. componentsPerAttribute: 3,
  170. values: bitangents,
  171. });
  172. }
  173. if (vertexFormat.st) {
  174. attributes.st = new GeometryAttribute.GeometryAttribute({
  175. componentDatatype: ComponentDatatype.ComponentDatatype.FLOAT,
  176. componentsPerAttribute: 2,
  177. values: textureCoordinates,
  178. });
  179. }
  180. return new GeometryAttribute.Geometry({
  181. attributes: attributes,
  182. indices: newIndices,
  183. primitiveType: GeometryAttribute.PrimitiveType.TRIANGLES,
  184. });
  185. }
  186. /**
  187. * A description of a polygon composed of arbitrary coplanar positions.
  188. *
  189. * @alias CoplanarPolygonGeometry
  190. * @constructor
  191. *
  192. * @param {Object} options Object with the following properties:
  193. * @param {PolygonHierarchy} options.polygonHierarchy A polygon hierarchy that can include holes.
  194. * @param {Number} [options.stRotation=0.0] The rotation of the texture coordinates, in radians. A positive rotation is counter-clockwise.
  195. * @param {VertexFormat} [options.vertexFormat=VertexFormat.DEFAULT] The vertex attributes to be computed.
  196. * @param {Ellipsoid} [options.ellipsoid=Ellipsoid.WGS84] The ellipsoid to be used as a reference.
  197. *
  198. * @example
  199. * var polygon = new Cesium.CoplanarPolygonGeometry({
  200. * positions : Cesium.Cartesian3.fromDegreesArrayHeights([
  201. * -90.0, 30.0, 0.0,
  202. * -90.0, 30.0, 1000.0,
  203. * -80.0, 30.0, 1000.0,
  204. * -80.0, 30.0, 0.0
  205. * ])
  206. * });
  207. * var geometry = Cesium.CoplanarPolygonGeometry.createGeometry(polygon);
  208. *
  209. * @see CoplanarPolygonGeometry.createGeometry
  210. */
  211. function CoplanarPolygonGeometry(options) {
  212. options = when.defaultValue(options, when.defaultValue.EMPTY_OBJECT);
  213. var polygonHierarchy = options.polygonHierarchy;
  214. //>>includeStart('debug', pragmas.debug);
  215. Check.Check.defined("options.polygonHierarchy", polygonHierarchy);
  216. //>>includeEnd('debug');
  217. var vertexFormat = when.defaultValue(options.vertexFormat, VertexFormat.VertexFormat.DEFAULT);
  218. this._vertexFormat = VertexFormat.VertexFormat.clone(vertexFormat);
  219. this._polygonHierarchy = polygonHierarchy;
  220. this._stRotation = when.defaultValue(options.stRotation, 0.0);
  221. this._ellipsoid = Cartesian2.Ellipsoid.clone(
  222. when.defaultValue(options.ellipsoid, Cartesian2.Ellipsoid.WGS84)
  223. );
  224. this._workerName = "createCoplanarPolygonGeometry";
  225. /**
  226. * The number of elements used to pack the object into an array.
  227. * @type {Number}
  228. */
  229. this.packedLength =
  230. PolygonGeometryLibrary.PolygonGeometryLibrary.computeHierarchyPackedLength(polygonHierarchy) +
  231. VertexFormat.VertexFormat.packedLength +
  232. Cartesian2.Ellipsoid.packedLength +
  233. 2;
  234. }
  235. /**
  236. * A description of a coplanar polygon from an array of positions.
  237. *
  238. * @param {Object} options Object with the following properties:
  239. * @param {Cartesian3[]} options.positions An array of positions that defined the corner points of the polygon.
  240. * @param {VertexFormat} [options.vertexFormat=VertexFormat.DEFAULT] The vertex attributes to be computed.
  241. * @param {Number} [options.stRotation=0.0] The rotation of the texture coordinates, in radians. A positive rotation is counter-clockwise.
  242. * @param {Ellipsoid} [options.ellipsoid=Ellipsoid.WGS84] The ellipsoid to be used as a reference.
  243. * @returns {CoplanarPolygonGeometry}
  244. *
  245. * @example
  246. * // create a polygon from points
  247. * var polygon = Cesium.CoplanarPolygonGeometry.fromPositions({
  248. * positions : Cesium.Cartesian3.fromDegreesArray([
  249. * -72.0, 40.0,
  250. * -70.0, 35.0,
  251. * -75.0, 30.0,
  252. * -70.0, 30.0,
  253. * -68.0, 40.0
  254. * ])
  255. * });
  256. * var geometry = Cesium.PolygonGeometry.createGeometry(polygon);
  257. *
  258. * @see PolygonGeometry#createGeometry
  259. */
  260. CoplanarPolygonGeometry.fromPositions = function (options) {
  261. options = when.defaultValue(options, when.defaultValue.EMPTY_OBJECT);
  262. //>>includeStart('debug', pragmas.debug);
  263. Check.Check.defined("options.positions", options.positions);
  264. //>>includeEnd('debug');
  265. var newOptions = {
  266. polygonHierarchy: {
  267. positions: options.positions,
  268. },
  269. vertexFormat: options.vertexFormat,
  270. stRotation: options.stRotation,
  271. ellipsoid: options.ellipsoid,
  272. };
  273. return new CoplanarPolygonGeometry(newOptions);
  274. };
  275. /**
  276. * Stores the provided instance into the provided array.
  277. *
  278. * @param {CoplanarPolygonGeometry} value The value to pack.
  279. * @param {Number[]} array The array to pack into.
  280. * @param {Number} [startingIndex=0] The index into the array at which to start packing the elements.
  281. *
  282. * @returns {Number[]} The array that was packed into
  283. */
  284. CoplanarPolygonGeometry.pack = function (value, array, startingIndex) {
  285. //>>includeStart('debug', pragmas.debug);
  286. Check.Check.typeOf.object("value", value);
  287. Check.Check.defined("array", array);
  288. //>>includeEnd('debug');
  289. startingIndex = when.defaultValue(startingIndex, 0);
  290. startingIndex = PolygonGeometryLibrary.PolygonGeometryLibrary.packPolygonHierarchy(
  291. value._polygonHierarchy,
  292. array,
  293. startingIndex
  294. );
  295. Cartesian2.Ellipsoid.pack(value._ellipsoid, array, startingIndex);
  296. startingIndex += Cartesian2.Ellipsoid.packedLength;
  297. VertexFormat.VertexFormat.pack(value._vertexFormat, array, startingIndex);
  298. startingIndex += VertexFormat.VertexFormat.packedLength;
  299. array[startingIndex++] = value._stRotation;
  300. array[startingIndex] = value.packedLength;
  301. return array;
  302. };
  303. var scratchEllipsoid = Cartesian2.Ellipsoid.clone(Cartesian2.Ellipsoid.UNIT_SPHERE);
  304. var scratchVertexFormat = new VertexFormat.VertexFormat();
  305. var scratchOptions = {
  306. polygonHierarchy: {},
  307. };
  308. /**
  309. * Retrieves an instance from a packed array.
  310. *
  311. * @param {Number[]} array The packed array.
  312. * @param {Number} [startingIndex=0] The starting index of the element to be unpacked.
  313. * @param {CoplanarPolygonGeometry} [result] The object into which to store the result.
  314. * @returns {CoplanarPolygonGeometry} The modified result parameter or a new CoplanarPolygonGeometry instance if one was not provided.
  315. */
  316. CoplanarPolygonGeometry.unpack = function (array, startingIndex, result) {
  317. //>>includeStart('debug', pragmas.debug);
  318. Check.Check.defined("array", array);
  319. //>>includeEnd('debug');
  320. startingIndex = when.defaultValue(startingIndex, 0);
  321. var polygonHierarchy = PolygonGeometryLibrary.PolygonGeometryLibrary.unpackPolygonHierarchy(
  322. array,
  323. startingIndex
  324. );
  325. startingIndex = polygonHierarchy.startingIndex;
  326. delete polygonHierarchy.startingIndex;
  327. var ellipsoid = Cartesian2.Ellipsoid.unpack(array, startingIndex, scratchEllipsoid);
  328. startingIndex += Cartesian2.Ellipsoid.packedLength;
  329. var vertexFormat = VertexFormat.VertexFormat.unpack(
  330. array,
  331. startingIndex,
  332. scratchVertexFormat
  333. );
  334. startingIndex += VertexFormat.VertexFormat.packedLength;
  335. var stRotation = array[startingIndex++];
  336. var packedLength = array[startingIndex];
  337. if (!when.defined(result)) {
  338. result = new CoplanarPolygonGeometry(scratchOptions);
  339. }
  340. result._polygonHierarchy = polygonHierarchy;
  341. result._ellipsoid = Cartesian2.Ellipsoid.clone(ellipsoid, result._ellipsoid);
  342. result._vertexFormat = VertexFormat.VertexFormat.clone(vertexFormat, result._vertexFormat);
  343. result._stRotation = stRotation;
  344. result.packedLength = packedLength;
  345. return result;
  346. };
  347. /**
  348. * Computes the geometric representation of an arbitrary coplanar polygon, including its vertices, indices, and a bounding sphere.
  349. *
  350. * @param {CoplanarPolygonGeometry} polygonGeometry A description of the polygon.
  351. * @returns {Geometry|undefined} The computed vertices and indices.
  352. */
  353. CoplanarPolygonGeometry.createGeometry = function (polygonGeometry) {
  354. var vertexFormat = polygonGeometry._vertexFormat;
  355. var polygonHierarchy = polygonGeometry._polygonHierarchy;
  356. var stRotation = polygonGeometry._stRotation;
  357. var outerPositions = polygonHierarchy.positions;
  358. outerPositions = arrayRemoveDuplicates.arrayRemoveDuplicates(
  359. outerPositions,
  360. Cartesian2.Cartesian3.equalsEpsilon,
  361. true
  362. );
  363. if (outerPositions.length < 3) {
  364. return;
  365. }
  366. var normal = scratchNormal;
  367. var tangent = scratchTangent;
  368. var bitangent = scratchBitangent;
  369. var axis1 = axis1Scratch;
  370. var axis2 = axis2Scratch;
  371. var validGeometry = CoplanarPolygonGeometryLibrary.CoplanarPolygonGeometryLibrary.computeProjectTo2DArguments(
  372. outerPositions,
  373. centerScratch,
  374. axis1,
  375. axis2
  376. );
  377. if (!validGeometry) {
  378. return undefined;
  379. }
  380. normal = Cartesian2.Cartesian3.cross(axis1, axis2, normal);
  381. normal = Cartesian2.Cartesian3.normalize(normal, normal);
  382. if (
  383. !Cartesian2.Cartesian3.equalsEpsilon(
  384. centerScratch,
  385. Cartesian2.Cartesian3.ZERO,
  386. _Math.CesiumMath.EPSILON6
  387. )
  388. ) {
  389. var surfaceNormal = polygonGeometry._ellipsoid.geodeticSurfaceNormal(
  390. centerScratch,
  391. surfaceNormalScratch
  392. );
  393. if (Cartesian2.Cartesian3.dot(normal, surfaceNormal) < 0) {
  394. normal = Cartesian2.Cartesian3.negate(normal, normal);
  395. axis1 = Cartesian2.Cartesian3.negate(axis1, axis1);
  396. }
  397. }
  398. var projectPoints = CoplanarPolygonGeometryLibrary.CoplanarPolygonGeometryLibrary.createProjectPointsTo2DFunction(
  399. centerScratch,
  400. axis1,
  401. axis2
  402. );
  403. var projectPoint = CoplanarPolygonGeometryLibrary.CoplanarPolygonGeometryLibrary.createProjectPointTo2DFunction(
  404. centerScratch,
  405. axis1,
  406. axis2
  407. );
  408. if (vertexFormat.tangent) {
  409. tangent = Cartesian2.Cartesian3.clone(axis1, tangent);
  410. }
  411. if (vertexFormat.bitangent) {
  412. bitangent = Cartesian2.Cartesian3.clone(axis2, bitangent);
  413. }
  414. var results = PolygonGeometryLibrary.PolygonGeometryLibrary.polygonsFromHierarchy(
  415. polygonHierarchy,
  416. projectPoints,
  417. false
  418. );
  419. var hierarchy = results.hierarchy;
  420. var polygons = results.polygons;
  421. if (hierarchy.length === 0) {
  422. return;
  423. }
  424. outerPositions = hierarchy[0].outerRing;
  425. var boundingSphere = Transforms.BoundingSphere.fromPoints(outerPositions);
  426. var boundingRectangle = PolygonGeometryLibrary.PolygonGeometryLibrary.computeBoundingRectangle(
  427. normal,
  428. projectPoint,
  429. outerPositions,
  430. stRotation,
  431. scratchBR
  432. );
  433. var geometries = [];
  434. for (var i = 0; i < polygons.length; i++) {
  435. var geometryInstance = new GeometryInstance.GeometryInstance({
  436. geometry: createGeometryFromPolygon(
  437. polygons[i],
  438. vertexFormat,
  439. boundingRectangle,
  440. stRotation,
  441. projectPoint,
  442. normal,
  443. tangent,
  444. bitangent
  445. ),
  446. });
  447. geometries.push(geometryInstance);
  448. }
  449. var geometry = GeometryPipeline.GeometryPipeline.combineInstances(geometries)[0];
  450. geometry.attributes.position.values = new Float64Array(
  451. geometry.attributes.position.values
  452. );
  453. geometry.indices = IndexDatatype.IndexDatatype.createTypedArray(
  454. geometry.attributes.position.values.length / 3,
  455. geometry.indices
  456. );
  457. var attributes = geometry.attributes;
  458. if (!vertexFormat.position) {
  459. delete attributes.position;
  460. }
  461. return new GeometryAttribute.Geometry({
  462. attributes: attributes,
  463. indices: geometry.indices,
  464. primitiveType: geometry.primitiveType,
  465. boundingSphere: boundingSphere,
  466. });
  467. };
  468. function createCoplanarPolygonGeometry(polygonGeometry, offset) {
  469. if (when.defined(offset)) {
  470. polygonGeometry = CoplanarPolygonGeometry.unpack(polygonGeometry, offset);
  471. }
  472. return CoplanarPolygonGeometry.createGeometry(polygonGeometry);
  473. }
  474. return createCoplanarPolygonGeometry;
  475. });
  476. //# sourceMappingURL=createCoplanarPolygonGeometry.js.map