@macrostrat/cesium-martini 1.4.0 → 1.5.0
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/.idea/cesium-martini.iml +9 -0
- package/.idea/codeStyles/Project.xml +57 -0
- package/.idea/codeStyles/codeStyleConfig.xml +5 -0
- package/.idea/inspectionProfiles/Project_Default.xml +6 -0
- package/.idea/libraries/cache.xml +769 -0
- package/.idea/misc.xml +6 -0
- package/.idea/modules.xml +8 -0
- package/.idea/prettier.xml +8 -0
- package/.idea/vcs.xml +20 -0
- package/.idea/workspace.xml +125 -0
- package/.prettierrc +3 -0
- package/README.md +109 -87
- package/babel.config.js +3 -0
- package/dist/index.cjs +657 -137
- package/dist/index.cjs.map +1 -1
- package/dist/index.js +651 -139
- package/dist/index.js.map +1 -1
- package/package.json +18 -12
- package/rollup.config.js +4 -1
- package/tsconfig.json +3 -6
- package/.babelrc +0 -3
- package/typings/worker-loader.d.ts +0 -10
package/dist/index.cjs
CHANGED
|
@@ -4,6 +4,14 @@ Object.defineProperty(exports, '__esModule', { value: true });
|
|
|
4
4
|
|
|
5
5
|
var cesium = require('cesium');
|
|
6
6
|
|
|
7
|
+
function _arrayLikeToArray(r, a) {
|
|
8
|
+
(null == a || a > r.length) && (a = r.length);
|
|
9
|
+
for (var e = 0, n = Array(a); e < a; e++) n[e] = r[e];
|
|
10
|
+
return n;
|
|
11
|
+
}
|
|
12
|
+
function _arrayWithoutHoles(r) {
|
|
13
|
+
if (Array.isArray(r)) return _arrayLikeToArray(r);
|
|
14
|
+
}
|
|
7
15
|
function _assertThisInitialized(e) {
|
|
8
16
|
if (void 0 === e) throw new ReferenceError("this hasn't been initialised - super() hasn't been called");
|
|
9
17
|
return e;
|
|
@@ -92,6 +100,12 @@ function _isNativeReflectConstruct() {
|
|
|
92
100
|
return !!t;
|
|
93
101
|
})();
|
|
94
102
|
}
|
|
103
|
+
function _iterableToArray(r) {
|
|
104
|
+
if ("undefined" != typeof Symbol && null != r[Symbol.iterator] || null != r["@@iterator"]) return Array.from(r);
|
|
105
|
+
}
|
|
106
|
+
function _nonIterableSpread() {
|
|
107
|
+
throw new TypeError("Invalid attempt to spread non-iterable instance.\nIn order to be iterable, non-array objects must have a [Symbol.iterator]() method.");
|
|
108
|
+
}
|
|
95
109
|
function ownKeys(e, r) {
|
|
96
110
|
var t = Object.keys(e);
|
|
97
111
|
if (Object.getOwnPropertySymbols) {
|
|
@@ -113,6 +127,26 @@ function _objectSpread2(e) {
|
|
|
113
127
|
}
|
|
114
128
|
return e;
|
|
115
129
|
}
|
|
130
|
+
function _objectWithoutProperties(e, t) {
|
|
131
|
+
if (null == e) return {};
|
|
132
|
+
var o,
|
|
133
|
+
r,
|
|
134
|
+
i = _objectWithoutPropertiesLoose(e, t);
|
|
135
|
+
if (Object.getOwnPropertySymbols) {
|
|
136
|
+
var n = Object.getOwnPropertySymbols(e);
|
|
137
|
+
for (r = 0; r < n.length; r++) o = n[r], -1 === t.indexOf(o) && {}.propertyIsEnumerable.call(e, o) && (i[o] = e[o]);
|
|
138
|
+
}
|
|
139
|
+
return i;
|
|
140
|
+
}
|
|
141
|
+
function _objectWithoutPropertiesLoose(r, e) {
|
|
142
|
+
if (null == r) return {};
|
|
143
|
+
var t = {};
|
|
144
|
+
for (var n in r) if ({}.hasOwnProperty.call(r, n)) {
|
|
145
|
+
if (-1 !== e.indexOf(n)) continue;
|
|
146
|
+
t[n] = r[n];
|
|
147
|
+
}
|
|
148
|
+
return t;
|
|
149
|
+
}
|
|
116
150
|
function _possibleConstructorReturn(t, e) {
|
|
117
151
|
if (e && ("object" == typeof e || "function" == typeof e)) return e;
|
|
118
152
|
if (void 0 !== e) throw new TypeError("Derived constructors may only return object or undefined");
|
|
@@ -434,6 +468,9 @@ function _superPropGet(t, o, e, r) {
|
|
|
434
468
|
return p.apply(e, t);
|
|
435
469
|
} : p;
|
|
436
470
|
}
|
|
471
|
+
function _toConsumableArray(r) {
|
|
472
|
+
return _arrayWithoutHoles(r) || _iterableToArray(r) || _unsupportedIterableToArray(r) || _nonIterableSpread();
|
|
473
|
+
}
|
|
437
474
|
function _toPrimitive(t, r) {
|
|
438
475
|
if ("object" != typeof t || !t) return t;
|
|
439
476
|
var e = t[Symbol.toPrimitive];
|
|
@@ -448,57 +485,43 @@ function _toPropertyKey(t) {
|
|
|
448
485
|
var i = _toPrimitive(t, "string");
|
|
449
486
|
return "symbol" == typeof i ? i : i + "";
|
|
450
487
|
}
|
|
488
|
+
function _unsupportedIterableToArray(r, a) {
|
|
489
|
+
if (r) {
|
|
490
|
+
if ("string" == typeof r) return _arrayLikeToArray(r, a);
|
|
491
|
+
var t = {}.toString.call(r).slice(8, -1);
|
|
492
|
+
return "Object" === t && r.constructor && (t = r.constructor.name), "Map" === t || "Set" === t ? Array.from(r) : "Arguments" === t || /^(?:Ui|I)nt(?:8|16|32)(?:Clamped)?Array$/.test(t) ? _arrayLikeToArray(r, a) : void 0;
|
|
493
|
+
}
|
|
494
|
+
}
|
|
451
495
|
|
|
452
|
-
var loadImage = function loadImage(url) {
|
|
453
|
-
return new Promise(function (resolve, reject) {
|
|
454
|
-
var img = new Image();
|
|
455
|
-
img.addEventListener("load", function () {
|
|
456
|
-
return resolve(img);
|
|
457
|
-
});
|
|
458
|
-
img.addEventListener("error", function (err) {
|
|
459
|
-
return reject(err);
|
|
460
|
-
});
|
|
461
|
-
img.crossOrigin = "anonymous";
|
|
462
|
-
img.src = url;
|
|
463
|
-
});
|
|
464
|
-
};
|
|
465
496
|
var DefaultHeightmapResource = /*#__PURE__*/function () {
|
|
466
497
|
function DefaultHeightmapResource() {
|
|
467
|
-
var
|
|
468
|
-
_opts$skipOddLevels,
|
|
469
|
-
_opts$tileSize,
|
|
470
|
-
_opts$maxZoom;
|
|
498
|
+
var _opts$tileSize, _opts$maxZoom;
|
|
471
499
|
var opts = arguments.length > 0 && arguments[0] !== undefined ? arguments[0] : {};
|
|
472
500
|
_classCallCheck(this, DefaultHeightmapResource);
|
|
473
501
|
_defineProperty(this, "resource", null);
|
|
474
502
|
_defineProperty(this, "tileSize", 256);
|
|
475
|
-
_defineProperty(this, "skipOddLevels", false);
|
|
476
|
-
_defineProperty(this, "getTilePixels", /*#__PURE__*/function () {
|
|
477
|
-
var _ref = _asyncToGenerator(/*#__PURE__*/_regeneratorRuntime().mark(function _callee(coords) {
|
|
478
|
-
var url, img;
|
|
479
|
-
return _regeneratorRuntime().wrap(function _callee$(_context) {
|
|
480
|
-
while (1) switch (_context.prev = _context.next) {
|
|
481
|
-
case 0:
|
|
482
|
-
url = _this.buildTileURL(coords);
|
|
483
|
-
_context.next = 3;
|
|
484
|
-
return loadImage(url);
|
|
485
|
-
case 3:
|
|
486
|
-
img = _context.sent;
|
|
487
|
-
return _context.abrupt("return", _this.getPixels(img));
|
|
488
|
-
case 5:
|
|
489
|
-
case "end":
|
|
490
|
-
return _context.stop();
|
|
491
|
-
}
|
|
492
|
-
}, _callee);
|
|
493
|
-
}));
|
|
494
|
-
return function (_x) {
|
|
495
|
-
return _ref.apply(this, arguments);
|
|
496
|
-
};
|
|
497
|
-
}());
|
|
498
503
|
if (opts.url) {
|
|
499
|
-
this.resource = cesium.Resource
|
|
504
|
+
this.resource = new cesium.Resource({
|
|
505
|
+
url: opts.url
|
|
506
|
+
});
|
|
507
|
+
}
|
|
508
|
+
this.skipZoomLevel = function () {
|
|
509
|
+
return false;
|
|
510
|
+
};
|
|
511
|
+
if (opts.skipZoomLevels) {
|
|
512
|
+
if (Array.isArray(opts.skipZoomLevels)) {
|
|
513
|
+
var _skipZoomLevels = opts.skipZoomLevels;
|
|
514
|
+
this.skipZoomLevel = function (z) {
|
|
515
|
+
return _skipZoomLevels.includes(z);
|
|
516
|
+
};
|
|
517
|
+
} else {
|
|
518
|
+
this.skipZoomLevel = opts.skipZoomLevels;
|
|
519
|
+
}
|
|
520
|
+
} else if (opts.skipOddLevels) {
|
|
521
|
+
this.skipZoomLevel = function (z) {
|
|
522
|
+
return z % 2 == 1;
|
|
523
|
+
};
|
|
500
524
|
}
|
|
501
|
-
this.skipOddLevels = (_opts$skipOddLevels = opts.skipOddLevels) !== null && _opts$skipOddLevels !== void 0 ? _opts$skipOddLevels : false;
|
|
502
525
|
this.tileSize = (_opts$tileSize = opts.tileSize) !== null && _opts$tileSize !== void 0 ? _opts$tileSize : 256;
|
|
503
526
|
this.maxZoom = (_opts$maxZoom = opts.maxZoom) !== null && _opts$maxZoom !== void 0 ? _opts$maxZoom : 15;
|
|
504
527
|
this.contextQueue = [];
|
|
@@ -534,26 +557,48 @@ var DefaultHeightmapResource = /*#__PURE__*/function () {
|
|
|
534
557
|
return pixels;
|
|
535
558
|
}
|
|
536
559
|
}, {
|
|
537
|
-
key: "
|
|
538
|
-
value: function
|
|
539
|
-
var _this$resource;
|
|
560
|
+
key: "getTileResource",
|
|
561
|
+
value: function getTileResource(tileCoords) {
|
|
540
562
|
// reverseY for TMS tiling (https://gist.github.com/tmcw/4954720)
|
|
541
563
|
// See tiling schemes here: https://www.maptiler.com/google-maps-coordinates-tile-bounds-projection/
|
|
542
564
|
var z = tileCoords.z,
|
|
543
565
|
y = tileCoords.y;
|
|
544
|
-
return
|
|
566
|
+
return this.resource.getDerivedResource({
|
|
545
567
|
templateValues: _objectSpread2(_objectSpread2({}, tileCoords), {}, {
|
|
546
568
|
reverseY: Math.pow(2, z) - y - 1
|
|
547
569
|
}),
|
|
548
570
|
preserveQueryParameters: true
|
|
549
|
-
})
|
|
571
|
+
});
|
|
572
|
+
}
|
|
573
|
+
}, {
|
|
574
|
+
key: "getTilePixels",
|
|
575
|
+
value: function getTilePixels(coords) {
|
|
576
|
+
var _this = this;
|
|
577
|
+
var resource = this.getTileResource(coords);
|
|
578
|
+
var request = resource.fetchImage({
|
|
579
|
+
preferImageBitmap: false,
|
|
580
|
+
// @ts-ignore
|
|
581
|
+
retryAttempts: 3
|
|
582
|
+
});
|
|
583
|
+
if (request == null) return undefined;
|
|
584
|
+
return request.then(function (img) {
|
|
585
|
+
return (
|
|
586
|
+
// @ts-ignore
|
|
587
|
+
_this.getPixels(img)
|
|
588
|
+
);
|
|
589
|
+
});
|
|
550
590
|
}
|
|
551
591
|
}, {
|
|
552
592
|
key: "getTileDataAvailable",
|
|
553
|
-
value: function getTileDataAvailable(
|
|
554
|
-
var z =
|
|
593
|
+
value: function getTileDataAvailable(_ref) {
|
|
594
|
+
var z = _ref.z;
|
|
555
595
|
if (z == this.maxZoom) return true;
|
|
556
|
-
|
|
596
|
+
/* Weird hack:
|
|
597
|
+
For some reason, request render mode breaks if zoom 1 tiles are disabled.
|
|
598
|
+
So we have to make sure that we always report zoom 1 tiles as available.
|
|
599
|
+
*/
|
|
600
|
+
if (z < 2) return true;
|
|
601
|
+
if (this.skipZoomLevel(z)) return false;
|
|
557
602
|
if (z > this.maxZoom) return false;
|
|
558
603
|
return true;
|
|
559
604
|
}
|
|
@@ -577,6 +622,7 @@ var MapboxTerrainResource = /*#__PURE__*/function (_DefaultHeightmapReso) {
|
|
|
577
622
|
_defineProperty(_this, "credit", new cesium.Credit("Mapbox"));
|
|
578
623
|
var highResolution = (_opts$highResolution = opts.highResolution) !== null && _opts$highResolution !== void 0 ? _opts$highResolution : false;
|
|
579
624
|
var format = (_opts$imageFormat = opts.imageFormat) !== null && _opts$imageFormat !== void 0 ? _opts$imageFormat : ImageFormat.WEBP;
|
|
625
|
+
var urlTemplate = opts.urlTemplate;
|
|
580
626
|
|
|
581
627
|
// overrides based on highResolution flag
|
|
582
628
|
if (highResolution) {
|
|
@@ -587,7 +633,10 @@ var MapboxTerrainResource = /*#__PURE__*/function (_DefaultHeightmapReso) {
|
|
|
587
633
|
_this.tileSize = 512;
|
|
588
634
|
}
|
|
589
635
|
}
|
|
590
|
-
|
|
636
|
+
var defaultURL = "https://api.mapbox.com/v4/mapbox.terrain-rgb/{z}/{x}/{y}".concat(highResolution ? "@2x" : "", ".").concat(format);
|
|
637
|
+
_this.resource = new cesium.Resource({
|
|
638
|
+
url: urlTemplate !== null && urlTemplate !== void 0 ? urlTemplate : defaultURL
|
|
639
|
+
});
|
|
591
640
|
if (opts.accessToken) {
|
|
592
641
|
_this.resource.setQueryParameters({
|
|
593
642
|
access_token: opts.accessToken
|
|
@@ -599,6 +648,76 @@ var MapboxTerrainResource = /*#__PURE__*/function (_DefaultHeightmapReso) {
|
|
|
599
648
|
return _createClass(MapboxTerrainResource);
|
|
600
649
|
}(DefaultHeightmapResource);
|
|
601
650
|
|
|
651
|
+
/** Mapbox Terrain-RGB default decode function
|
|
652
|
+
* (r * 256 * 256) / 10 + (g * 256) / 10 + b / 10 - 10000
|
|
653
|
+
*/
|
|
654
|
+
var defaultMapboxDecodeRgb = function defaultMapboxDecodeRgb(r, g, b, a) {
|
|
655
|
+
return r * 6553.6 + g * 25.6 + b * 0.1 - 10000;
|
|
656
|
+
};
|
|
657
|
+
function rgbTerrainToGrid(png, decodeRgb) {
|
|
658
|
+
// maybe we should do this on the GPU using REGL?
|
|
659
|
+
// but that would require GPU -> CPU -> GPU
|
|
660
|
+
var gridSize = png.shape[0] + 1;
|
|
661
|
+
var terrain = new Float32Array(gridSize * gridSize);
|
|
662
|
+
var tileSize = png.shape[0];
|
|
663
|
+
var decode = decodeRgb !== null && decodeRgb !== void 0 ? decodeRgb : defaultMapboxDecodeRgb;
|
|
664
|
+
|
|
665
|
+
// decode terrain values
|
|
666
|
+
for (var y = 0; y < tileSize; y++) {
|
|
667
|
+
for (var x = 0; x < tileSize; x++) {
|
|
668
|
+
var yc = y;
|
|
669
|
+
var _r = png.get(x, yc, 0);
|
|
670
|
+
var _g = png.get(x, yc, 1);
|
|
671
|
+
var _b = png.get(x, yc, 2);
|
|
672
|
+
var _a = png.get(x, yc, 3);
|
|
673
|
+
terrain[y * gridSize + x] = decode(_r, _g, _b, _a);
|
|
674
|
+
}
|
|
675
|
+
}
|
|
676
|
+
// backfill right and bottom borders
|
|
677
|
+
for (var _x = 0; _x < gridSize - 1; _x++) {
|
|
678
|
+
terrain[gridSize * (gridSize - 1) + _x] = terrain[gridSize * (gridSize - 2) + _x];
|
|
679
|
+
}
|
|
680
|
+
for (var _y = 0; _y < gridSize; _y++) {
|
|
681
|
+
terrain[gridSize * _y + gridSize - 1] = terrain[gridSize * _y + gridSize - 2];
|
|
682
|
+
}
|
|
683
|
+
return terrain;
|
|
684
|
+
}
|
|
685
|
+
function subsetByWindow(array, window, augmented) {
|
|
686
|
+
var sz = Math.sqrt(array.length);
|
|
687
|
+
var x0 = window.x0;
|
|
688
|
+
var x1 = window.x1;
|
|
689
|
+
var y0 = window.y0;
|
|
690
|
+
var y1 = window.y1;
|
|
691
|
+
var aug = augmented ? 1 : 0;
|
|
692
|
+
var n = Math.floor(x1 - x0) + aug;
|
|
693
|
+
var m = Math.floor(y1 - y0) + aug;
|
|
694
|
+
var result = new Float32Array(n * m);
|
|
695
|
+
for (var i = 0; i < m; i++) {
|
|
696
|
+
for (var j = 0; j < n; j++) {
|
|
697
|
+
result[i * n + j] = array[(i + y0) * sz + j + x0];
|
|
698
|
+
}
|
|
699
|
+
}
|
|
700
|
+
return result;
|
|
701
|
+
}
|
|
702
|
+
function testMeshData() {
|
|
703
|
+
return {
|
|
704
|
+
minimumHeight: -100,
|
|
705
|
+
maximumHeight: 2101,
|
|
706
|
+
quantizedVertices: new Uint16Array([
|
|
707
|
+
// order is SW NW SE NE
|
|
708
|
+
// longitude
|
|
709
|
+
0, 0, 32767, 32767,
|
|
710
|
+
// latitude
|
|
711
|
+
0, 32767, 0, 32767,
|
|
712
|
+
// heights
|
|
713
|
+
16384, 0, 32767, 16384]),
|
|
714
|
+
indices: new Uint16Array([0, 3, 1, 0, 2, 3]),
|
|
715
|
+
westIndices: [0, 1],
|
|
716
|
+
southIndices: [0, 1],
|
|
717
|
+
eastIndices: [2, 3],
|
|
718
|
+
northIndices: [1, 3]
|
|
719
|
+
};
|
|
720
|
+
}
|
|
602
721
|
function _emptyMesh(n) {
|
|
603
722
|
n = Math.max(n, 2);
|
|
604
723
|
var nTriangles = Math.pow(n - 1, 2) * 2;
|
|
@@ -660,6 +779,68 @@ function emptyMesh(n) {
|
|
|
660
779
|
}
|
|
661
780
|
}
|
|
662
781
|
|
|
782
|
+
/** Terrain workers should return a quantized mesh */
|
|
783
|
+
|
|
784
|
+
function createQuantizedMeshData(tile, mesh, tileSize, terrain) {
|
|
785
|
+
/** Terrain is passed through so we can keep track of it
|
|
786
|
+
* for overscaled tiles
|
|
787
|
+
*/
|
|
788
|
+
|
|
789
|
+
var xvals = [];
|
|
790
|
+
var yvals = [];
|
|
791
|
+
var heightMeters = [];
|
|
792
|
+
var northIndices = [];
|
|
793
|
+
var southIndices = [];
|
|
794
|
+
var eastIndices = [];
|
|
795
|
+
var westIndices = [];
|
|
796
|
+
var minimumHeight = Infinity;
|
|
797
|
+
var maximumHeight = -Infinity;
|
|
798
|
+
var scalar = 32768.0 / tileSize;
|
|
799
|
+
|
|
800
|
+
// There appears to be a problem with the x/y indexing when using 512x512 tiles
|
|
801
|
+
// This may be solved by increasing the minumumErrorLevel in the terrain provider
|
|
802
|
+
for (var ix = 0; ix < mesh.vertices.length / 2; ix++) {
|
|
803
|
+
var vertexIx = ix;
|
|
804
|
+
var px = mesh.vertices[ix * 2];
|
|
805
|
+
var py = mesh.vertices[ix * 2 + 1];
|
|
806
|
+
var height = tile.terrain[py * (tileSize + 1) + px];
|
|
807
|
+
if (height > maximumHeight) maximumHeight = height;
|
|
808
|
+
if (height < minimumHeight) minimumHeight = height;
|
|
809
|
+
heightMeters.push(height);
|
|
810
|
+
if (py == 0) northIndices.push(vertexIx);
|
|
811
|
+
if (py == tileSize) southIndices.push(vertexIx);
|
|
812
|
+
if (px == 0) westIndices.push(vertexIx);
|
|
813
|
+
if (px == tileSize) eastIndices.push(vertexIx);
|
|
814
|
+
var xv = px * scalar;
|
|
815
|
+
var yv = (tileSize - py) * scalar;
|
|
816
|
+
xvals.push(xv);
|
|
817
|
+
yvals.push(yv);
|
|
818
|
+
}
|
|
819
|
+
var heightRange = maximumHeight - minimumHeight;
|
|
820
|
+
var heights = heightMeters.map(function (d) {
|
|
821
|
+
if (heightRange < 1) return 0;
|
|
822
|
+
return (d - minimumHeight) * (32768.0 / heightRange);
|
|
823
|
+
});
|
|
824
|
+
var triangles = new Uint16Array(mesh.triangles);
|
|
825
|
+
var quantizedVertices = new Uint16Array(//verts
|
|
826
|
+
[].concat(xvals, yvals, _toConsumableArray(heights)));
|
|
827
|
+
|
|
828
|
+
// SE NW NE
|
|
829
|
+
// NE NW SE
|
|
830
|
+
|
|
831
|
+
return {
|
|
832
|
+
minimumHeight: minimumHeight,
|
|
833
|
+
maximumHeight: maximumHeight,
|
|
834
|
+
quantizedVertices: quantizedVertices,
|
|
835
|
+
indices: triangles,
|
|
836
|
+
westIndices: westIndices,
|
|
837
|
+
southIndices: southIndices,
|
|
838
|
+
eastIndices: eastIndices,
|
|
839
|
+
northIndices: northIndices,
|
|
840
|
+
quantizedHeights: terrain
|
|
841
|
+
};
|
|
842
|
+
}
|
|
843
|
+
|
|
663
844
|
var resolves = {};
|
|
664
845
|
var rejects = {};
|
|
665
846
|
var globalMsgId = 0; // Activate calculation in the worker, returning a promise
|
|
@@ -667,17 +848,17 @@ function sendMessage(_x, _x2, _x3) {
|
|
|
667
848
|
return _sendMessage.apply(this, arguments);
|
|
668
849
|
} // Handle incoming calculation result
|
|
669
850
|
function _sendMessage() {
|
|
670
|
-
_sendMessage = _asyncToGenerator(/*#__PURE__*/_regeneratorRuntime().mark(function
|
|
851
|
+
_sendMessage = _asyncToGenerator(/*#__PURE__*/_regeneratorRuntime().mark(function _callee3(worker, payload, transferableObjects) {
|
|
671
852
|
var msgId, msg;
|
|
672
|
-
return _regeneratorRuntime().wrap(function
|
|
673
|
-
while (1) switch (
|
|
853
|
+
return _regeneratorRuntime().wrap(function _callee3$(_context3) {
|
|
854
|
+
while (1) switch (_context3.prev = _context3.next) {
|
|
674
855
|
case 0:
|
|
675
856
|
msgId = globalMsgId++;
|
|
676
857
|
msg = {
|
|
677
858
|
id: msgId,
|
|
678
859
|
payload: payload
|
|
679
860
|
};
|
|
680
|
-
return
|
|
861
|
+
return _context3.abrupt("return", new Promise(function (resolve, reject) {
|
|
681
862
|
// save callbacks for later
|
|
682
863
|
resolves[msgId] = resolve;
|
|
683
864
|
rejects[msgId] = reject;
|
|
@@ -685,9 +866,9 @@ function _sendMessage() {
|
|
|
685
866
|
}));
|
|
686
867
|
case 3:
|
|
687
868
|
case "end":
|
|
688
|
-
return
|
|
869
|
+
return _context3.stop();
|
|
689
870
|
}
|
|
690
|
-
},
|
|
871
|
+
}, _callee3);
|
|
691
872
|
}));
|
|
692
873
|
return _sendMessage.apply(this, arguments);
|
|
693
874
|
}
|
|
@@ -720,6 +901,9 @@ function handleMessage(msg) {
|
|
|
720
901
|
var WorkerFarm = /*#__PURE__*/function () {
|
|
721
902
|
function WorkerFarm(opts) {
|
|
722
903
|
_classCallCheck(this, WorkerFarm);
|
|
904
|
+
_defineProperty(this, "inProgressWorkers", 0);
|
|
905
|
+
_defineProperty(this, "maxWorkers", 5);
|
|
906
|
+
_defineProperty(this, "processingQueue", []);
|
|
723
907
|
this.worker = opts.worker;
|
|
724
908
|
this.worker.onmessage = handleMessage;
|
|
725
909
|
}
|
|
@@ -727,14 +911,17 @@ var WorkerFarm = /*#__PURE__*/function () {
|
|
|
727
911
|
key: "scheduleTask",
|
|
728
912
|
value: function () {
|
|
729
913
|
var _scheduleTask = _asyncToGenerator(/*#__PURE__*/_regeneratorRuntime().mark(function _callee(params, transferableObjects) {
|
|
914
|
+
var res;
|
|
730
915
|
return _regeneratorRuntime().wrap(function _callee$(_context) {
|
|
731
916
|
while (1) switch (_context.prev = _context.next) {
|
|
732
917
|
case 0:
|
|
733
918
|
_context.next = 2;
|
|
734
919
|
return sendMessage(this.worker, params, transferableObjects);
|
|
735
920
|
case 2:
|
|
736
|
-
|
|
737
|
-
|
|
921
|
+
res = _context.sent;
|
|
922
|
+
this.releaseWorker();
|
|
923
|
+
return _context.abrupt("return", res);
|
|
924
|
+
case 5:
|
|
738
925
|
case "end":
|
|
739
926
|
return _context.stop();
|
|
740
927
|
}
|
|
@@ -745,6 +932,45 @@ var WorkerFarm = /*#__PURE__*/function () {
|
|
|
745
932
|
}
|
|
746
933
|
return scheduleTask;
|
|
747
934
|
}()
|
|
935
|
+
}, {
|
|
936
|
+
key: "holdForAvailableWorker",
|
|
937
|
+
value: function () {
|
|
938
|
+
var _holdForAvailableWorker = _asyncToGenerator(/*#__PURE__*/_regeneratorRuntime().mark(function _callee2() {
|
|
939
|
+
var _this = this;
|
|
940
|
+
var resultPromise;
|
|
941
|
+
return _regeneratorRuntime().wrap(function _callee2$(_context2) {
|
|
942
|
+
while (1) switch (_context2.prev = _context2.next) {
|
|
943
|
+
case 0:
|
|
944
|
+
if (this.inProgressWorkers > this.maxWorkers) {
|
|
945
|
+
resultPromise = new Promise(function (resolve, reject) {
|
|
946
|
+
_this.processingQueue.push(resolve);
|
|
947
|
+
});
|
|
948
|
+
} else {
|
|
949
|
+
resultPromise = Promise.resolve(null);
|
|
950
|
+
}
|
|
951
|
+
_context2.next = 3;
|
|
952
|
+
return resultPromise;
|
|
953
|
+
case 3:
|
|
954
|
+
this.inProgressWorkers += 1;
|
|
955
|
+
case 4:
|
|
956
|
+
case "end":
|
|
957
|
+
return _context2.stop();
|
|
958
|
+
}
|
|
959
|
+
}, _callee2, this);
|
|
960
|
+
}));
|
|
961
|
+
function holdForAvailableWorker() {
|
|
962
|
+
return _holdForAvailableWorker.apply(this, arguments);
|
|
963
|
+
}
|
|
964
|
+
return holdForAvailableWorker;
|
|
965
|
+
}()
|
|
966
|
+
}, {
|
|
967
|
+
key: "releaseWorker",
|
|
968
|
+
value: function releaseWorker() {
|
|
969
|
+
this.inProgressWorkers -= 1;
|
|
970
|
+
if (this.processingQueue.length > 0) {
|
|
971
|
+
this.processingQueue.shift()();
|
|
972
|
+
}
|
|
973
|
+
}
|
|
748
974
|
}]);
|
|
749
975
|
}();
|
|
750
976
|
|
|
@@ -792,7 +1018,276 @@ var WorkerFarmTerrainDecoder = /*#__PURE__*/function (_DefaultTerrainDecode) {
|
|
|
792
1018
|
}]);
|
|
793
1019
|
}(DefaultTerrainDecoder);
|
|
794
1020
|
|
|
1021
|
+
function decodeBase64(base64, enableUnicode) {
|
|
1022
|
+
var binaryString = atob(base64);
|
|
1023
|
+
if (enableUnicode) {
|
|
1024
|
+
var binaryView = new Uint8Array(binaryString.length);
|
|
1025
|
+
for (var i = 0, n = binaryString.length; i < n; ++i) {
|
|
1026
|
+
binaryView[i] = binaryString.charCodeAt(i);
|
|
1027
|
+
}
|
|
1028
|
+
const decoder = new TextDecoder("utf-16le");
|
|
1029
|
+
return decoder.decode(new Uint16Array(binaryView.buffer));
|
|
1030
|
+
}
|
|
1031
|
+
return binaryString;
|
|
1032
|
+
}
|
|
1033
|
+
|
|
1034
|
+
function createURL(base64, sourcemapArg, enableUnicodeArg) {
|
|
1035
|
+
var sourcemap = sourcemapArg === undefined ? null : sourcemapArg;
|
|
1036
|
+
var enableUnicode = enableUnicodeArg === undefined ? false : enableUnicodeArg;
|
|
1037
|
+
var source = decodeBase64(base64, enableUnicode);
|
|
1038
|
+
var start = source.indexOf('\n', 10) + 1;
|
|
1039
|
+
var body = source.substring(start) + (sourcemap ? '\/\/# sourceMappingURL=' + sourcemap : '');
|
|
1040
|
+
var blob = new Blob([body], { type: 'application/javascript' });
|
|
1041
|
+
return URL.createObjectURL(blob);
|
|
1042
|
+
}
|
|
1043
|
+
|
|
1044
|
+
function createBase64WorkerFactory(base64, sourcemapArg, enableUnicodeArg) {
|
|
1045
|
+
var url;
|
|
1046
|
+
return function WorkerFactory(options) {
|
|
1047
|
+
url = url || createURL(base64, sourcemapArg, enableUnicodeArg);
|
|
1048
|
+
return new Worker(url, options);
|
|
1049
|
+
};
|
|
1050
|
+
}
|
|
1051
|
+
|
|
1052
|
+
var WorkerFactory$1 = /*#__PURE__*/createBase64WorkerFactory('/* rollup-plugin-web-worker-loader */
(function () {
  'use strict';

  function _arrayLikeToArray(r, a) {
    (null == a || a > r.length) && (a = r.length);
    for (var e = 0, n = Array(a); e < a; e++) n[e] = r[e];
    return n;
  }
  function _arrayWithoutHoles(r) {
    if (Array.isArray(r)) return _arrayLikeToArray(r);
  }
  function _iterableToArray(r) {
    if ("undefined" != typeof Symbol && null != r[Symbol.iterator] || null != r["@@iterator"]) return Array.from(r);
  }
  function _nonIterableSpread() {
    throw new TypeError("Invalid attempt to spread non-iterable instance.\nIn order to be iterable, non-array objects must have a [Symbol.iterator]() method.");
  }
  function _toConsumableArray(r) {
    return _arrayWithoutHoles(r) || _iterableToArray(r) || _unsupportedIterableToArray(r) || _nonIterableSpread();
  }
  function _unsupportedIterableToArray(r, a) {
    if (r) {
      if ("string" == typeof r) return _arrayLikeToArray(r, a);
      var t = {}.toString.call(r).slice(8, -1);
      return "Object" === t && r.constructor && (t = r.constructor.name), "Map" === t || "Set" === t ? Array.from(r) : "Arguments" === t || /^(?:Ui|I)nt(?:8|16|32)(?:Clamped)?Array$/.test(t) ? _arrayLikeToArray(r, a) : void 0;
    }
  }

  /** Terrain workers should return a quantized mesh */

  function createQuantizedMeshData(tile, mesh, tileSize, terrain) {
    /** Terrain is passed through so we can keep track of it
     * for overscaled tiles
     */

    var xvals = [];
    var yvals = [];
    var heightMeters = [];
    var northIndices = [];
    var southIndices = [];
    var eastIndices = [];
    var westIndices = [];
    var minimumHeight = Infinity;
    var maximumHeight = -Infinity;
    var scalar = 32768.0 / tileSize;

    // There appears to be a problem with the x/y indexing when using 512x512 tiles
    // This may be solved by increasing the minumumErrorLevel in the terrain provider
    for (var ix = 0; ix < mesh.vertices.length / 2; ix++) {
      var vertexIx = ix;
      var px = mesh.vertices[ix * 2];
      var py = mesh.vertices[ix * 2 + 1];
      var height = tile.terrain[py * (tileSize + 1) + px];
      if (height > maximumHeight) maximumHeight = height;
      if (height < minimumHeight) minimumHeight = height;
      heightMeters.push(height);
      if (py == 0) northIndices.push(vertexIx);
      if (py == tileSize) southIndices.push(vertexIx);
      if (px == 0) westIndices.push(vertexIx);
      if (px == tileSize) eastIndices.push(vertexIx);
      var xv = px * scalar;
      var yv = (tileSize - py) * scalar;
      xvals.push(xv);
      yvals.push(yv);
    }
    var heightRange = maximumHeight - minimumHeight;
    var heights = heightMeters.map(function (d) {
      if (heightRange < 1) return 0;
      return (d - minimumHeight) * (32768.0 / heightRange);
    });
    var triangles = new Uint16Array(mesh.triangles);
    var quantizedVertices = new Uint16Array(//verts
    [].concat(xvals, yvals, _toConsumableArray(heights)));

    // SE NW NE
    // NE NW SE

    return {
      minimumHeight: minimumHeight,
      maximumHeight: maximumHeight,
      quantizedVertices: quantizedVertices,
      indices: triangles,
      westIndices: westIndices,
      southIndices: southIndices,
      eastIndices: eastIndices,
      northIndices: northIndices,
      quantizedHeights: terrain
    };
  }

  class Martini {
      constructor(gridSize = 257) {
          this.gridSize = gridSize;
          const tileSize = gridSize - 1;
          if (tileSize & (tileSize - 1)) throw new Error(
              `Expected grid size to be 2^n+1, got ${gridSize}.`);

          this.numTriangles = tileSize * tileSize * 2 - 2;
          this.numParentTriangles = this.numTriangles - tileSize * tileSize;

          this.indices = new Uint32Array(this.gridSize * this.gridSize);

          // coordinates for all possible triangles in an RTIN tile
          this.coords = new Uint16Array(this.numTriangles * 4);

          // get triangle coordinates from its index in an implicit binary tree
          for (let i = 0; i < this.numTriangles; i++) {
              let id = i + 2;
              let ax = 0, ay = 0, bx = 0, by = 0, cx = 0, cy = 0;
              if (id & 1) {
                  bx = by = cx = tileSize; // bottom-left triangle
              } else {
                  ax = ay = cy = tileSize; // top-right triangle
              }
              while ((id >>= 1) > 1) {
                  const mx = (ax + bx) >> 1;
                  const my = (ay + by) >> 1;

                  if (id & 1) { // left half
                      bx = ax; by = ay;
                      ax = cx; ay = cy;
                  } else { // right half
                      ax = bx; ay = by;
                      bx = cx; by = cy;
                  }
                  cx = mx; cy = my;
              }
              const k = i * 4;
              this.coords[k + 0] = ax;
              this.coords[k + 1] = ay;
              this.coords[k + 2] = bx;
              this.coords[k + 3] = by;
          }
      }

      createTile(terrain) {
          return new Tile(terrain, this);
      }
  }

  class Tile {
      constructor(terrain, martini) {
          const size = martini.gridSize;
          if (terrain.length !== size * size) throw new Error(
              `Expected terrain data of length ${size * size} (${size} x ${size}), got ${terrain.length}.`);

          this.terrain = terrain;
          this.martini = martini;
          this.errors = new Float32Array(terrain.length);
          this.update();
      }

      update() {
          const {numTriangles, numParentTriangles, coords, gridSize: size} = this.martini;
          const {terrain, errors} = this;

          // iterate over all possible triangles, starting from the smallest level
          for (let i = numTriangles - 1; i >= 0; i--) {
              const k = i * 4;
              const ax = coords[k + 0];
              const ay = coords[k + 1];
              const bx = coords[k + 2];
              const by = coords[k + 3];
              const mx = (ax + bx) >> 1;
              const my = (ay + by) >> 1;
              const cx = mx + my - ay;
              const cy = my + ax - mx;

              // calculate error in the middle of the long edge of the triangle
              const interpolatedHeight = (terrain[ay * size + ax] + terrain[by * size + bx]) / 2;
              const middleIndex = my * size + mx;
              const middleError = Math.abs(interpolatedHeight - terrain[middleIndex]);

              errors[middleIndex] = Math.max(errors[middleIndex], middleError);

              if (i < numParentTriangles) { // bigger triangles; accumulate error with children
                  const leftChildIndex = ((ay + cy) >> 1) * size + ((ax + cx) >> 1);
                  const rightChildIndex = ((by + cy) >> 1) * size + ((bx + cx) >> 1);
                  errors[middleIndex] = Math.max(errors[middleIndex], errors[leftChildIndex], errors[rightChildIndex]);
              }
          }
      }

      getMesh(maxError = 0, maxLength = null) {
          const {gridSize: size, indices} = this.martini;
          const {errors} = this;
          let numVertices = 0;
          let numTriangles = 0;
          const max = size - 1;

          // The maxLength parameter will cause triangles to be generated until the legs are below this length
          // It is meant to support cases where a certain mesh density is required to do spherical math on digital globes
          const maxScale = maxLength || size;

          // use an index grid to keep track of vertices that were already used to avoid duplication
          indices.fill(0);

          // retrieve mesh in two stages that both traverse the error map:
          // - countElements: find used vertices (and assign each an index), and count triangles (for minimum allocation)
          // - processTriangle: fill the allocated vertices & triangles typed arrays

          function countElements(ax, ay, bx, by, cx, cy) {
              const mx = (ax + bx) >> 1;
              const my = (ay + by) >> 1;

              const legLength = Math.abs(ax - cx) + Math.abs(ay - cy);
              if ((legLength > 1 && errors[my * size + mx] > maxError) || legLength > maxScale) {
                  countElements(cx, cy, ax, ay, mx, my);
                  countElements(bx, by, cx, cy, mx, my);
              } else {
                  indices[ay * size + ax] = indices[ay * size + ax] || ++numVertices;
                  indices[by * size + bx] = indices[by * size + bx] || ++numVertices;
                  indices[cy * size + cx] = indices[cy * size + cx] || ++numVertices;
                  numTriangles++;
              }
          }
          countElements(0, 0, max, max, max, 0);
          countElements(max, max, 0, 0, 0, max);

          const vertices = new Uint16Array(numVertices * 2);
          const triangles = new Uint32Array(numTriangles * 3);
          let triIndex = 0;

          function processTriangle(ax, ay, bx, by, cx, cy) {
              const mx = (ax + bx) >> 1;
              const my = (ay + by) >> 1;

              const legLength = Math.abs(ax - cx) + Math.abs(ay - cy);
              if ((legLength > 1 && errors[my * size + mx] > maxError) || legLength > maxScale) {
                  // triangle doesn't approximate the surface well enough; drill down further
                  processTriangle(cx, cy, ax, ay, mx, my);
                  processTriangle(bx, by, cx, cy, mx, my);

              } else {
                  // add a triangle
                  const a = indices[ay * size + ax] - 1;
                  const b = indices[by * size + bx] - 1;
                  const c = indices[cy * size + cx] - 1;

                  vertices[2 * a] = ax;
                  vertices[2 * a + 1] = ay;

                  vertices[2 * b] = bx;
                  vertices[2 * b + 1] = by;

                  vertices[2 * c] = cx;
                  vertices[2 * c + 1] = cy;

                  triangles[triIndex++] = a;
                  triangles[triIndex++] = b;
                  triangles[triIndex++] = c;
              }
          }
          processTriangle(0, 0, max, max, max, 0);
          processTriangle(max, max, 0, 0, 0, max);

          return {vertices, triangles};
      }
  }

  /** Worker to upsample terrain meshes */
  // https://github.com/CesiumGS/cesium/blob/1.76/Source/WorkersES6/createVerticesFromQuantizedTerrainMesh.js

  var martiniCache = {};
  function decodeTerrain(parameters, transferableObjects) {
    var _martiniCache$tileSiz;
    var heightData = parameters.heightData,
      _parameters$tileSize = parameters.tileSize,
      tileSize = _parameters$tileSize === void 0 ? 256 : _parameters$tileSize,
      errorLevel = parameters.errorLevel,
      maxVertexDistance = parameters.maxVertexDistance;

    // Height data can be either an array of numbers (for pre-existing terrain data)
    // or an image data array (for decoding from an image)

    var terrain = heightData;

    // Tile size must be maintained through the life of the worker
    (_martiniCache$tileSiz = martiniCache[tileSize]) !== null && _martiniCache$tileSiz !== void 0 ? _martiniCache$tileSiz : martiniCache[tileSize] = new Martini(tileSize + 1);
    var tile = martiniCache[tileSize].createTile(terrain);

    // get a mesh (vertices and triangles indices) for a 10m error
    var mesh = tile.getMesh(errorLevel, Math.min(maxVertexDistance, tileSize));
    var res = createQuantizedMeshData(tile, mesh, tileSize,
    // Only include vertex data if anticipate upscaling tile
    terrain );
    transferableObjects.push(res.indices.buffer);
    transferableObjects.push(res.quantizedVertices.buffer);
    if (res.quantizedHeights) {
      transferableObjects.push(res.quantizedHeights.buffer);
    }
    return res;
  }
  self.onmessage = function (msg) {
    var _msg$data = msg.data,
      id = _msg$data.id,
      payload = _msg$data.payload;
    if (id == null) return;
    var objects = [];
    var res = null;
    try {
      res = decodeTerrain(payload, objects);
      // @ts-ignore
      self.postMessage({
        id: id,
        payload: res
      }, objects);
    } catch (err) {
      var _err$message;
      var _msg = (_err$message = err.message) !== null && _err$message !== void 0 ? _err$message : err;
      self.postMessage({
        id: id,
        err: _msg.toString()
      });
    } finally {
      res = null;
      objects = null;
    }
  };

})();

', null, false);
|
|
1053
|
+
/* eslint-enable */
|
|
1054
|
+
|
|
1055
|
+
var _excluded = ["tilingScheme", "overscaleFactor"];
|
|
1056
|
+
function createTerrainMesh(data, meta) {
|
|
1057
|
+
var minimumHeight = data.minimumHeight,
|
|
1058
|
+
maximumHeight = data.maximumHeight,
|
|
1059
|
+
quantizedVertices = data.quantizedVertices,
|
|
1060
|
+
indices = data.indices,
|
|
1061
|
+
westIndices = data.westIndices,
|
|
1062
|
+
southIndices = data.southIndices,
|
|
1063
|
+
eastIndices = data.eastIndices,
|
|
1064
|
+
northIndices = data.northIndices,
|
|
1065
|
+
quantizedHeights = data.quantizedHeights;
|
|
1066
|
+
var errorLevel = meta.errorLevel,
|
|
1067
|
+
tileSize = meta.tileSize,
|
|
1068
|
+
maxVertexDistance = meta.maxVertexDistance,
|
|
1069
|
+
tileRect = meta.tileRect,
|
|
1070
|
+
ellipsoid = meta.ellipsoid,
|
|
1071
|
+
overscaleFactor = meta.overscaleFactor;
|
|
1072
|
+
var err = errorLevel;
|
|
1073
|
+
var skirtHeight = err * 20;
|
|
1074
|
+
|
|
1075
|
+
// Check if tileRect is not NaNs
|
|
1076
|
+
if (isNaN(tileRect.east) || isNaN(tileRect.north)) {
|
|
1077
|
+
throw new Error("Invalid tile rect");
|
|
1078
|
+
}
|
|
1079
|
+
var center = cesium.Rectangle.center(tileRect);
|
|
1080
|
+
|
|
1081
|
+
// Calculating occlusion height is kind of messy currently, but it definitely works
|
|
1082
|
+
var halfAngle = tileRect.width / 2;
|
|
1083
|
+
var dr = Math.cos(halfAngle); // half tile width since our ref point is at the center
|
|
1084
|
+
|
|
1085
|
+
var occlusionHeight = dr * ellipsoid.maximumRadius + maximumHeight;
|
|
1086
|
+
if (halfAngle > Math.PI / 4) {
|
|
1087
|
+
occlusionHeight = (1 + halfAngle) * ellipsoid.maximumRadius;
|
|
1088
|
+
}
|
|
1089
|
+
var occlusionPoint = new cesium.Cartographic(center.longitude, center.latitude, occlusionHeight) // Scaling factor of two just to be sure.
|
|
1090
|
+
;
|
|
1091
|
+
var horizonOcclusionPoint = ellipsoid.transformPositionToScaledSpace(cesium.Cartographic.toCartesian(occlusionPoint));
|
|
1092
|
+
var orientedBoundingBox = cesium.OrientedBoundingBox.fromRectangle(tileRect, minimumHeight, maximumHeight, ellipsoid);
|
|
1093
|
+
var boundingSphere = cesium.BoundingSphere.fromOrientedBoundingBox(orientedBoundingBox);
|
|
1094
|
+
return new RasterTerrainData({
|
|
1095
|
+
minimumHeight: minimumHeight,
|
|
1096
|
+
maximumHeight: maximumHeight,
|
|
1097
|
+
quantizedVertices: quantizedVertices,
|
|
1098
|
+
indices: indices,
|
|
1099
|
+
boundingSphere: boundingSphere,
|
|
1100
|
+
orientedBoundingBox: orientedBoundingBox,
|
|
1101
|
+
horizonOcclusionPoint: horizonOcclusionPoint,
|
|
1102
|
+
westIndices: westIndices,
|
|
1103
|
+
southIndices: southIndices,
|
|
1104
|
+
eastIndices: eastIndices,
|
|
1105
|
+
northIndices: northIndices,
|
|
1106
|
+
westSkirtHeight: skirtHeight,
|
|
1107
|
+
southSkirtHeight: skirtHeight,
|
|
1108
|
+
eastSkirtHeight: skirtHeight,
|
|
1109
|
+
northSkirtHeight: skirtHeight,
|
|
1110
|
+
childTileMask: 15,
|
|
1111
|
+
createdByUpsampling: overscaleFactor > 0,
|
|
1112
|
+
errorLevel: err,
|
|
1113
|
+
maxVertexDistance: maxVertexDistance,
|
|
1114
|
+
tileSize: tileSize,
|
|
1115
|
+
quantizedHeights: quantizedHeights
|
|
1116
|
+
});
|
|
1117
|
+
}
|
|
1118
|
+
function createEmptyMesh(opts) {
|
|
1119
|
+
var tileRect = opts.tileRect,
|
|
1120
|
+
tileCoord = opts.tileCoord,
|
|
1121
|
+
errorLevel = opts.errorLevel,
|
|
1122
|
+
ellipsoid = opts.ellipsoid,
|
|
1123
|
+
maxVertexDistance = opts.maxVertexDistance;
|
|
1124
|
+
var center = cesium.Rectangle.center(tileRect);
|
|
1125
|
+
var z = tileCoord.z;
|
|
1126
|
+
var latScalar = Math.min(Math.abs(Math.sin(center.latitude)), 0.995);
|
|
1127
|
+
var v = Math.max(Math.ceil(200 / (z + 1) * Math.pow(1 - latScalar, 0.25)), 4);
|
|
1128
|
+
var output = emptyMesh(v);
|
|
1129
|
+
// We use zero for some undefined values
|
|
1130
|
+
return createTerrainMesh(output, {
|
|
1131
|
+
tileRect: tileRect,
|
|
1132
|
+
ellipsoid: ellipsoid,
|
|
1133
|
+
errorLevel: errorLevel,
|
|
1134
|
+
overscaleFactor: 0,
|
|
1135
|
+
maxVertexDistance: maxVertexDistance,
|
|
1136
|
+
tileSize: output.tileSize
|
|
1137
|
+
});
|
|
1138
|
+
}
|
|
1139
|
+
var UpsampleTracker = /*#__PURE__*/function () {
|
|
1140
|
+
function UpsampleTracker() {
|
|
1141
|
+
_classCallCheck(this, UpsampleTracker);
|
|
1142
|
+
this.ne = false;
|
|
1143
|
+
this.nw = false;
|
|
1144
|
+
this.se = false;
|
|
1145
|
+
this.sw = false;
|
|
1146
|
+
}
|
|
1147
|
+
return _createClass(UpsampleTracker, [{
|
|
1148
|
+
key: "finished",
|
|
1149
|
+
value: function finished() {
|
|
1150
|
+
return this.ne && this.nw && this.se && this.sw;
|
|
1151
|
+
}
|
|
1152
|
+
}]);
|
|
1153
|
+
}();
|
|
1154
|
+
var RasterTerrainData = /*#__PURE__*/function (_QuantizedMeshTerrain) {
|
|
1155
|
+
function RasterTerrainData(opts) {
|
|
1156
|
+
var _opts$maxVertexDistan;
|
|
1157
|
+
var _this;
|
|
1158
|
+
_classCallCheck(this, RasterTerrainData);
|
|
1159
|
+
_this = _callSuper(this, RasterTerrainData, [opts]);
|
|
1160
|
+
_this.quantizedHeights = opts.quantizedHeights;
|
|
1161
|
+
_this.errorLevel = opts.errorLevel;
|
|
1162
|
+
_this.maxVertexDistance = (_opts$maxVertexDistan = opts.maxVertexDistance) !== null && _opts$maxVertexDistan !== void 0 ? _opts$maxVertexDistan : opts.tileSize;
|
|
1163
|
+
_this.tileSize = opts.tileSize;
|
|
1164
|
+
_this.upsampleTracker = new UpsampleTracker();
|
|
1165
|
+
return _this;
|
|
1166
|
+
}
|
|
1167
|
+
_inherits(RasterTerrainData, _QuantizedMeshTerrain);
|
|
1168
|
+
return _createClass(RasterTerrainData, [{
|
|
1169
|
+
key: "upsample",
|
|
1170
|
+
value: function upsample(tilingScheme, thisX, thisY, thisLevel, descendantX, descendantY, descendantLevel) {
|
|
1171
|
+
if (this.quantizedHeights == null) {
|
|
1172
|
+
return _superPropGet(RasterTerrainData, "upsample", this, 3)([tilingScheme, thisX, thisY, thisLevel, descendantX, descendantY, descendantLevel]);
|
|
1173
|
+
} // Something wonky about our tiling scheme, perhaps
|
|
1174
|
+
// 12/2215/2293 @2x
|
|
1175
|
+
//const url = `https://a.tiles.mapbox.com/v4/mapbox.terrain-rgb/${z}/${x}/${y}${hires}.${this.format}?access_token=${this.accessToken}`;
|
|
1176
|
+
|
|
1177
|
+
var x = descendantX;
|
|
1178
|
+
var y = descendantY;
|
|
1179
|
+
var z = descendantLevel;
|
|
1180
|
+
|
|
1181
|
+
//console.log(`Upsampling terrain data from zoom ${thisLevel} to ` + tile);
|
|
1182
|
+
|
|
1183
|
+
var dz = z - thisLevel;
|
|
1184
|
+
var scalar = Math.pow(2, dz);
|
|
1185
|
+
var ellipsoid = tilingScheme.ellipsoid;
|
|
1186
|
+
var err = this.errorLevel / scalar;
|
|
1187
|
+
var maxVertexDistance = Math.min(this.maxVertexDistance * scalar, this.tileSize);
|
|
1188
|
+
var upscaledX = thisX * scalar;
|
|
1189
|
+
var upscaledY = thisY * scalar;
|
|
1190
|
+
var dx = x - upscaledX;
|
|
1191
|
+
var dy = y - upscaledY;
|
|
1192
|
+
var x0 = dx * this.tileSize / scalar;
|
|
1193
|
+
var x1 = (dx + 1) * this.tileSize / scalar;
|
|
1194
|
+
var y0 = dy * this.tileSize / scalar;
|
|
1195
|
+
var y1 = (dy + 1) * this.tileSize / scalar;
|
|
1196
|
+
var window = {
|
|
1197
|
+
x0: x0,
|
|
1198
|
+
x1: x1,
|
|
1199
|
+
y0: y0,
|
|
1200
|
+
y1: y1
|
|
1201
|
+
};
|
|
1202
|
+
var res = buildOverscaledTerrainTile({
|
|
1203
|
+
tilingScheme: tilingScheme,
|
|
1204
|
+
heightData: subsetByWindow(this.quantizedHeights, window, true),
|
|
1205
|
+
maxVertexDistance: maxVertexDistance,
|
|
1206
|
+
x: x,
|
|
1207
|
+
y: y,
|
|
1208
|
+
z: z,
|
|
1209
|
+
errorLevel: err,
|
|
1210
|
+
ellipsoidRadius: ellipsoid.maximumRadius,
|
|
1211
|
+
tileSize: x1 - x0,
|
|
1212
|
+
overscaleFactor: dz
|
|
1213
|
+
});
|
|
1214
|
+
if (dz == 1) {
|
|
1215
|
+
// If we've got a single child tile, we can track that we've upsampled the parent.
|
|
1216
|
+
var quadrant = getQuadrant(dx, dy);
|
|
1217
|
+
this.upsampleTracker[quadrant] = true;
|
|
1218
|
+
}
|
|
1219
|
+
if (this.upsampleTracker.finished()) {
|
|
1220
|
+
// We've upsampled all child tiles and don't need to keep terrain data around anymore.
|
|
1221
|
+
this.quantizedHeights = undefined;
|
|
1222
|
+
}
|
|
1223
|
+
return res;
|
|
1224
|
+
}
|
|
1225
|
+
}]);
|
|
1226
|
+
}(cesium.QuantizedMeshTerrainData);
|
|
1227
|
+
function getQuadrant(dx, dy) {
|
|
1228
|
+
if (dx == 0 && dy == 0) return "sw";
|
|
1229
|
+
if (dx == 0 && dy == 1) return "nw";
|
|
1230
|
+
if (dx == 1 && dy == 0) return "se";
|
|
1231
|
+
if (dx == 1 && dy == 1) return "ne";
|
|
1232
|
+
throw new Error("Invalid quadrant");
|
|
1233
|
+
}
|
|
1234
|
+
function buildOverscaledTerrainTile(_x) {
|
|
1235
|
+
return _buildOverscaledTerrainTile.apply(this, arguments);
|
|
1236
|
+
}
|
|
1237
|
+
function _buildOverscaledTerrainTile() {
|
|
1238
|
+
_buildOverscaledTerrainTile = _asyncToGenerator(/*#__PURE__*/_regeneratorRuntime().mark(function _callee(opts) {
|
|
1239
|
+
var tilingScheme, overscaleFactor, workerOpts, x, y, z, tileRect, ellipsoid, errorLevel, maxVertexDistance, tileSize, res;
|
|
1240
|
+
return _regeneratorRuntime().wrap(function _callee$(_context) {
|
|
1241
|
+
while (1) switch (_context.prev = _context.next) {
|
|
1242
|
+
case 0:
|
|
1243
|
+
tilingScheme = opts.tilingScheme, overscaleFactor = opts.overscaleFactor, workerOpts = _objectWithoutProperties(opts, _excluded);
|
|
1244
|
+
x = workerOpts.x, y = workerOpts.y, z = workerOpts.z;
|
|
1245
|
+
tileRect = tilingScheme.tileXYToRectangle(x, y, z);
|
|
1246
|
+
ellipsoid = tilingScheme.ellipsoid;
|
|
1247
|
+
errorLevel = workerOpts.errorLevel, maxVertexDistance = workerOpts.maxVertexDistance, tileSize = workerOpts.tileSize;
|
|
1248
|
+
_context.prev = 5;
|
|
1249
|
+
_context.next = 8;
|
|
1250
|
+
return upsamplerFarm.scheduleTask(workerOpts, [workerOpts.heightData.buffer]);
|
|
1251
|
+
case 8:
|
|
1252
|
+
res = _context.sent;
|
|
1253
|
+
return _context.abrupt("return", createTerrainMesh(res, {
|
|
1254
|
+
tileRect: tileRect,
|
|
1255
|
+
ellipsoid: ellipsoid,
|
|
1256
|
+
errorLevel: errorLevel,
|
|
1257
|
+
overscaleFactor: overscaleFactor,
|
|
1258
|
+
tileSize: tileSize,
|
|
1259
|
+
// Maximum vertex distance
|
|
1260
|
+
maxVertexDistance: maxVertexDistance
|
|
1261
|
+
}));
|
|
1262
|
+
case 12:
|
|
1263
|
+
_context.prev = 12;
|
|
1264
|
+
_context.t0 = _context["catch"](5);
|
|
1265
|
+
return _context.abrupt("return", createEmptyMesh({
|
|
1266
|
+
tileRect: tileRect,
|
|
1267
|
+
errorLevel: errorLevel,
|
|
1268
|
+
ellipsoid: ellipsoid,
|
|
1269
|
+
tileCoord: {
|
|
1270
|
+
x: x,
|
|
1271
|
+
y: y,
|
|
1272
|
+
z: z
|
|
1273
|
+
},
|
|
1274
|
+
tileSize: 0
|
|
1275
|
+
}));
|
|
1276
|
+
case 15:
|
|
1277
|
+
case "end":
|
|
1278
|
+
return _context.stop();
|
|
1279
|
+
}
|
|
1280
|
+
}, _callee, null, [[5, 12]]);
|
|
1281
|
+
}));
|
|
1282
|
+
return _buildOverscaledTerrainTile.apply(this, arguments);
|
|
1283
|
+
}
|
|
1284
|
+
var upsamplerFarm = new WorkerFarm({
|
|
1285
|
+
worker: new WorkerFactory$1(),
|
|
1286
|
+
maxWorkers: 5
|
|
1287
|
+
});
|
|
1288
|
+
|
|
795
1289
|
// https://github.com/CesiumGS/cesium/blob/1.68/Source/Scene/MapboxImageryProvider.js#L42
|
|
1290
|
+
|
|
796
1291
|
var StretchedTilingScheme = /*#__PURE__*/function (_WebMercatorTilingSch) {
|
|
797
1292
|
function StretchedTilingScheme() {
|
|
798
1293
|
_classCallCheck(this, StretchedTilingScheme);
|
|
@@ -804,7 +1299,6 @@ var StretchedTilingScheme = /*#__PURE__*/function (_WebMercatorTilingSch) {
|
|
|
804
1299
|
value: function tileXYToRectangle(x, y, level, res) {
|
|
805
1300
|
var result = _superPropGet(StretchedTilingScheme, "tileXYToRectangle", this, 3)([x, y, level]);
|
|
806
1301
|
if (y == 0) {
|
|
807
|
-
//console.log("Top row", res, y, level);
|
|
808
1302
|
result.north = Math.PI / 2;
|
|
809
1303
|
}
|
|
810
1304
|
if (y + 1 == Math.pow(2, level)) {
|
|
@@ -817,7 +1311,7 @@ var StretchedTilingScheme = /*#__PURE__*/function (_WebMercatorTilingSch) {
|
|
|
817
1311
|
var MartiniTerrainProvider = /*#__PURE__*/function () {
|
|
818
1312
|
// @ts-ignore
|
|
819
1313
|
function MartiniTerrainProvider() {
|
|
820
|
-
var _this$resource$credit, _opts$minZoomLevel, _opts$fillPoles, _opts$detailScalar, _opts$
|
|
1314
|
+
var _this$resource$credit, _opts$maxWorkers, _opts$minZoomLevel, _opts$fillPoles, _opts$minimumErrorLev, _opts$ellipsoid, _opts$detailScalar, _ref, _opts$tilingScheme$el, _opts$tilingScheme;
|
|
821
1315
|
var opts = arguments.length > 0 && arguments[0] !== undefined ? arguments[0] : {};
|
|
822
1316
|
_classCallCheck(this, MartiniTerrainProvider);
|
|
823
1317
|
_defineProperty(this, "hasWaterMask", false);
|
|
@@ -826,6 +1320,7 @@ var MartiniTerrainProvider = /*#__PURE__*/function () {
|
|
|
826
1320
|
_defineProperty(this, "availability", null);
|
|
827
1321
|
_defineProperty(this, "errorEvent", new cesium.Event());
|
|
828
1322
|
_defineProperty(this, "levelOfDetailScalar", null);
|
|
1323
|
+
_defineProperty(this, "maxWorkers", 5);
|
|
829
1324
|
_defineProperty(this, "minError", 0.1);
|
|
830
1325
|
_defineProperty(this, "fillPoles", true);
|
|
831
1326
|
_defineProperty(this, "_errorAtMinZoom", 1000);
|
|
@@ -836,9 +1331,9 @@ var MartiniTerrainProvider = /*#__PURE__*/function () {
|
|
|
836
1331
|
this.resource = opts.resource;
|
|
837
1332
|
this.credit = (_this$resource$credit = this.resource.credit) !== null && _this$resource$credit !== void 0 ? _this$resource$credit : new cesium.Credit("Mapbox");
|
|
838
1333
|
this.decoder = opts.decoder;
|
|
1334
|
+
this.maxWorkers = (_opts$maxWorkers = opts.maxWorkers) !== null && _opts$maxWorkers !== void 0 ? _opts$maxWorkers : 5;
|
|
839
1335
|
if (!this.decoder) {
|
|
840
|
-
var
|
|
841
|
-
var maxWorkers = (_opts$maxWorkers = opts.maxWorkers) !== null && _opts$maxWorkers !== void 0 ? _opts$maxWorkers : 5;
|
|
1336
|
+
var maxWorkers = this.maxWorkers;
|
|
842
1337
|
if (maxWorkers > 0) {
|
|
843
1338
|
this.decoder = new WorkerFarmTerrainDecoder({
|
|
844
1339
|
maxWorkers: maxWorkers
|
|
@@ -849,32 +1344,40 @@ var MartiniTerrainProvider = /*#__PURE__*/function () {
|
|
|
849
1344
|
}
|
|
850
1345
|
this.minZoomLevel = (_opts$minZoomLevel = opts.minZoomLevel) !== null && _opts$minZoomLevel !== void 0 ? _opts$minZoomLevel : 3;
|
|
851
1346
|
this.fillPoles = (_opts$fillPoles = opts.fillPoles) !== null && _opts$fillPoles !== void 0 ? _opts$fillPoles : true;
|
|
852
|
-
this.levelOfDetailScalar = ((_opts$detailScalar = opts.detailScalar) !== null && _opts$detailScalar !== void 0 ? _opts$detailScalar : 4.0) + cesium.Math.EPSILON5;
|
|
853
1347
|
this.ready = true;
|
|
854
1348
|
this.readyPromise = Promise.resolve(true);
|
|
855
1349
|
this.minError = (_opts$minimumErrorLev = opts.minimumErrorLevel) !== null && _opts$minimumErrorLev !== void 0 ? _opts$minimumErrorLev : 0.1;
|
|
856
1350
|
this.errorEvent.addEventListener(console.log, this);
|
|
857
1351
|
this.ellipsoid = (_opts$ellipsoid = opts.ellipsoid) !== null && _opts$ellipsoid !== void 0 ? _opts$ellipsoid : cesium.Ellipsoid.WGS84;
|
|
858
|
-
|
|
859
|
-
|
|
860
|
-
|
|
1352
|
+
if (opts.tilingScheme == null) {
|
|
1353
|
+
var scheme = cesium.WebMercatorTilingScheme;
|
|
1354
|
+
if (this.fillPoles) {
|
|
1355
|
+
scheme = StretchedTilingScheme;
|
|
1356
|
+
}
|
|
1357
|
+
this.tilingScheme = new scheme({
|
|
1358
|
+
numberOfLevelZeroTilesX: 1,
|
|
1359
|
+
numberOfLevelZeroTilesY: 1,
|
|
1360
|
+
ellipsoid: this.ellipsoid
|
|
1361
|
+
});
|
|
1362
|
+
} else {
|
|
1363
|
+
this.tilingScheme = opts.tilingScheme;
|
|
861
1364
|
}
|
|
862
|
-
this.
|
|
863
|
-
|
|
864
|
-
|
|
865
|
-
|
|
866
|
-
});
|
|
1365
|
+
this.levelOfDetailScalar = ((_opts$detailScalar = opts.detailScalar) !== null && _opts$detailScalar !== void 0 ? _opts$detailScalar : 2.0) + cesium.Math.EPSILON5;
|
|
1366
|
+
|
|
1367
|
+
//this.errorEvent.addEventListener(console.log, this);
|
|
1368
|
+
this.ellipsoid = (_ref = (_opts$tilingScheme$el = (_opts$tilingScheme = opts.tilingScheme) === null || _opts$tilingScheme === void 0 ? void 0 : _opts$tilingScheme.ellipsoid) !== null && _opts$tilingScheme$el !== void 0 ? _opts$tilingScheme$el : opts.ellipsoid) !== null && _ref !== void 0 ? _ref : cesium.Ellipsoid.WGS84;
|
|
867
1369
|
this._errorAtMinZoom = this.errorAtZoom(this.minZoomLevel);
|
|
868
1370
|
}
|
|
869
1371
|
return _createClass(MartiniTerrainProvider, [{
|
|
870
1372
|
key: "requestTileGeometry",
|
|
871
1373
|
value: function requestTileGeometry(x, y, z, request) {
|
|
872
1374
|
// Look for tiles both below the zoom level and below the error threshold for the zoom level at the equator...
|
|
873
|
-
if (z < this.minZoomLevel || this.scaledErrorForTile(x, y, z) > this._errorAtMinZoom) {
|
|
1375
|
+
if (this.minZoomLevel != 0 && (z < this.minZoomLevel || this.scaledErrorForTile(x, y, z) > this._errorAtMinZoom)) {
|
|
874
1376
|
// If we are below the minimum zoom level, we return empty heightmaps
|
|
875
1377
|
// to avoid unnecessary requests for low-resolution data.
|
|
876
1378
|
return Promise.resolve(this.emptyMesh(x, y, z));
|
|
877
1379
|
}
|
|
1380
|
+
|
|
878
1381
|
// Note: we still load a TON of tiles near the poles. We might need to do some overzooming here...
|
|
879
1382
|
return this.decoder.requestTileGeometry({
|
|
880
1383
|
x: x,
|
|
@@ -885,53 +1388,88 @@ var MartiniTerrainProvider = /*#__PURE__*/function () {
|
|
|
885
1388
|
}, {
|
|
886
1389
|
key: "processTile",
|
|
887
1390
|
value: function () {
|
|
888
|
-
var _processTile = _asyncToGenerator(/*#__PURE__*/_regeneratorRuntime().mark(function _callee(
|
|
889
|
-
var x, y, z, _this$resource, tileSize, getTilePixels, px, pixelData,
|
|
1391
|
+
var _processTile = _asyncToGenerator(/*#__PURE__*/_regeneratorRuntime().mark(function _callee(_ref2) {
|
|
1392
|
+
var x, y, z, tileRect, errorLevel, maxVertexDistance, _this$resource, tileSize, getTilePixels, r1, px, pixelData, params, res, meta;
|
|
890
1393
|
return _regeneratorRuntime().wrap(function _callee$(_context) {
|
|
891
1394
|
while (1) switch (_context.prev = _context.next) {
|
|
892
1395
|
case 0:
|
|
893
|
-
x =
|
|
894
|
-
|
|
1396
|
+
x = _ref2.x, y = _ref2.y, z = _ref2.z;
|
|
1397
|
+
// Something wonky about our tiling scheme, perhaps
|
|
1398
|
+
// 12/2215/2293 @2x
|
|
1399
|
+
//const url = `https://a.tiles.mapbox.com/v4/mapbox.terrain-rgb/${z}/${x}/${y}${hires}.${this.format}?access_token=${this.accessToken}`;
|
|
1400
|
+
tileRect = this.tilingScheme.tileXYToRectangle(x, y, z);
|
|
1401
|
+
errorLevel = this.errorAtZoom(z);
|
|
1402
|
+
maxVertexDistance = this.maxVertexDistance(tileRect);
|
|
1403
|
+
_context.prev = 4;
|
|
895
1404
|
_this$resource = this.resource, tileSize = _this$resource.tileSize, getTilePixels = _this$resource.getTilePixels;
|
|
896
|
-
|
|
897
|
-
return getTilePixels({
|
|
1405
|
+
r1 = getTilePixels.bind(this.resource, {
|
|
898
1406
|
x: x,
|
|
899
1407
|
y: y,
|
|
900
1408
|
z: z
|
|
901
1409
|
});
|
|
902
|
-
|
|
1410
|
+
if (!(r1 == null)) {
|
|
1411
|
+
_context.next = 9;
|
|
1412
|
+
break;
|
|
1413
|
+
}
|
|
1414
|
+
return _context.abrupt("return");
|
|
1415
|
+
case 9:
|
|
1416
|
+
_context.next = 11;
|
|
1417
|
+
return r1();
|
|
1418
|
+
case 11:
|
|
903
1419
|
px = _context.sent;
|
|
904
1420
|
pixelData = px.data;
|
|
905
|
-
|
|
906
|
-
|
|
907
|
-
|
|
1421
|
+
if (!(pixelData == null)) {
|
|
1422
|
+
_context.next = 15;
|
|
1423
|
+
break;
|
|
1424
|
+
}
|
|
1425
|
+
return _context.abrupt("return");
|
|
1426
|
+
case 15:
|
|
1427
|
+
///const center = Rectangle.center(tileRect);
|
|
908
1428
|
params = {
|
|
909
1429
|
imageData: pixelData,
|
|
910
|
-
|
|
1430
|
+
maxVertexDistance: maxVertexDistance,
|
|
911
1431
|
x: x,
|
|
912
1432
|
y: y,
|
|
913
1433
|
z: z,
|
|
914
|
-
errorLevel:
|
|
915
|
-
ellipsoidRadius: this.ellipsoid.maximumRadius,
|
|
1434
|
+
errorLevel: errorLevel,
|
|
1435
|
+
ellipsoidRadius: this.tilingScheme.ellipsoid.maximumRadius,
|
|
916
1436
|
tileSize: tileSize
|
|
917
1437
|
};
|
|
918
|
-
_context.next =
|
|
1438
|
+
_context.next = 18;
|
|
919
1439
|
return this.decoder.decodeTerrain(params, pixelData.buffer);
|
|
920
|
-
case
|
|
1440
|
+
case 18:
|
|
921
1441
|
res = _context.sent;
|
|
922
|
-
|
|
923
|
-
|
|
924
|
-
|
|
925
|
-
|
|
926
|
-
|
|
927
|
-
|
|
928
|
-
|
|
929
|
-
|
|
1442
|
+
meta = {
|
|
1443
|
+
ellipsoid: this.tilingScheme.ellipsoid,
|
|
1444
|
+
errorLevel: errorLevel,
|
|
1445
|
+
overscaleFactor: 0,
|
|
1446
|
+
maxVertexDistance: maxVertexDistance,
|
|
1447
|
+
tileRect: tileRect,
|
|
1448
|
+
tileSize: tileSize
|
|
1449
|
+
};
|
|
1450
|
+
/** This builds a final terrain mesh object that can optionally
|
|
1451
|
+
* be upscaled to a higher resolution.
|
|
1452
|
+
*/
|
|
1453
|
+
return _context.abrupt("return", createTerrainMesh(res, meta));
|
|
930
1454
|
case 23:
|
|
1455
|
+
_context.prev = 23;
|
|
1456
|
+
_context.t0 = _context["catch"](4);
|
|
1457
|
+
return _context.abrupt("return", createEmptyMesh({
|
|
1458
|
+
tileRect: tileRect,
|
|
1459
|
+
errorLevel: errorLevel,
|
|
1460
|
+
ellipsoid: this.tilingScheme.ellipsoid,
|
|
1461
|
+
tileCoord: {
|
|
1462
|
+
x: x,
|
|
1463
|
+
y: y,
|
|
1464
|
+
z: z
|
|
1465
|
+
},
|
|
1466
|
+
tileSize: 0
|
|
1467
|
+
}));
|
|
1468
|
+
case 26:
|
|
931
1469
|
case "end":
|
|
932
1470
|
return _context.stop();
|
|
933
1471
|
}
|
|
934
|
-
}, _callee, this, [[
|
|
1472
|
+
}, _callee, this, [[4, 23]]);
|
|
935
1473
|
}));
|
|
936
1474
|
function processTile(_x) {
|
|
937
1475
|
return _processTile.apply(this, arguments);
|
|
@@ -989,9 +1527,8 @@ var MartiniTerrainProvider = /*#__PURE__*/function () {
|
|
|
989
1527
|
if (halfAngle > Math.PI / 4) {
|
|
990
1528
|
occlusionHeight = (1 + halfAngle) * this.ellipsoid.maximumRadius;
|
|
991
1529
|
}
|
|
992
|
-
var occlusionPoint = new cesium.Cartographic(center.longitude, center.latitude, occlusionHeight
|
|
993
|
-
|
|
994
|
-
);
|
|
1530
|
+
var occlusionPoint = new cesium.Cartographic(center.longitude, center.latitude, occlusionHeight) // Scaling factor of two just to be sure.
|
|
1531
|
+
;
|
|
995
1532
|
var horizonOcclusionPoint = this.ellipsoid.transformPositionToScaledSpace(cesium.Cartographic.toCartesian(occlusionPoint));
|
|
996
1533
|
var orientedBoundingBox = cesium.OrientedBoundingBox.fromRectangle(tileRect, minimumHeight, maximumHeight, this.tilingScheme.ellipsoid);
|
|
997
1534
|
var boundingSphere = cesium.BoundingSphere.fromOrientedBoundingBox(orientedBoundingBox);
|
|
@@ -999,7 +1536,8 @@ var MartiniTerrainProvider = /*#__PURE__*/function () {
|
|
|
999
1536
|
// SE NW NE
|
|
1000
1537
|
// NE NW SE
|
|
1001
1538
|
|
|
1002
|
-
|
|
1539
|
+
/** TODO: we need to create raster terrain data. */
|
|
1540
|
+
return new cesium.QuantizedMeshTerrainData({
|
|
1003
1541
|
minimumHeight: minimumHeight,
|
|
1004
1542
|
maximumHeight: maximumHeight,
|
|
1005
1543
|
quantizedVertices: quantizedVertices,
|
|
@@ -1017,16 +1555,21 @@ var MartiniTerrainProvider = /*#__PURE__*/function () {
|
|
|
1017
1555
|
northSkirtHeight: skirtHeight,
|
|
1018
1556
|
childTileMask: 15
|
|
1019
1557
|
});
|
|
1020
|
-
return result;
|
|
1021
1558
|
}
|
|
1022
1559
|
}, {
|
|
1023
1560
|
key: "getLevelMaximumGeometricError",
|
|
1024
1561
|
value: function getLevelMaximumGeometricError(level) {
|
|
1025
1562
|
var levelZeroMaximumGeometricError = cesium.TerrainProvider.getEstimatedLevelZeroGeometricErrorForAHeightmap(this.tilingScheme.ellipsoid, 65, this.tilingScheme.getNumberOfXTilesAtLevel(0));
|
|
1026
1563
|
|
|
1027
|
-
|
|
1028
|
-
|
|
1029
|
-
|
|
1564
|
+
/*
|
|
1565
|
+
Scalar to control overzooming
|
|
1566
|
+
- also seems to control zooming for imagery layers
|
|
1567
|
+
- This scalar was causing trouble for non-256 tile sizes,
|
|
1568
|
+
and we've removed it for now. It could be reintroduced
|
|
1569
|
+
if it seems necessary
|
|
1570
|
+
*/
|
|
1571
|
+
var scalar = 1; //this.resource.tileSize / 256 ;
|
|
1572
|
+
|
|
1030
1573
|
return levelZeroMaximumGeometricError / scalar / (1 << level);
|
|
1031
1574
|
}
|
|
1032
1575
|
}, {
|
|
@@ -1041,38 +1584,7 @@ var MartiniTerrainProvider = /*#__PURE__*/function () {
|
|
|
1041
1584
|
}]);
|
|
1042
1585
|
}();
|
|
1043
1586
|
|
|
1044
|
-
function decodeBase64(base64, enableUnicode) {
|
|
1045
|
-
var binaryString = atob(base64);
|
|
1046
|
-
if (enableUnicode) {
|
|
1047
|
-
var binaryView = new Uint8Array(binaryString.length);
|
|
1048
|
-
for (var i = 0, n = binaryString.length; i < n; ++i) {
|
|
1049
|
-
binaryView[i] = binaryString.charCodeAt(i);
|
|
1050
|
-
}
|
|
1051
|
-
const decoder = new TextDecoder("utf-16le");
|
|
1052
|
-
return decoder.decode(new Uint16Array(binaryView.buffer));
|
|
1053
|
-
}
|
|
1054
|
-
return binaryString;
|
|
1055
|
-
}
|
|
1056
|
-
|
|
1057
|
-
function createURL(base64, sourcemapArg, enableUnicodeArg) {
|
|
1058
|
-
var sourcemap = sourcemapArg === undefined ? null : sourcemapArg;
|
|
1059
|
-
var enableUnicode = enableUnicodeArg === undefined ? false : enableUnicodeArg;
|
|
1060
|
-
var source = decodeBase64(base64, enableUnicode);
|
|
1061
|
-
var start = source.indexOf('\n', 10) + 1;
|
|
1062
|
-
var body = source.substring(start) + (sourcemap ? '\/\/# sourceMappingURL=' + sourcemap : '');
|
|
1063
|
-
var blob = new Blob([body], { type: 'application/javascript' });
|
|
1064
|
-
return URL.createObjectURL(blob);
|
|
1065
|
-
}
|
|
1066
|
-
|
|
1067
|
-
function createBase64WorkerFactory(base64, sourcemapArg, enableUnicodeArg) {
|
|
1068
|
-
var url;
|
|
1069
|
-
return function WorkerFactory(options) {
|
|
1070
|
-
url = url || createURL(base64, sourcemapArg, enableUnicodeArg);
|
|
1071
|
-
return new Worker(url, options);
|
|
1072
|
-
};
|
|
1073
|
-
}
|
|
1074
|
-
|
|
1075
|
-
var WorkerFactory = /*#__PURE__*/createBase64WorkerFactory('/* rollup-plugin-web-worker-loader */
(function () {
  'use strict';

  function _arrayLikeToArray(r, a) {
    (null == a || a > r.length) && (a = r.length);
    for (var e = 0, n = Array(a); e < a; e++) n[e] = r[e];
    return n;
  }
  function _arrayWithoutHoles(r) {
    if (Array.isArray(r)) return _arrayLikeToArray(r);
  }
  function _iterableToArray(r) {
    if ("undefined" != typeof Symbol && null != r[Symbol.iterator] || null != r["@@iterator"]) return Array.from(r);
  }
  function _nonIterableSpread() {
    throw new TypeError("Invalid attempt to spread non-iterable instance.\nIn order to be iterable, non-array objects must have a [Symbol.iterator]() method.");
  }
  function _toConsumableArray(r) {
    return _arrayWithoutHoles(r) || _iterableToArray(r) || _unsupportedIterableToArray(r) || _nonIterableSpread();
  }
  function _unsupportedIterableToArray(r, a) {
    if (r) {
      if ("string" == typeof r) return _arrayLikeToArray(r, a);
      var t = {}.toString.call(r).slice(8, -1);
      return "Object" === t && r.constructor && (t = r.constructor.name), "Map" === t || "Set" === t ? Array.from(r) : "Arguments" === t || /^(?:Ui|I)nt(?:8|16|32)(?:Clamped)?Array$/.test(t) ? _arrayLikeToArray(r, a) : void 0;
    }
  }

  // We should save these
  //const canvas = new OffscreenCanvas(256, 256);
  //const ctx = canvas.getContext("2d");

  /** Mapbox Terrain-RGB default decode function
  *  (r * 256 * 256) / 10 + (g * 256) / 10 + b / 10 - 10000
  */
  var defaultMapboxDecodeRgb = function defaultMapboxDecodeRgb(r, g, b, a) {
    return r * 6553.6 + g * 25.6 + b * 0.1 - 10000;
  };
  function rgbTerrainToGrid(png, decodeRgb) {
    // maybe we should do this on the GPU using REGL?
    // but that would require GPU -> CPU -> GPU
    var gridSize = png.shape[0] + 1;
    var terrain = new Float32Array(gridSize * gridSize);
    var tileSize = png.shape[0];
    var decode = decodeRgb !== null && decodeRgb !== void 0 ? decodeRgb : defaultMapboxDecodeRgb;

    // decode terrain values
    for (var y = 0; y < tileSize; y++) {
      for (var x = 0; x < tileSize; x++) {
        var yc = y;
        var _r = png.get(x, yc, 0);
        var _g = png.get(x, yc, 1);
        var _b = png.get(x, yc, 2);
        var _a = png.get(x, yc, 3);
        terrain[y * gridSize + x] = decode(_r, _g, _b, _a);
      }
    }
    // backfill right and bottom borders
    for (var _x = 0; _x < gridSize - 1; _x++) {
      terrain[gridSize * (gridSize - 1) + _x] = terrain[gridSize * (gridSize - 2) + _x];
    }
    for (var _y = 0; _y < gridSize; _y++) {
      terrain[gridSize * _y + gridSize - 1] = terrain[gridSize * _y + gridSize - 2];
    }
    return terrain;
  }
  function createQuantizedMeshData(tile, mesh, tileSize) {
    var xvals = [];
    var yvals = [];
    var heightMeters = [];
    var northIndices = [];
    var southIndices = [];
    var eastIndices = [];
    var westIndices = [];
    var minimumHeight = Infinity;
    var maximumHeight = -Infinity;
    var scalar = 32768.0 / tileSize;
    for (var ix = 0; ix < mesh.vertices.length / 2; ix++) {
      var vertexIx = ix;
      var px = mesh.vertices[ix * 2];
      var py = mesh.vertices[ix * 2 + 1];
      var height = tile.terrain[py * (tileSize + 1) + px];
      if (height > maximumHeight) maximumHeight = height;
      if (height < minimumHeight) minimumHeight = height;
      heightMeters.push(height);
      if (py == 0) northIndices.push(vertexIx);
      if (py == tileSize) southIndices.push(vertexIx);
      if (px == 0) westIndices.push(vertexIx);
      if (px == tileSize) eastIndices.push(vertexIx);
      var xv = px * scalar;
      var yv = (tileSize - py) * scalar;
      xvals.push(xv);
      yvals.push(yv);
    }
    var heightRange = maximumHeight - minimumHeight;
    var heights = heightMeters.map(function (d) {
      if (heightRange < 1) return 0;
      return (d - minimumHeight) * (32767.0 / heightRange);
    });
    var triangles = new Uint16Array(mesh.triangles);
    var quantizedVertices = new Uint16Array(//verts
    [].concat(xvals, yvals, _toConsumableArray(heights)));

    // SE NW NE
    // NE NW SE

    return {
      minimumHeight: minimumHeight,
      maximumHeight: maximumHeight,
      quantizedVertices: quantizedVertices,
      indices: triangles,
      westIndices: westIndices,
      southIndices: southIndices,
      eastIndices: eastIndices,
      northIndices: northIndices
    };
  }

  function iota(n) {
    var result = new Array(n);
    for (var i = 0; i < n; ++i) {
      result[i] = i;
    }
    return result;
  }
  var iota_1 = iota;

  /*!
   * Determine if an object is a Buffer
   *
   * @author   Feross Aboukhadijeh <https://feross.org>
   * @license  MIT
   */
  // The _isBuffer check is for Safari 5-7 support, because it's missing
  // Object.prototype.constructor. Remove this eventually
  var isBuffer_1 = function (obj) {
    return obj != null && (isBuffer(obj) || isSlowBuffer(obj) || !!obj._isBuffer);
  };
  function isBuffer(obj) {
    return !!obj.constructor && typeof obj.constructor.isBuffer === 'function' && obj.constructor.isBuffer(obj);
  }

  // For Node v0.10 support. Remove this eventually.
  function isSlowBuffer(obj) {
    return typeof obj.readFloatLE === 'function' && typeof obj.slice === 'function' && isBuffer(obj.slice(0, 0));
  }

  var hasTypedArrays = typeof Float64Array !== "undefined";
  function compare1st(a, b) {
    return a[0] - b[0];
  }
  function order() {
    var stride = this.stride;
    var terms = new Array(stride.length);
    var i;
    for (i = 0; i < terms.length; ++i) {
      terms[i] = [Math.abs(stride[i]), i];
    }
    terms.sort(compare1st);
    var result = new Array(terms.length);
    for (i = 0; i < result.length; ++i) {
      result[i] = terms[i][1];
    }
    return result;
  }
  function compileConstructor(dtype, dimension) {
    var className = ["View", dimension, "d", dtype].join("");
    if (dimension < 0) {
      className = "View_Nil" + dtype;
    }
    var useGetters = dtype === "generic";
    if (dimension === -1) {
      //Special case for trivial arrays
      var code = "function " + className + "(a){this.data=a;};\
var proto=" + className + ".prototype;\
proto.dtype='" + dtype + "';\
proto.index=function(){return -1};\
proto.size=0;\
proto.dimension=-1;\
proto.shape=proto.stride=proto.order=[];\
proto.lo=proto.hi=proto.transpose=proto.step=\
function(){return new " + className + "(this.data);};\
proto.get=proto.set=function(){};\
proto.pick=function(){return null};\
return function construct_" + className + "(a){return new " + className + "(a);}";
      var procedure = new Function(code);
      return procedure();
    } else if (dimension === 0) {
      //Special case for 0d arrays
      var code = "function " + className + "(a,d) {\
this.data = a;\
this.offset = d\
};\
var proto=" + className + ".prototype;\
proto.dtype='" + dtype + "';\
proto.index=function(){return this.offset};\
proto.dimension=0;\
proto.size=1;\
proto.shape=\
proto.stride=\
proto.order=[];\
proto.lo=\
proto.hi=\
proto.transpose=\
proto.step=function " + className + "_copy() {\
return new " + className + "(this.data,this.offset)\
};\
proto.pick=function " + className + "_pick(){\
return TrivialArray(this.data);\
};\
proto.valueOf=proto.get=function " + className + "_get(){\
return " + (useGetters ? "this.data.get(this.offset)" : "this.data[this.offset]") + "};\
proto.set=function " + className + "_set(v){\
return " + (useGetters ? "this.data.set(this.offset,v)" : "this.data[this.offset]=v") + "\
};\
return function construct_" + className + "(a,b,c,d){return new " + className + "(a,d)}";
      var procedure = new Function("TrivialArray", code);
      return procedure(CACHED_CONSTRUCTORS[dtype][0]);
    }
    var code = ["'use strict'"];

    //Create constructor for view
    var indices = iota_1(dimension);
    var args = indices.map(function (i) {
      return "i" + i;
    });
    var index_str = "this.offset+" + indices.map(function (i) {
      return "this.stride[" + i + "]*i" + i;
    }).join("+");
    var shapeArg = indices.map(function (i) {
      return "b" + i;
    }).join(",");
    var strideArg = indices.map(function (i) {
      return "c" + i;
    }).join(",");
    code.push("function " + className + "(a," + shapeArg + "," + strideArg + ",d){this.data=a", "this.shape=[" + shapeArg + "]", "this.stride=[" + strideArg + "]", "this.offset=d|0}", "var proto=" + className + ".prototype", "proto.dtype='" + dtype + "'", "proto.dimension=" + dimension);

    //view.size:
    code.push("Object.defineProperty(proto,'size',{get:function " + className + "_size(){\
return " + indices.map(function (i) {
      return "this.shape[" + i + "]";
    }).join("*"), "}})");

    //view.order:
    if (dimension === 1) {
      code.push("proto.order=[0]");
    } else {
      code.push("Object.defineProperty(proto,'order',{get:");
      if (dimension < 4) {
        code.push("function " + className + "_order(){");
        if (dimension === 2) {
          code.push("return (Math.abs(this.stride[0])>Math.abs(this.stride[1]))?[1,0]:[0,1]}})");
        } else if (dimension === 3) {
          code.push("var s0=Math.abs(this.stride[0]),s1=Math.abs(this.stride[1]),s2=Math.abs(this.stride[2]);\
if(s0>s1){\
if(s1>s2){\
return [2,1,0];\
}else if(s0>s2){\
return [1,2,0];\
}else{\
return [1,0,2];\
}\
}else if(s0>s2){\
return [2,0,1];\
}else if(s2>s1){\
return [0,1,2];\
}else{\
return [0,2,1];\
}}})");
        }
      } else {
        code.push("ORDER})");
      }
    }

    //view.set(i0, ..., v):
    code.push("proto.set=function " + className + "_set(" + args.join(",") + ",v){");
    if (useGetters) {
      code.push("return this.data.set(" + index_str + ",v)}");
    } else {
      code.push("return this.data[" + index_str + "]=v}");
    }

    //view.get(i0, ...):
    code.push("proto.get=function " + className + "_get(" + args.join(",") + "){");
    if (useGetters) {
      code.push("return this.data.get(" + index_str + ")}");
    } else {
      code.push("return this.data[" + index_str + "]}");
    }

    //view.index:
    code.push("proto.index=function " + className + "_index(", args.join(), "){return " + index_str + "}");

    //view.hi():
    code.push("proto.hi=function " + className + "_hi(" + args.join(",") + "){return new " + className + "(this.data," + indices.map(function (i) {
      return ["(typeof i", i, "!=='number'||i", i, "<0)?this.shape[", i, "]:i", i, "|0"].join("");
    }).join(",") + "," + indices.map(function (i) {
      return "this.stride[" + i + "]";
    }).join(",") + ",this.offset)}");

    //view.lo():
    var a_vars = indices.map(function (i) {
      return "a" + i + "=this.shape[" + i + "]";
    });
    var c_vars = indices.map(function (i) {
      return "c" + i + "=this.stride[" + i + "]";
    });
    code.push("proto.lo=function " + className + "_lo(" + args.join(",") + "){var b=this.offset,d=0," + a_vars.join(",") + "," + c_vars.join(","));
    for (var i = 0; i < dimension; ++i) {
      code.push("if(typeof i" + i + "==='number'&&i" + i + ">=0){\
d=i" + i + "|0;\
b+=c" + i + "*d;\
a" + i + "-=d}");
    }
    code.push("return new " + className + "(this.data," + indices.map(function (i) {
      return "a" + i;
    }).join(",") + "," + indices.map(function (i) {
      return "c" + i;
    }).join(",") + ",b)}");

    //view.step():
    code.push("proto.step=function " + className + "_step(" + args.join(",") + "){var " + indices.map(function (i) {
      return "a" + i + "=this.shape[" + i + "]";
    }).join(",") + "," + indices.map(function (i) {
      return "b" + i + "=this.stride[" + i + "]";
    }).join(",") + ",c=this.offset,d=0,ceil=Math.ceil");
    for (var i = 0; i < dimension; ++i) {
      code.push("if(typeof i" + i + "==='number'){\
d=i" + i + "|0;\
if(d<0){\
c+=b" + i + "*(a" + i + "-1);\
a" + i + "=ceil(-a" + i + "/d)\
}else{\
a" + i + "=ceil(a" + i + "/d)\
}\
b" + i + "*=d\
}");
    }
    code.push("return new " + className + "(this.data," + indices.map(function (i) {
      return "a" + i;
    }).join(",") + "," + indices.map(function (i) {
      return "b" + i;
    }).join(",") + ",c)}");

    //view.transpose():
    var tShape = new Array(dimension);
    var tStride = new Array(dimension);
    for (var i = 0; i < dimension; ++i) {
      tShape[i] = "a[i" + i + "]";
      tStride[i] = "b[i" + i + "]";
    }
    code.push("proto.transpose=function " + className + "_transpose(" + args + "){" + args.map(function (n, idx) {
      return n + "=(" + n + "===undefined?" + idx + ":" + n + "|0)";
    }).join(";"), "var a=this.shape,b=this.stride;return new " + className + "(this.data," + tShape.join(",") + "," + tStride.join(",") + ",this.offset)}");

    //view.pick():
    code.push("proto.pick=function " + className + "_pick(" + args + "){var a=[],b=[],c=this.offset");
    for (var i = 0; i < dimension; ++i) {
      code.push("if(typeof i" + i + "==='number'&&i" + i + ">=0){c=(c+this.stride[" + i + "]*i" + i + ")|0}else{a.push(this.shape[" + i + "]);b.push(this.stride[" + i + "])}");
    }
    code.push("var ctor=CTOR_LIST[a.length+1];return ctor(this.data,a,b,c)}");

    //Add return statement
    code.push("return function construct_" + className + "(data,shape,stride,offset){return new " + className + "(data," + indices.map(function (i) {
      return "shape[" + i + "]";
    }).join(",") + "," + indices.map(function (i) {
      return "stride[" + i + "]";
    }).join(",") + ",offset)}");

    //Compile procedure
    var procedure = new Function("CTOR_LIST", "ORDER", code.join("\n"));
    return procedure(CACHED_CONSTRUCTORS[dtype], order);
  }
  function arrayDType(data) {
    if (isBuffer_1(data)) {
      return "buffer";
    }
    if (hasTypedArrays) {
      switch (Object.prototype.toString.call(data)) {
        case "[object Float64Array]":
          return "float64";
        case "[object Float32Array]":
          return "float32";
        case "[object Int8Array]":
          return "int8";
        case "[object Int16Array]":
          return "int16";
        case "[object Int32Array]":
          return "int32";
        case "[object Uint8Array]":
          return "uint8";
        case "[object Uint16Array]":
          return "uint16";
        case "[object Uint32Array]":
          return "uint32";
        case "[object Uint8ClampedArray]":
          return "uint8_clamped";
        case "[object BigInt64Array]":
          return "bigint64";
        case "[object BigUint64Array]":
          return "biguint64";
      }
    }
    if (Array.isArray(data)) {
      return "array";
    }
    return "generic";
  }
  var CACHED_CONSTRUCTORS = {
    "float32": [],
    "float64": [],
    "int8": [],
    "int16": [],
    "int32": [],
    "uint8": [],
    "uint16": [],
    "uint32": [],
    "array": [],
    "uint8_clamped": [],
    "bigint64": [],
    "biguint64": [],
    "buffer": [],
    "generic": []
  };
  function wrappedNDArrayCtor(data, shape, stride, offset) {
    if (data === undefined) {
      var ctor = CACHED_CONSTRUCTORS.array[0];
      return ctor([]);
    } else if (typeof data === "number") {
      data = [data];
    }
    if (shape === undefined) {
      shape = [data.length];
    }
    var d = shape.length;
    if (stride === undefined) {
      stride = new Array(d);
      for (var i = d - 1, sz = 1; i >= 0; --i) {
        stride[i] = sz;
        sz *= shape[i];
      }
    }
    if (offset === undefined) {
      offset = 0;
      for (var i = 0; i < d; ++i) {
        if (stride[i] < 0) {
          offset -= (shape[i] - 1) * stride[i];
        }
      }
    }
    var dtype = arrayDType(data);
    var ctor_list = CACHED_CONSTRUCTORS[dtype];
    while (ctor_list.length <= d + 1) {
      ctor_list.push(compileConstructor(dtype, ctor_list.length - 1));
    }
    var ctor = ctor_list[d + 1];
    return ctor(data, shape, stride, offset);
  }
  var ndarray = wrappedNDArrayCtor;

  class Martini {
    constructor(gridSize = 257) {
      this.gridSize = gridSize;
      const tileSize = gridSize - 1;
      if (tileSize & tileSize - 1) throw new Error(`Expected grid size to be 2^n+1, got ${gridSize}.`);
      this.numTriangles = tileSize * tileSize * 2 - 2;
      this.numParentTriangles = this.numTriangles - tileSize * tileSize;
      this.indices = new Uint32Array(this.gridSize * this.gridSize);

      // coordinates for all possible triangles in an RTIN tile
      this.coords = new Uint16Array(this.numTriangles * 4);

      // get triangle coordinates from its index in an implicit binary tree
      for (let i = 0; i < this.numTriangles; i++) {
        let id = i + 2;
        let ax = 0,
          ay = 0,
          bx = 0,
          by = 0,
          cx = 0,
          cy = 0;
        if (id & 1) {
          bx = by = cx = tileSize; // bottom-left triangle
        } else {
          ax = ay = cy = tileSize; // top-right triangle
        }
        while ((id >>= 1) > 1) {
          const mx = ax + bx >> 1;
          const my = ay + by >> 1;
          if (id & 1) {
            // left half
            bx = ax;
            by = ay;
            ax = cx;
            ay = cy;
          } else {
            // right half
            ax = bx;
            ay = by;
            bx = cx;
            by = cy;
          }
          cx = mx;
          cy = my;
        }
        const k = i * 4;
        this.coords[k + 0] = ax;
        this.coords[k + 1] = ay;
        this.coords[k + 2] = bx;
        this.coords[k + 3] = by;
      }
    }
    createTile(terrain) {
      return new Tile(terrain, this);
    }
  }
  class Tile {
    constructor(terrain, martini) {
      const size = martini.gridSize;
      if (terrain.length !== size * size) throw new Error(`Expected terrain data of length ${size * size} (${size} x ${size}), got ${terrain.length}.`);
      this.terrain = terrain;
      this.martini = martini;
      this.errors = new Float32Array(terrain.length);
      this.update();
    }
    update() {
      const {
        numTriangles,
        numParentTriangles,
        coords,
        gridSize: size
      } = this.martini;
      const {
        terrain,
        errors
      } = this;

      // iterate over all possible triangles, starting from the smallest level
      for (let i = numTriangles - 1; i >= 0; i--) {
        const k = i * 4;
        const ax = coords[k + 0];
        const ay = coords[k + 1];
        const bx = coords[k + 2];
        const by = coords[k + 3];
        const mx = ax + bx >> 1;
        const my = ay + by >> 1;
        const cx = mx + my - ay;
        const cy = my + ax - mx;

        // calculate error in the middle of the long edge of the triangle
        const interpolatedHeight = (terrain[ay * size + ax] + terrain[by * size + bx]) / 2;
        const middleIndex = my * size + mx;
        const middleError = Math.abs(interpolatedHeight - terrain[middleIndex]);
        errors[middleIndex] = Math.max(errors[middleIndex], middleError);
        if (i < numParentTriangles) {
          // bigger triangles; accumulate error with children
          const leftChildIndex = (ay + cy >> 1) * size + (ax + cx >> 1);
          const rightChildIndex = (by + cy >> 1) * size + (bx + cx >> 1);
          errors[middleIndex] = Math.max(errors[middleIndex], errors[leftChildIndex], errors[rightChildIndex]);
        }
      }
    }
    getMesh(maxError = 0, maxLength = null) {
      const {
        gridSize: size,
        indices
      } = this.martini;
      const {
        errors
      } = this;
      let numVertices = 0;
      let numTriangles = 0;
      const max = size - 1;

      // The maxLength parameter will cause triangles to be generated until the legs are below this length
      // It is meant to support cases where a certain mesh density is required to do spherical math on digital globes
      const maxScale = maxLength || size;

      // use an index grid to keep track of vertices that were already used to avoid duplication
      indices.fill(0);

      // retrieve mesh in two stages that both traverse the error map:
      // - countElements: find used vertices (and assign each an index), and count triangles (for minimum allocation)
      // - processTriangle: fill the allocated vertices & triangles typed arrays

      function countElements(ax, ay, bx, by, cx, cy) {
        const mx = ax + bx >> 1;
        const my = ay + by >> 1;
        const legLength = Math.abs(ax - cx) + Math.abs(ay - cy);
        if (legLength > 1 && errors[my * size + mx] > maxError || legLength > maxScale) {
          countElements(cx, cy, ax, ay, mx, my);
          countElements(bx, by, cx, cy, mx, my);
        } else {
          indices[ay * size + ax] = indices[ay * size + ax] || ++numVertices;
          indices[by * size + bx] = indices[by * size + bx] || ++numVertices;
          indices[cy * size + cx] = indices[cy * size + cx] || ++numVertices;
          numTriangles++;
        }
      }
      countElements(0, 0, max, max, max, 0);
      countElements(max, max, 0, 0, 0, max);
      const vertices = new Uint16Array(numVertices * 2);
      const triangles = new Uint32Array(numTriangles * 3);
      let triIndex = 0;
      function processTriangle(ax, ay, bx, by, cx, cy) {
        const mx = ax + bx >> 1;
        const my = ay + by >> 1;
        const legLength = Math.abs(ax - cx) + Math.abs(ay - cy);
        if (legLength > 1 && errors[my * size + mx] > maxError || legLength > maxScale) {
          // triangle doesn't approximate the surface well enough; drill down further
          processTriangle(cx, cy, ax, ay, mx, my);
          processTriangle(bx, by, cx, cy, mx, my);
        } else {
          // add a triangle
          const a = indices[ay * size + ax] - 1;
          const b = indices[by * size + bx] - 1;
          const c = indices[cy * size + cx] - 1;
          vertices[2 * a] = ax;
          vertices[2 * a + 1] = ay;
          vertices[2 * b] = bx;
          vertices[2 * b + 1] = by;
          vertices[2 * c] = cx;
          vertices[2 * c + 1] = cy;
          triangles[triIndex++] = a;
          triangles[triIndex++] = b;
          triangles[triIndex++] = c;
        }
      }
      processTriangle(0, 0, max, max, max, 0);
      processTriangle(max, max, 0, 0, 0, max);
      return {
        vertices,
        triangles
      };
    }
  }

  // https://github.com/CesiumGS/cesium/blob/1.76/Source/WorkersES6/createVerticesFromQuantizedTerrainMesh.js

  var martini = null;
  function decodeTerrain(parameters, transferableObjects) {
    var imageData = parameters.imageData,
      _parameters$tileSize = parameters.tileSize,
      tileSize = _parameters$tileSize === void 0 ? 256 : _parameters$tileSize,
      errorLevel = parameters.errorLevel;
    var pixels = ndarray(new Uint8Array(imageData), [tileSize, tileSize, 4], [4, 4 * tileSize, 1], 0);

    // Tile size must be maintained through the life of the worker
    martini !== null && martini !== void 0 ? martini : martini = new Martini(tileSize + 1);
    var terrain = rgbTerrainToGrid(pixels);
    var tile = martini.createTile(terrain);

    // get a mesh (vertices and triangles indices) for a 10m error
    var mesh = tile.getMesh(errorLevel, parameters.maxLength);
    return createQuantizedMeshData(tile, mesh, tileSize);
  }
  self.onmessage = function (msg) {
    var _msg$data = msg.data,
      id = _msg$data.id,
      payload = _msg$data.payload;
    if (id == null) return;
    var objects = [];
    var res = null;
    try {
      res = decodeTerrain(payload);
      objects.push(res.indices.buffer);
      objects.push(res.quantizedVertices.buffer);
      self.postMessage({
        id: id,
        payload: res
      }, objects);
    } catch (err) {
      self.postMessage({
        id: id,
        err: err.toString()
      });
    } finally {
      res = null;
      objects = null;
    }
  };

})();

', null, false);
|
|
1587
|
+
var WorkerFactory = /*#__PURE__*/createBase64WorkerFactory('/* rollup-plugin-web-worker-loader */
(function () {
  'use strict';

  function _arrayLikeToArray(r, a) {
    (null == a || a > r.length) && (a = r.length);
    for (var e = 0, n = Array(a); e < a; e++) n[e] = r[e];
    return n;
  }
  function _arrayWithoutHoles(r) {
    if (Array.isArray(r)) return _arrayLikeToArray(r);
  }
  function _iterableToArray(r) {
    if ("undefined" != typeof Symbol && null != r[Symbol.iterator] || null != r["@@iterator"]) return Array.from(r);
  }
  function _nonIterableSpread() {
    throw new TypeError("Invalid attempt to spread non-iterable instance.\nIn order to be iterable, non-array objects must have a [Symbol.iterator]() method.");
  }
  function _toConsumableArray(r) {
    return _arrayWithoutHoles(r) || _iterableToArray(r) || _unsupportedIterableToArray(r) || _nonIterableSpread();
  }
  function _unsupportedIterableToArray(r, a) {
    if (r) {
      if ("string" == typeof r) return _arrayLikeToArray(r, a);
      var t = {}.toString.call(r).slice(8, -1);
      return "Object" === t && r.constructor && (t = r.constructor.name), "Map" === t || "Set" === t ? Array.from(r) : "Arguments" === t || /^(?:Ui|I)nt(?:8|16|32)(?:Clamped)?Array$/.test(t) ? _arrayLikeToArray(r, a) : void 0;
    }
  }

  /** Mapbox Terrain-RGB default decode function
   *  (r * 256 * 256) / 10 + (g * 256) / 10 + b / 10 - 10000
   */
  var defaultMapboxDecodeRgb = function defaultMapboxDecodeRgb(r, g, b, a) {
    return r * 6553.6 + g * 25.6 + b * 0.1 - 10000;
  };
  function rgbTerrainToGrid(png, decodeRgb) {
    // maybe we should do this on the GPU using REGL?
    // but that would require GPU -> CPU -> GPU
    var gridSize = png.shape[0] + 1;
    var terrain = new Float32Array(gridSize * gridSize);
    var tileSize = png.shape[0];
    var decode = decodeRgb !== null && decodeRgb !== void 0 ? decodeRgb : defaultMapboxDecodeRgb;

    // decode terrain values
    for (var y = 0; y < tileSize; y++) {
      for (var x = 0; x < tileSize; x++) {
        var yc = y;
        var _r = png.get(x, yc, 0);
        var _g = png.get(x, yc, 1);
        var _b = png.get(x, yc, 2);
        var _a = png.get(x, yc, 3);
        terrain[y * gridSize + x] = decode(_r, _g, _b, _a);
      }
    }
    // backfill right and bottom borders
    for (var _x = 0; _x < gridSize - 1; _x++) {
      terrain[gridSize * (gridSize - 1) + _x] = terrain[gridSize * (gridSize - 2) + _x];
    }
    for (var _y = 0; _y < gridSize; _y++) {
      terrain[gridSize * _y + gridSize - 1] = terrain[gridSize * _y + gridSize - 2];
    }
    return terrain;
  }

  /** Terrain workers should return a quantized mesh */

  function createQuantizedMeshData(tile, mesh, tileSize, terrain) {
    /** Terrain is passed through so we can keep track of it
     * for overscaled tiles
     */

    var xvals = [];
    var yvals = [];
    var heightMeters = [];
    var northIndices = [];
    var southIndices = [];
    var eastIndices = [];
    var westIndices = [];
    var minimumHeight = Infinity;
    var maximumHeight = -Infinity;
    var scalar = 32768.0 / tileSize;

    // There appears to be a problem with the x/y indexing when using 512x512 tiles
    // This may be solved by increasing the minumumErrorLevel in the terrain provider
    for (var ix = 0; ix < mesh.vertices.length / 2; ix++) {
      var vertexIx = ix;
      var px = mesh.vertices[ix * 2];
      var py = mesh.vertices[ix * 2 + 1];
      var height = tile.terrain[py * (tileSize + 1) + px];
      if (height > maximumHeight) maximumHeight = height;
      if (height < minimumHeight) minimumHeight = height;
      heightMeters.push(height);
      if (py == 0) northIndices.push(vertexIx);
      if (py == tileSize) southIndices.push(vertexIx);
      if (px == 0) westIndices.push(vertexIx);
      if (px == tileSize) eastIndices.push(vertexIx);
      var xv = px * scalar;
      var yv = (tileSize - py) * scalar;
      xvals.push(xv);
      yvals.push(yv);
    }
    var heightRange = maximumHeight - minimumHeight;
    var heights = heightMeters.map(function (d) {
      if (heightRange < 1) return 0;
      return (d - minimumHeight) * (32768.0 / heightRange);
    });
    var triangles = new Uint16Array(mesh.triangles);
    var quantizedVertices = new Uint16Array(//verts
    [].concat(xvals, yvals, _toConsumableArray(heights)));

    // SE NW NE
    // NE NW SE

    return {
      minimumHeight: minimumHeight,
      maximumHeight: maximumHeight,
      quantizedVertices: quantizedVertices,
      indices: triangles,
      westIndices: westIndices,
      southIndices: southIndices,
      eastIndices: eastIndices,
      northIndices: northIndices,
      quantizedHeights: terrain
    };
  }

  function iota(n) {
    var result = new Array(n);
    for(var i=0; i<n; ++i) {
      result[i] = i;
    }
    return result
  }

  var iota_1 = iota;

  /*!
   * Determine if an object is a Buffer
   *
   * @author   Feross Aboukhadijeh <https://feross.org>
   * @license  MIT
   */
  // The _isBuffer check is for Safari 5-7 support, because it's missing
  // Object.prototype.constructor. Remove this eventually
  var isBuffer_1 = function (obj) {
    return obj != null && (isBuffer(obj) || isSlowBuffer(obj) || !!obj._isBuffer)
  };

  function isBuffer (obj) {
    return !!obj.constructor && typeof obj.constructor.isBuffer === 'function' && obj.constructor.isBuffer(obj)
  }

  // For Node v0.10 support. Remove this eventually.
  function isSlowBuffer (obj) {
    return typeof obj.readFloatLE === 'function' && typeof obj.slice === 'function' && isBuffer(obj.slice(0, 0))
  }

  var hasTypedArrays  = ((typeof Float64Array) !== "undefined");

  function compare1st(a, b) {
    return a[0] - b[0]
  }

  function order() {
    var stride = this.stride;
    var terms = new Array(stride.length);
    var i;
    for(i=0; i<terms.length; ++i) {
      terms[i] = [Math.abs(stride[i]), i];
    }
    terms.sort(compare1st);
    var result = new Array(terms.length);
    for(i=0; i<result.length; ++i) {
      result[i] = terms[i][1];
    }
    return result
  }

  function compileConstructor(dtype, dimension) {
    var className = ["View", dimension, "d", dtype].join("");
    if(dimension < 0) {
      className = "View_Nil" + dtype;
    }
    var useGetters = (dtype === "generic");

    if(dimension === -1) {
      //Special case for trivial arrays
      var code =
        "function "+className+"(a){this.data=a;};\
var proto="+className+".prototype;\
proto.dtype='"+dtype+"';\
proto.index=function(){return -1};\
proto.size=0;\
proto.dimension=-1;\
proto.shape=proto.stride=proto.order=[];\
proto.lo=proto.hi=proto.transpose=proto.step=\
function(){return new "+className+"(this.data);};\
proto.get=proto.set=function(){};\
proto.pick=function(){return null};\
return function construct_"+className+"(a){return new "+className+"(a);}";
      var procedure = new Function(code);
      return procedure()
    } else if(dimension === 0) {
      //Special case for 0d arrays
      var code =
        "function "+className+"(a,d) {\
this.data = a;\
this.offset = d\
};\
var proto="+className+".prototype;\
proto.dtype='"+dtype+"';\
proto.index=function(){return this.offset};\
proto.dimension=0;\
proto.size=1;\
proto.shape=\
proto.stride=\
proto.order=[];\
proto.lo=\
proto.hi=\
proto.transpose=\
proto.step=function "+className+"_copy() {\
return new "+className+"(this.data,this.offset)\
};\
proto.pick=function "+className+"_pick(){\
return TrivialArray(this.data);\
};\
proto.valueOf=proto.get=function "+className+"_get(){\
return "+(useGetters ? "this.data.get(this.offset)" : "this.data[this.offset]")+
  "};\
proto.set=function "+className+"_set(v){\
return "+(useGetters ? "this.data.set(this.offset,v)" : "this.data[this.offset]=v")+"\
};\
return function construct_"+className+"(a,b,c,d){return new "+className+"(a,d)}";
      var procedure = new Function("TrivialArray", code);
      return procedure(CACHED_CONSTRUCTORS[dtype][0])
    }

    var code = ["'use strict'"];

    //Create constructor for view
    var indices = iota_1(dimension);
    var args = indices.map(function(i) { return "i"+i });
    var index_str = "this.offset+" + indices.map(function(i) {
          return "this.stride[" + i + "]*i" + i
        }).join("+");
    var shapeArg = indices.map(function(i) {
        return "b"+i
      }).join(",");
    var strideArg = indices.map(function(i) {
        return "c"+i
      }).join(",");
    code.push(
      "function "+className+"(a," + shapeArg + "," + strideArg + ",d){this.data=a",
        "this.shape=[" + shapeArg + "]",
        "this.stride=[" + strideArg + "]",
        "this.offset=d|0}",
      "var proto="+className+".prototype",
      "proto.dtype='"+dtype+"'",
      "proto.dimension="+dimension);

    //view.size:
    code.push("Object.defineProperty(proto,'size',{get:function "+className+"_size(){\
return "+indices.map(function(i) { return "this.shape["+i+"]" }).join("*"),
  "}})");

    //view.order:
    if(dimension === 1) {
      code.push("proto.order=[0]");
    } else {
      code.push("Object.defineProperty(proto,'order',{get:");
      if(dimension < 4) {
        code.push("function "+className+"_order(){");
        if(dimension === 2) {
          code.push("return (Math.abs(this.stride[0])>Math.abs(this.stride[1]))?[1,0]:[0,1]}})");
        } else if(dimension === 3) {
          code.push(
  "var s0=Math.abs(this.stride[0]),s1=Math.abs(this.stride[1]),s2=Math.abs(this.stride[2]);\
if(s0>s1){\
if(s1>s2){\
return [2,1,0];\
}else if(s0>s2){\
return [1,2,0];\
}else{\
return [1,0,2];\
}\
}else if(s0>s2){\
return [2,0,1];\
}else if(s2>s1){\
return [0,1,2];\
}else{\
return [0,2,1];\
}}})");
        }
      } else {
        code.push("ORDER})");
      }
    }

    //view.set(i0, ..., v):
    code.push(
  "proto.set=function "+className+"_set("+args.join(",")+",v){");
    if(useGetters) {
      code.push("return this.data.set("+index_str+",v)}");
    } else {
      code.push("return this.data["+index_str+"]=v}");
    }

    //view.get(i0, ...):
    code.push("proto.get=function "+className+"_get("+args.join(",")+"){");
    if(useGetters) {
      code.push("return this.data.get("+index_str+")}");
    } else {
      code.push("return this.data["+index_str+"]}");
    }

    //view.index:
    code.push(
      "proto.index=function "+className+"_index(", args.join(), "){return "+index_str+"}");

    //view.hi():
    code.push("proto.hi=function "+className+"_hi("+args.join(",")+"){return new "+className+"(this.data,"+
      indices.map(function(i) {
        return ["(typeof i",i,"!=='number'||i",i,"<0)?this.shape[", i, "]:i", i,"|0"].join("")
      }).join(",")+","+
      indices.map(function(i) {
        return "this.stride["+i + "]"
      }).join(",")+",this.offset)}");

    //view.lo():
    var a_vars = indices.map(function(i) { return "a"+i+"=this.shape["+i+"]" });
    var c_vars = indices.map(function(i) { return "c"+i+"=this.stride["+i+"]" });
    code.push("proto.lo=function "+className+"_lo("+args.join(",")+"){var b=this.offset,d=0,"+a_vars.join(",")+","+c_vars.join(","));
    for(var i=0; i<dimension; ++i) {
      code.push(
  "if(typeof i"+i+"==='number'&&i"+i+">=0){\
d=i"+i+"|0;\
b+=c"+i+"*d;\
a"+i+"-=d}");
    }
    code.push("return new "+className+"(this.data,"+
      indices.map(function(i) {
        return "a"+i
      }).join(",")+","+
      indices.map(function(i) {
        return "c"+i
      }).join(",")+",b)}");

    //view.step():
    code.push("proto.step=function "+className+"_step("+args.join(",")+"){var "+
      indices.map(function(i) {
        return "a"+i+"=this.shape["+i+"]"
      }).join(",")+","+
      indices.map(function(i) {
        return "b"+i+"=this.stride["+i+"]"
      }).join(",")+",c=this.offset,d=0,ceil=Math.ceil");
    for(var i=0; i<dimension; ++i) {
      code.push(
  "if(typeof i"+i+"==='number'){\
d=i"+i+"|0;\
if(d<0){\
c+=b"+i+"*(a"+i+"-1);\
a"+i+"=ceil(-a"+i+"/d)\
}else{\
a"+i+"=ceil(a"+i+"/d)\
}\
b"+i+"*=d\
}");
    }
    code.push("return new "+className+"(this.data,"+
      indices.map(function(i) {
        return "a" + i
      }).join(",")+","+
      indices.map(function(i) {
        return "b" + i
      }).join(",")+",c)}");

    //view.transpose():
    var tShape = new Array(dimension);
    var tStride = new Array(dimension);
    for(var i=0; i<dimension; ++i) {
      tShape[i] = "a[i"+i+"]";
      tStride[i] = "b[i"+i+"]";
    }
    code.push("proto.transpose=function "+className+"_transpose("+args+"){"+
      args.map(function(n,idx) { return n + "=(" + n + "===undefined?" + idx + ":" + n + "|0)"}).join(";"),
      "var a=this.shape,b=this.stride;return new "+className+"(this.data,"+tShape.join(",")+","+tStride.join(",")+",this.offset)}");

    //view.pick():
    code.push("proto.pick=function "+className+"_pick("+args+"){var a=[],b=[],c=this.offset");
    for(var i=0; i<dimension; ++i) {
      code.push("if(typeof i"+i+"==='number'&&i"+i+">=0){c=(c+this.stride["+i+"]*i"+i+")|0}else{a.push(this.shape["+i+"]);b.push(this.stride["+i+"])}");
    }
    code.push("var ctor=CTOR_LIST[a.length+1];return ctor(this.data,a,b,c)}");

    //Add return statement
    code.push("return function construct_"+className+"(data,shape,stride,offset){return new "+className+"(data,"+
      indices.map(function(i) {
        return "shape["+i+"]"
      }).join(",")+","+
      indices.map(function(i) {
        return "stride["+i+"]"
      }).join(",")+",offset)}");

    //Compile procedure
    var procedure = new Function("CTOR_LIST", "ORDER", code.join("\n"));
    return procedure(CACHED_CONSTRUCTORS[dtype], order)
  }

  function arrayDType(data) {
    if(isBuffer_1(data)) {
      return "buffer"
    }
    if(hasTypedArrays) {
      switch(Object.prototype.toString.call(data)) {
        case "[object Float64Array]":
          return "float64"
        case "[object Float32Array]":
          return "float32"
        case "[object Int8Array]":
          return "int8"
        case "[object Int16Array]":
          return "int16"
        case "[object Int32Array]":
          return "int32"
        case "[object Uint8Array]":
          return "uint8"
        case "[object Uint16Array]":
          return "uint16"
        case "[object Uint32Array]":
          return "uint32"
        case "[object Uint8ClampedArray]":
          return "uint8_clamped"
        case "[object BigInt64Array]":
          return "bigint64"
        case "[object BigUint64Array]":
          return "biguint64"
      }
    }
    if(Array.isArray(data)) {
      return "array"
    }
    return "generic"
  }

  var CACHED_CONSTRUCTORS = {
    "float32":[],
    "float64":[],
    "int8":[],
    "int16":[],
    "int32":[],
    "uint8":[],
    "uint16":[],
    "uint32":[],
    "array":[],
    "uint8_clamped":[],
    "bigint64": [],
    "biguint64": [],
    "buffer":[],
    "generic":[]
  }

  ;
  function wrappedNDArrayCtor(data, shape, stride, offset) {
    if(data === undefined) {
      var ctor = CACHED_CONSTRUCTORS.array[0];
      return ctor([])
    } else if(typeof data === "number") {
      data = [data];
    }
    if(shape === undefined) {
      shape = [ data.length ];
    }
    var d = shape.length;
    if(stride === undefined) {
      stride = new Array(d);
      for(var i=d-1, sz=1; i>=0; --i) {
        stride[i] = sz;
        sz *= shape[i];
      }
    }
    if(offset === undefined) {
      offset = 0;
      for(var i=0; i<d; ++i) {
        if(stride[i] < 0) {
          offset -= (shape[i]-1)*stride[i];
        }
      }
    }
    var dtype = arrayDType(data);
    var ctor_list = CACHED_CONSTRUCTORS[dtype];
    while(ctor_list.length <= d+1) {
      ctor_list.push(compileConstructor(dtype, ctor_list.length-1));
    }
    var ctor = ctor_list[d+1];
    return ctor(data, shape, stride, offset)
  }

  var ndarray = wrappedNDArrayCtor;

  class Martini {
      constructor(gridSize = 257) {
          this.gridSize = gridSize;
          const tileSize = gridSize - 1;
          if (tileSize & (tileSize - 1)) throw new Error(
              `Expected grid size to be 2^n+1, got ${gridSize}.`);

          this.numTriangles = tileSize * tileSize * 2 - 2;
          this.numParentTriangles = this.numTriangles - tileSize * tileSize;

          this.indices = new Uint32Array(this.gridSize * this.gridSize);

          // coordinates for all possible triangles in an RTIN tile
          this.coords = new Uint16Array(this.numTriangles * 4);

          // get triangle coordinates from its index in an implicit binary tree
          for (let i = 0; i < this.numTriangles; i++) {
              let id = i + 2;
              let ax = 0, ay = 0, bx = 0, by = 0, cx = 0, cy = 0;
              if (id & 1) {
                  bx = by = cx = tileSize; // bottom-left triangle
              } else {
                  ax = ay = cy = tileSize; // top-right triangle
              }
              while ((id >>= 1) > 1) {
                  const mx = (ax + bx) >> 1;
                  const my = (ay + by) >> 1;

                  if (id & 1) { // left half
                      bx = ax; by = ay;
                      ax = cx; ay = cy;
                  } else { // right half
                      ax = bx; ay = by;
                      bx = cx; by = cy;
                  }
                  cx = mx; cy = my;
              }
              const k = i * 4;
              this.coords[k + 0] = ax;
              this.coords[k + 1] = ay;
              this.coords[k + 2] = bx;
              this.coords[k + 3] = by;
          }
      }

      createTile(terrain) {
          return new Tile(terrain, this);
      }
  }

  class Tile {
      constructor(terrain, martini) {
          const size = martini.gridSize;
          if (terrain.length !== size * size) throw new Error(
              `Expected terrain data of length ${size * size} (${size} x ${size}), got ${terrain.length}.`);

          this.terrain = terrain;
          this.martini = martini;
          this.errors = new Float32Array(terrain.length);
          this.update();
      }

      update() {
          const {numTriangles, numParentTriangles, coords, gridSize: size} = this.martini;
          const {terrain, errors} = this;

          // iterate over all possible triangles, starting from the smallest level
          for (let i = numTriangles - 1; i >= 0; i--) {
              const k = i * 4;
              const ax = coords[k + 0];
              const ay = coords[k + 1];
              const bx = coords[k + 2];
              const by = coords[k + 3];
              const mx = (ax + bx) >> 1;
              const my = (ay + by) >> 1;
              const cx = mx + my - ay;
              const cy = my + ax - mx;

              // calculate error in the middle of the long edge of the triangle
              const interpolatedHeight = (terrain[ay * size + ax] + terrain[by * size + bx]) / 2;
              const middleIndex = my * size + mx;
              const middleError = Math.abs(interpolatedHeight - terrain[middleIndex]);

              errors[middleIndex] = Math.max(errors[middleIndex], middleError);

              if (i < numParentTriangles) { // bigger triangles; accumulate error with children
                  const leftChildIndex = ((ay + cy) >> 1) * size + ((ax + cx) >> 1);
                  const rightChildIndex = ((by + cy) >> 1) * size + ((bx + cx) >> 1);
                  errors[middleIndex] = Math.max(errors[middleIndex], errors[leftChildIndex], errors[rightChildIndex]);
              }
          }
      }

      getMesh(maxError = 0, maxLength = null) {
          const {gridSize: size, indices} = this.martini;
          const {errors} = this;
          let numVertices = 0;
          let numTriangles = 0;
          const max = size - 1;

          // The maxLength parameter will cause triangles to be generated until the legs are below this length
          // It is meant to support cases where a certain mesh density is required to do spherical math on digital globes
          const maxScale = maxLength || size;

          // use an index grid to keep track of vertices that were already used to avoid duplication
          indices.fill(0);

          // retrieve mesh in two stages that both traverse the error map:
          // - countElements: find used vertices (and assign each an index), and count triangles (for minimum allocation)
          // - processTriangle: fill the allocated vertices & triangles typed arrays

          function countElements(ax, ay, bx, by, cx, cy) {
              const mx = (ax + bx) >> 1;
              const my = (ay + by) >> 1;

              const legLength = Math.abs(ax - cx) + Math.abs(ay - cy);
              if ((legLength > 1 && errors[my * size + mx] > maxError) || legLength > maxScale) {
                  countElements(cx, cy, ax, ay, mx, my);
                  countElements(bx, by, cx, cy, mx, my);
              } else {
                  indices[ay * size + ax] = indices[ay * size + ax] || ++numVertices;
                  indices[by * size + bx] = indices[by * size + bx] || ++numVertices;
                  indices[cy * size + cx] = indices[cy * size + cx] || ++numVertices;
                  numTriangles++;
              }
          }
          countElements(0, 0, max, max, max, 0);
          countElements(max, max, 0, 0, 0, max);

          const vertices = new Uint16Array(numVertices * 2);
          const triangles = new Uint32Array(numTriangles * 3);
          let triIndex = 0;

          function processTriangle(ax, ay, bx, by, cx, cy) {
              const mx = (ax + bx) >> 1;
              const my = (ay + by) >> 1;

              const legLength = Math.abs(ax - cx) + Math.abs(ay - cy);
              if ((legLength > 1 && errors[my * size + mx] > maxError) || legLength > maxScale) {
                  // triangle doesn't approximate the surface well enough; drill down further
                  processTriangle(cx, cy, ax, ay, mx, my);
                  processTriangle(bx, by, cx, cy, mx, my);

              } else {
                  // add a triangle
                  const a = indices[ay * size + ax] - 1;
                  const b = indices[by * size + bx] - 1;
                  const c = indices[cy * size + cx] - 1;

                  vertices[2 * a] = ax;
                  vertices[2 * a + 1] = ay;

                  vertices[2 * b] = bx;
                  vertices[2 * b + 1] = by;

                  vertices[2 * c] = cx;
                  vertices[2 * c + 1] = cy;

                  triangles[triIndex++] = a;
                  triangles[triIndex++] = b;
                  triangles[triIndex++] = c;
              }
          }
          processTriangle(0, 0, max, max, max, 0);
          processTriangle(max, max, 0, 0, 0, max);

          return {vertices, triangles};
      }
  }

  // https://github.com/CesiumGS/cesium/blob/1.76/Source/WorkersES6/createVerticesFromQuantizedTerrainMesh.js

  var martiniCache = {};
  function decodeTerrain(parameters, transferableObjects) {
    var _martiniCache$tileSiz;
    var imageData = parameters.imageData,
      _parameters$tileSize = parameters.tileSize,
      tileSize = _parameters$tileSize === void 0 ? 256 : _parameters$tileSize,
      errorLevel = parameters.errorLevel,
      maxVertexDistance = parameters.maxVertexDistance;

    // Height data can be either an array of numbers (for pre-existing terrain data)
    // or an image data array (for decoding from an image)

    var heightData = {
      type: "image",
      array: imageData
    };
    var terrain;
    {
      var array = heightData.array;
      var pixels = ndarray(new Uint8Array(array), [tileSize, tileSize, 4], [4, 4 * tileSize, 1], 0);
      terrain = rgbTerrainToGrid(pixels);
    }

    // Tile size must be maintained through the life of the worker
    (_martiniCache$tileSiz = martiniCache[tileSize]) !== null && _martiniCache$tileSiz !== void 0 ? _martiniCache$tileSiz : martiniCache[tileSize] = new Martini(tileSize + 1);
    var tile = martiniCache[tileSize].createTile(terrain);

    // get a mesh (vertices and triangles indices) for a 10m error
    var mesh = tile.getMesh(errorLevel, Math.min(maxVertexDistance, tileSize));
    var res = createQuantizedMeshData(tile, mesh, tileSize,
    // Only include vertex data if anticipate upscaling tile
    terrain );
    transferableObjects.push(res.indices.buffer);
    transferableObjects.push(res.quantizedVertices.buffer);
    if (res.quantizedHeights) {
      transferableObjects.push(res.quantizedHeights.buffer);
    }
    return res;
  }
  self.onmessage = function (msg) {
    var _msg$data = msg.data,
      id = _msg$data.id,
      payload = _msg$data.payload;
    if (id == null) return;
    var objects = [];
    var res = null;
    try {
      res = decodeTerrain(payload, objects);
      // @ts-ignore
      self.postMessage({
        id: id,
        payload: res
      }, objects);
    } catch (err) {
      var _err$message;
      var _msg = (_err$message = err.message) !== null && _err$message !== void 0 ? _err$message : err;
      self.postMessage({
        id: id,
        err: _msg.toString()
      });
    } finally {
      res = null;
      objects = null;
    }
  };

})();

', null, false);
|
|
1076
1588
|
/* eslint-enable */
|
|
1077
1589
|
|
|
1078
1590
|
var MapboxTerrainProvider = /*#__PURE__*/function (_MartiniTerrainProvid) {
|
|
@@ -1093,7 +1605,15 @@ var MapboxTerrainProvider = /*#__PURE__*/function (_MartiniTerrainProvid) {
|
|
|
1093
1605
|
}(MartiniTerrainProvider);
|
|
1094
1606
|
|
|
1095
1607
|
exports.DefaultHeightmapResource = DefaultHeightmapResource;
|
|
1608
|
+
exports.DefaultTerrainDecoder = DefaultTerrainDecoder;
|
|
1096
1609
|
exports.MapboxTerrainResource = MapboxTerrainResource;
|
|
1097
1610
|
exports.MartiniTerrainProvider = MartiniTerrainProvider;
|
|
1611
|
+
exports.StretchedTilingScheme = StretchedTilingScheme;
|
|
1612
|
+
exports.WorkerFarmTerrainDecoder = WorkerFarmTerrainDecoder;
|
|
1613
|
+
exports.createQuantizedMeshData = createQuantizedMeshData;
|
|
1098
1614
|
exports["default"] = MapboxTerrainProvider;
|
|
1615
|
+
exports.emptyMesh = emptyMesh;
|
|
1616
|
+
exports.rgbTerrainToGrid = rgbTerrainToGrid;
|
|
1617
|
+
exports.subsetByWindow = subsetByWindow;
|
|
1618
|
+
exports.testMeshData = testMeshData;
|
|
1099
1619
|
//# sourceMappingURL=index.cjs.map
|