createRectangleOutlineGeometry.js 20 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560
  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', './IndexDatatype-25023891', './GeometryOffsetAttribute-d746452d', './EllipsoidRhumbLine-5f1492e5', './PolygonPipeline-9f9b7763', './RectangleGeometryLibrary-3ec48410'], function (when, Check, _Math, Cartesian2, Transforms, RuntimeError, WebGLConstants, ComponentDatatype, GeometryAttribute, GeometryAttributes, IndexDatatype, GeometryOffsetAttribute, EllipsoidRhumbLine, PolygonPipeline, RectangleGeometryLibrary) { 'use strict';
  24. var bottomBoundingSphere = new Transforms.BoundingSphere();
  25. var topBoundingSphere = new Transforms.BoundingSphere();
  26. var positionScratch = new Cartesian2.Cartesian3();
  27. var rectangleScratch = new Cartesian2.Rectangle();
  28. function constructRectangle(geometry, computedOptions) {
  29. var ellipsoid = geometry._ellipsoid;
  30. var height = computedOptions.height;
  31. var width = computedOptions.width;
  32. var northCap = computedOptions.northCap;
  33. var southCap = computedOptions.southCap;
  34. var rowHeight = height;
  35. var widthMultiplier = 2;
  36. var size = 0;
  37. var corners = 4;
  38. if (northCap) {
  39. widthMultiplier -= 1;
  40. rowHeight -= 1;
  41. size += 1;
  42. corners -= 2;
  43. }
  44. if (southCap) {
  45. widthMultiplier -= 1;
  46. rowHeight -= 1;
  47. size += 1;
  48. corners -= 2;
  49. }
  50. size += widthMultiplier * width + 2 * rowHeight - corners;
  51. var positions = new Float64Array(size * 3);
  52. var posIndex = 0;
  53. var row = 0;
  54. var col;
  55. var position = positionScratch;
  56. if (northCap) {
  57. RectangleGeometryLibrary.RectangleGeometryLibrary.computePosition(
  58. computedOptions,
  59. ellipsoid,
  60. false,
  61. row,
  62. 0,
  63. position
  64. );
  65. positions[posIndex++] = position.x;
  66. positions[posIndex++] = position.y;
  67. positions[posIndex++] = position.z;
  68. } else {
  69. for (col = 0; col < width; col++) {
  70. RectangleGeometryLibrary.RectangleGeometryLibrary.computePosition(
  71. computedOptions,
  72. ellipsoid,
  73. false,
  74. row,
  75. col,
  76. position
  77. );
  78. positions[posIndex++] = position.x;
  79. positions[posIndex++] = position.y;
  80. positions[posIndex++] = position.z;
  81. }
  82. }
  83. col = width - 1;
  84. for (row = 1; row < height; row++) {
  85. RectangleGeometryLibrary.RectangleGeometryLibrary.computePosition(
  86. computedOptions,
  87. ellipsoid,
  88. false,
  89. row,
  90. col,
  91. position
  92. );
  93. positions[posIndex++] = position.x;
  94. positions[posIndex++] = position.y;
  95. positions[posIndex++] = position.z;
  96. }
  97. row = height - 1;
  98. if (!southCap) {
  99. // if southCap is true, we dont need to add any more points because the south pole point was added by the iteration above
  100. for (col = width - 2; col >= 0; col--) {
  101. RectangleGeometryLibrary.RectangleGeometryLibrary.computePosition(
  102. computedOptions,
  103. ellipsoid,
  104. false,
  105. row,
  106. col,
  107. position
  108. );
  109. positions[posIndex++] = position.x;
  110. positions[posIndex++] = position.y;
  111. positions[posIndex++] = position.z;
  112. }
  113. }
  114. col = 0;
  115. for (row = height - 2; row > 0; row--) {
  116. RectangleGeometryLibrary.RectangleGeometryLibrary.computePosition(
  117. computedOptions,
  118. ellipsoid,
  119. false,
  120. row,
  121. col,
  122. position
  123. );
  124. positions[posIndex++] = position.x;
  125. positions[posIndex++] = position.y;
  126. positions[posIndex++] = position.z;
  127. }
  128. var indicesSize = (positions.length / 3) * 2;
  129. var indices = IndexDatatype.IndexDatatype.createTypedArray(
  130. positions.length / 3,
  131. indicesSize
  132. );
  133. var index = 0;
  134. for (var i = 0; i < positions.length / 3 - 1; i++) {
  135. indices[index++] = i;
  136. indices[index++] = i + 1;
  137. }
  138. indices[index++] = positions.length / 3 - 1;
  139. indices[index++] = 0;
  140. var geo = new GeometryAttribute.Geometry({
  141. attributes: new GeometryAttributes.GeometryAttributes(),
  142. primitiveType: GeometryAttribute.PrimitiveType.LINES,
  143. });
  144. geo.attributes.position = new GeometryAttribute.GeometryAttribute({
  145. componentDatatype: ComponentDatatype.ComponentDatatype.DOUBLE,
  146. componentsPerAttribute: 3,
  147. values: positions,
  148. });
  149. geo.indices = indices;
  150. return geo;
  151. }
  152. function constructExtrudedRectangle(rectangleGeometry, computedOptions) {
  153. var surfaceHeight = rectangleGeometry._surfaceHeight;
  154. var extrudedHeight = rectangleGeometry._extrudedHeight;
  155. var ellipsoid = rectangleGeometry._ellipsoid;
  156. var minHeight = extrudedHeight;
  157. var maxHeight = surfaceHeight;
  158. var geo = constructRectangle(rectangleGeometry, computedOptions);
  159. var height = computedOptions.height;
  160. var width = computedOptions.width;
  161. var topPositions = PolygonPipeline.PolygonPipeline.scaleToGeodeticHeight(
  162. geo.attributes.position.values,
  163. maxHeight,
  164. ellipsoid,
  165. false
  166. );
  167. var length = topPositions.length;
  168. var positions = new Float64Array(length * 2);
  169. positions.set(topPositions);
  170. var bottomPositions = PolygonPipeline.PolygonPipeline.scaleToGeodeticHeight(
  171. geo.attributes.position.values,
  172. minHeight,
  173. ellipsoid
  174. );
  175. positions.set(bottomPositions, length);
  176. geo.attributes.position.values = positions;
  177. var northCap = computedOptions.northCap;
  178. var southCap = computedOptions.southCap;
  179. var corners = 4;
  180. if (northCap) {
  181. corners -= 1;
  182. }
  183. if (southCap) {
  184. corners -= 1;
  185. }
  186. var indicesSize = (positions.length / 3 + corners) * 2;
  187. var indices = IndexDatatype.IndexDatatype.createTypedArray(
  188. positions.length / 3,
  189. indicesSize
  190. );
  191. length = positions.length / 6;
  192. var index = 0;
  193. for (var i = 0; i < length - 1; i++) {
  194. indices[index++] = i;
  195. indices[index++] = i + 1;
  196. indices[index++] = i + length;
  197. indices[index++] = i + length + 1;
  198. }
  199. indices[index++] = length - 1;
  200. indices[index++] = 0;
  201. indices[index++] = length + length - 1;
  202. indices[index++] = length;
  203. indices[index++] = 0;
  204. indices[index++] = length;
  205. var bottomCorner;
  206. if (northCap) {
  207. bottomCorner = height - 1;
  208. } else {
  209. var topRightCorner = width - 1;
  210. indices[index++] = topRightCorner;
  211. indices[index++] = topRightCorner + length;
  212. bottomCorner = width + height - 2;
  213. }
  214. indices[index++] = bottomCorner;
  215. indices[index++] = bottomCorner + length;
  216. if (!southCap) {
  217. var bottomLeftCorner = width + bottomCorner - 1;
  218. indices[index++] = bottomLeftCorner;
  219. indices[index] = bottomLeftCorner + length;
  220. }
  221. geo.indices = indices;
  222. return geo;
  223. }
  224. /**
  225. * A description of the outline of a a cartographic rectangle on an ellipsoid centered at the origin.
  226. *
  227. * @alias RectangleOutlineGeometry
  228. * @constructor
  229. *
  230. * @param {Object} options Object with the following properties:
  231. * @param {Rectangle} options.rectangle A cartographic rectangle with north, south, east and west properties in radians.
  232. * @param {Ellipsoid} [options.ellipsoid=Ellipsoid.WGS84] The ellipsoid on which the rectangle lies.
  233. * @param {Number} [options.granularity=CesiumMath.RADIANS_PER_DEGREE] The distance, in radians, between each latitude and longitude. Determines the number of positions in the buffer.
  234. * @param {Number} [options.height=0.0] The distance in meters between the rectangle and the ellipsoid surface.
  235. * @param {Number} [options.rotation=0.0] The rotation of the rectangle, in radians. A positive rotation is counter-clockwise.
  236. * @param {Number} [options.extrudedHeight] The distance in meters between the rectangle's extruded face and the ellipsoid surface.
  237. *
  238. * @exception {DeveloperError} <code>options.rectangle.north</code> must be in the interval [<code>-Pi/2</code>, <code>Pi/2</code>].
  239. * @exception {DeveloperError} <code>options.rectangle.south</code> must be in the interval [<code>-Pi/2</code>, <code>Pi/2</code>].
  240. * @exception {DeveloperError} <code>options.rectangle.east</code> must be in the interval [<code>-Pi</code>, <code>Pi</code>].
  241. * @exception {DeveloperError} <code>options.rectangle.west</code> must be in the interval [<code>-Pi</code>, <code>Pi</code>].
  242. * @exception {DeveloperError} <code>options.rectangle.north</code> must be greater than <code>rectangle.south</code>.
  243. *
  244. * @see RectangleOutlineGeometry#createGeometry
  245. *
  246. * @example
  247. * var rectangle = new Cesium.RectangleOutlineGeometry({
  248. * ellipsoid : Cesium.Ellipsoid.WGS84,
  249. * rectangle : Cesium.Rectangle.fromDegrees(-80.0, 39.0, -74.0, 42.0),
  250. * height : 10000.0
  251. * });
  252. * var geometry = Cesium.RectangleOutlineGeometry.createGeometry(rectangle);
  253. */
  254. function RectangleOutlineGeometry(options) {
  255. options = when.defaultValue(options, when.defaultValue.EMPTY_OBJECT);
  256. var rectangle = options.rectangle;
  257. var granularity = when.defaultValue(
  258. options.granularity,
  259. _Math.CesiumMath.RADIANS_PER_DEGREE
  260. );
  261. var ellipsoid = when.defaultValue(options.ellipsoid, Cartesian2.Ellipsoid.WGS84);
  262. var rotation = when.defaultValue(options.rotation, 0.0);
  263. //>>includeStart('debug', pragmas.debug);
  264. if (!when.defined(rectangle)) {
  265. throw new Check.DeveloperError("rectangle is required.");
  266. }
  267. Cartesian2.Rectangle.validate(rectangle);
  268. if (rectangle.north < rectangle.south) {
  269. throw new Check.DeveloperError(
  270. "options.rectangle.north must be greater than options.rectangle.south"
  271. );
  272. }
  273. //>>includeEnd('debug');
  274. var height = when.defaultValue(options.height, 0.0);
  275. var extrudedHeight = when.defaultValue(options.extrudedHeight, height);
  276. this._rectangle = Cartesian2.Rectangle.clone(rectangle);
  277. this._granularity = granularity;
  278. this._ellipsoid = ellipsoid;
  279. this._surfaceHeight = Math.max(height, extrudedHeight);
  280. this._rotation = rotation;
  281. this._extrudedHeight = Math.min(height, extrudedHeight);
  282. this._offsetAttribute = options.offsetAttribute;
  283. this._workerName = "createRectangleOutlineGeometry";
  284. }
  285. /**
  286. * The number of elements used to pack the object into an array.
  287. * @type {Number}
  288. */
  289. RectangleOutlineGeometry.packedLength =
  290. Cartesian2.Rectangle.packedLength + Cartesian2.Ellipsoid.packedLength + 5;
  291. /**
  292. * Stores the provided instance into the provided array.
  293. *
  294. * @param {RectangleOutlineGeometry} value The value to pack.
  295. * @param {Number[]} array The array to pack into.
  296. * @param {Number} [startingIndex=0] The index into the array at which to start packing the elements.
  297. *
  298. * @returns {Number[]} The array that was packed into
  299. */
  300. RectangleOutlineGeometry.pack = function (value, array, startingIndex) {
  301. //>>includeStart('debug', pragmas.debug);
  302. if (!when.defined(value)) {
  303. throw new Check.DeveloperError("value is required");
  304. }
  305. if (!when.defined(array)) {
  306. throw new Check.DeveloperError("array is required");
  307. }
  308. //>>includeEnd('debug');
  309. startingIndex = when.defaultValue(startingIndex, 0);
  310. Cartesian2.Rectangle.pack(value._rectangle, array, startingIndex);
  311. startingIndex += Cartesian2.Rectangle.packedLength;
  312. Cartesian2.Ellipsoid.pack(value._ellipsoid, array, startingIndex);
  313. startingIndex += Cartesian2.Ellipsoid.packedLength;
  314. array[startingIndex++] = value._granularity;
  315. array[startingIndex++] = value._surfaceHeight;
  316. array[startingIndex++] = value._rotation;
  317. array[startingIndex++] = value._extrudedHeight;
  318. array[startingIndex] = when.defaultValue(value._offsetAttribute, -1);
  319. return array;
  320. };
  321. var scratchRectangle = new Cartesian2.Rectangle();
  322. var scratchEllipsoid = Cartesian2.Ellipsoid.clone(Cartesian2.Ellipsoid.UNIT_SPHERE);
  323. var scratchOptions = {
  324. rectangle: scratchRectangle,
  325. ellipsoid: scratchEllipsoid,
  326. granularity: undefined,
  327. height: undefined,
  328. rotation: undefined,
  329. extrudedHeight: undefined,
  330. offsetAttribute: undefined,
  331. };
  332. /**
  333. * Retrieves an instance from a packed array.
  334. *
  335. * @param {Number[]} array The packed array.
  336. * @param {Number} [startingIndex=0] The starting index of the element to be unpacked.
  337. * @param {RectangleOutlineGeometry} [result] The object into which to store the result.
  338. * @returns {RectangleOutlineGeometry} The modified result parameter or a new Quaternion instance if one was not provided.
  339. */
  340. RectangleOutlineGeometry.unpack = function (array, startingIndex, result) {
  341. //>>includeStart('debug', pragmas.debug);
  342. if (!when.defined(array)) {
  343. throw new Check.DeveloperError("array is required");
  344. }
  345. //>>includeEnd('debug');
  346. startingIndex = when.defaultValue(startingIndex, 0);
  347. var rectangle = Cartesian2.Rectangle.unpack(array, startingIndex, scratchRectangle);
  348. startingIndex += Cartesian2.Rectangle.packedLength;
  349. var ellipsoid = Cartesian2.Ellipsoid.unpack(array, startingIndex, scratchEllipsoid);
  350. startingIndex += Cartesian2.Ellipsoid.packedLength;
  351. var granularity = array[startingIndex++];
  352. var height = array[startingIndex++];
  353. var rotation = array[startingIndex++];
  354. var extrudedHeight = array[startingIndex++];
  355. var offsetAttribute = array[startingIndex];
  356. if (!when.defined(result)) {
  357. scratchOptions.granularity = granularity;
  358. scratchOptions.height = height;
  359. scratchOptions.rotation = rotation;
  360. scratchOptions.extrudedHeight = extrudedHeight;
  361. scratchOptions.offsetAttribute =
  362. offsetAttribute === -1 ? undefined : offsetAttribute;
  363. return new RectangleOutlineGeometry(scratchOptions);
  364. }
  365. result._rectangle = Cartesian2.Rectangle.clone(rectangle, result._rectangle);
  366. result._ellipsoid = Cartesian2.Ellipsoid.clone(ellipsoid, result._ellipsoid);
  367. result._surfaceHeight = height;
  368. result._rotation = rotation;
  369. result._extrudedHeight = extrudedHeight;
  370. result._offsetAttribute =
  371. offsetAttribute === -1 ? undefined : offsetAttribute;
  372. return result;
  373. };
  374. var nwScratch = new Cartesian2.Cartographic();
  375. /**
  376. * Computes the geometric representation of an outline of a rectangle, including its vertices, indices, and a bounding sphere.
  377. *
  378. * @param {RectangleOutlineGeometry} rectangleGeometry A description of the rectangle outline.
  379. * @returns {Geometry|undefined} The computed vertices and indices.
  380. *
  381. * @exception {DeveloperError} Rotated rectangle is invalid.
  382. */
  383. RectangleOutlineGeometry.createGeometry = function (rectangleGeometry) {
  384. var rectangle = rectangleGeometry._rectangle;
  385. var ellipsoid = rectangleGeometry._ellipsoid;
  386. var computedOptions = RectangleGeometryLibrary.RectangleGeometryLibrary.computeOptions(
  387. rectangle,
  388. rectangleGeometry._granularity,
  389. rectangleGeometry._rotation,
  390. 0,
  391. rectangleScratch,
  392. nwScratch
  393. );
  394. var geometry;
  395. var boundingSphere;
  396. if (
  397. _Math.CesiumMath.equalsEpsilon(
  398. rectangle.north,
  399. rectangle.south,
  400. _Math.CesiumMath.EPSILON10
  401. ) ||
  402. _Math.CesiumMath.equalsEpsilon(
  403. rectangle.east,
  404. rectangle.west,
  405. _Math.CesiumMath.EPSILON10
  406. )
  407. ) {
  408. return undefined;
  409. }
  410. var surfaceHeight = rectangleGeometry._surfaceHeight;
  411. var extrudedHeight = rectangleGeometry._extrudedHeight;
  412. var extrude = !_Math.CesiumMath.equalsEpsilon(
  413. surfaceHeight,
  414. extrudedHeight,
  415. 0,
  416. _Math.CesiumMath.EPSILON2
  417. );
  418. var offsetValue;
  419. if (extrude) {
  420. geometry = constructExtrudedRectangle(rectangleGeometry, computedOptions);
  421. if (when.defined(rectangleGeometry._offsetAttribute)) {
  422. var size = geometry.attributes.position.values.length / 3;
  423. var offsetAttribute = new Uint8Array(size);
  424. if (rectangleGeometry._offsetAttribute === GeometryOffsetAttribute.GeometryOffsetAttribute.TOP) {
  425. offsetAttribute = GeometryOffsetAttribute.arrayFill(offsetAttribute, 1, 0, size / 2);
  426. } else {
  427. offsetValue =
  428. rectangleGeometry._offsetAttribute === GeometryOffsetAttribute.GeometryOffsetAttribute.NONE
  429. ? 0
  430. : 1;
  431. offsetAttribute = GeometryOffsetAttribute.arrayFill(offsetAttribute, offsetValue);
  432. }
  433. geometry.attributes.applyOffset = new GeometryAttribute.GeometryAttribute({
  434. componentDatatype: ComponentDatatype.ComponentDatatype.UNSIGNED_BYTE,
  435. componentsPerAttribute: 1,
  436. values: offsetAttribute,
  437. });
  438. }
  439. var topBS = Transforms.BoundingSphere.fromRectangle3D(
  440. rectangle,
  441. ellipsoid,
  442. surfaceHeight,
  443. topBoundingSphere
  444. );
  445. var bottomBS = Transforms.BoundingSphere.fromRectangle3D(
  446. rectangle,
  447. ellipsoid,
  448. extrudedHeight,
  449. bottomBoundingSphere
  450. );
  451. boundingSphere = Transforms.BoundingSphere.union(topBS, bottomBS);
  452. } else {
  453. geometry = constructRectangle(rectangleGeometry, computedOptions);
  454. geometry.attributes.position.values = PolygonPipeline.PolygonPipeline.scaleToGeodeticHeight(
  455. geometry.attributes.position.values,
  456. surfaceHeight,
  457. ellipsoid,
  458. false
  459. );
  460. if (when.defined(rectangleGeometry._offsetAttribute)) {
  461. var length = geometry.attributes.position.values.length;
  462. var applyOffset = new Uint8Array(length / 3);
  463. offsetValue =
  464. rectangleGeometry._offsetAttribute === GeometryOffsetAttribute.GeometryOffsetAttribute.NONE
  465. ? 0
  466. : 1;
  467. GeometryOffsetAttribute.arrayFill(applyOffset, offsetValue);
  468. geometry.attributes.applyOffset = new GeometryAttribute.GeometryAttribute({
  469. componentDatatype: ComponentDatatype.ComponentDatatype.UNSIGNED_BYTE,
  470. componentsPerAttribute: 1,
  471. values: applyOffset,
  472. });
  473. }
  474. boundingSphere = Transforms.BoundingSphere.fromRectangle3D(
  475. rectangle,
  476. ellipsoid,
  477. surfaceHeight
  478. );
  479. }
  480. return new GeometryAttribute.Geometry({
  481. attributes: geometry.attributes,
  482. indices: geometry.indices,
  483. primitiveType: GeometryAttribute.PrimitiveType.LINES,
  484. boundingSphere: boundingSphere,
  485. offsetAttribute: rectangleGeometry._offsetAttribute,
  486. });
  487. };
  488. function createRectangleOutlineGeometry(rectangleGeometry, offset) {
  489. if (when.defined(offset)) {
  490. rectangleGeometry = RectangleOutlineGeometry.unpack(
  491. rectangleGeometry,
  492. offset
  493. );
  494. }
  495. rectangleGeometry._ellipsoid = Cartesian2.Ellipsoid.clone(rectangleGeometry._ellipsoid);
  496. rectangleGeometry._rectangle = Cartesian2.Rectangle.clone(rectangleGeometry._rectangle);
  497. return RectangleOutlineGeometry.createGeometry(rectangleGeometry);
  498. }
  499. return createRectangleOutlineGeometry;
  500. });
  501. //# sourceMappingURL=createRectangleOutlineGeometry.js.map