@loaders.gl/tile-converter 4.0.0-alpha.23 → 4.0.0-alpha.25

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.
Files changed (118) hide show
  1. package/dist/3d-tiles-converter/helpers/b3dm-converter.d.ts +4 -4
  2. package/dist/3d-tiles-converter/helpers/load-i3s.d.ts.map +1 -1
  3. package/dist/converter.min.js +105 -105
  4. package/dist/dist.min.js +725 -652
  5. package/dist/es5/3d-tiles-converter/helpers/b3dm-converter.js +18 -18
  6. package/dist/es5/3d-tiles-converter/helpers/b3dm-converter.js.map +1 -1
  7. package/dist/es5/3d-tiles-converter/helpers/load-i3s.js.map +1 -1
  8. package/dist/es5/deps-installer/deps-installer.js +1 -1
  9. package/dist/es5/i3s-converter/helpers/batch-ids-extensions.js +16 -14
  10. package/dist/es5/i3s-converter/helpers/batch-ids-extensions.js.map +1 -1
  11. package/dist/es5/i3s-converter/helpers/feature-attributes.js +6 -18
  12. package/dist/es5/i3s-converter/helpers/feature-attributes.js.map +1 -1
  13. package/dist/es5/i3s-converter/helpers/geometry-attributes.js +83 -44
  14. package/dist/es5/i3s-converter/helpers/geometry-attributes.js.map +1 -1
  15. package/dist/es5/i3s-converter/helpers/geometry-converter.js +13 -83
  16. package/dist/es5/i3s-converter/helpers/geometry-converter.js.map +1 -1
  17. package/dist/es5/i3s-converter/helpers/preprocess-3d-tiles.js +15 -8
  18. package/dist/es5/i3s-converter/helpers/preprocess-3d-tiles.js.map +1 -1
  19. package/dist/es5/i3s-converter/i3s-converter.js.map +1 -1
  20. package/dist/es5/i3s-converter/types.js.map +1 -1
  21. package/dist/es5/pgm-loader.js +11 -3
  22. package/dist/es5/pgm-loader.js.map +1 -1
  23. package/dist/esm/3d-tiles-converter/helpers/b3dm-converter.js +8 -8
  24. package/dist/esm/3d-tiles-converter/helpers/b3dm-converter.js.map +1 -1
  25. package/dist/esm/3d-tiles-converter/helpers/load-i3s.js.map +1 -1
  26. package/dist/esm/deps-installer/deps-installer.js +1 -1
  27. package/dist/esm/i3s-converter/helpers/batch-ids-extensions.js +5 -15
  28. package/dist/esm/i3s-converter/helpers/batch-ids-extensions.js.map +1 -1
  29. package/dist/esm/i3s-converter/helpers/feature-attributes.js +5 -5
  30. package/dist/esm/i3s-converter/helpers/feature-attributes.js.map +1 -1
  31. package/dist/esm/i3s-converter/helpers/geometry-attributes.js +76 -34
  32. package/dist/esm/i3s-converter/helpers/geometry-attributes.js.map +1 -1
  33. package/dist/esm/i3s-converter/helpers/geometry-converter.js +10 -80
  34. package/dist/esm/i3s-converter/helpers/geometry-converter.js.map +1 -1
  35. package/dist/esm/i3s-converter/helpers/preprocess-3d-tiles.js +15 -9
  36. package/dist/esm/i3s-converter/helpers/preprocess-3d-tiles.js.map +1 -1
  37. package/dist/esm/i3s-converter/i3s-converter.js.map +1 -1
  38. package/dist/esm/i3s-converter/types.js.map +1 -1
  39. package/dist/esm/i3s-server/bin/i3s-server.min.js +75 -75
  40. package/dist/esm/pgm-loader.js +7 -4
  41. package/dist/esm/pgm-loader.js.map +1 -1
  42. package/dist/i3s-converter/helpers/batch-ids-extensions.d.ts +4 -1
  43. package/dist/i3s-converter/helpers/batch-ids-extensions.d.ts.map +1 -1
  44. package/dist/i3s-converter/helpers/feature-attributes.d.ts +6 -6
  45. package/dist/i3s-converter/helpers/feature-attributes.d.ts.map +1 -1
  46. package/dist/i3s-converter/helpers/geometry-attributes.d.ts.map +1 -1
  47. package/dist/i3s-converter/helpers/geometry-converter.d.ts +1 -1
  48. package/dist/i3s-converter/helpers/geometry-converter.d.ts.map +1 -1
  49. package/dist/i3s-converter/helpers/preprocess-3d-tiles.d.ts.map +1 -1
  50. package/dist/i3s-converter/i3s-converter.d.ts.map +1 -1
  51. package/dist/i3s-converter/types.d.ts +13 -5
  52. package/dist/i3s-converter/types.d.ts.map +1 -1
  53. package/dist/pgm-loader.d.ts +9 -2
  54. package/dist/pgm-loader.d.ts.map +1 -1
  55. package/dist/slpk-extractor.min.js +42 -42
  56. package/package.json +14 -14
  57. package/src/3d-tiles-converter/helpers/b3dm-converter.ts +8 -8
  58. package/src/3d-tiles-converter/helpers/load-i3s.ts +1 -0
  59. package/src/i3s-converter/helpers/batch-ids-extensions.ts +14 -35
  60. package/src/i3s-converter/helpers/feature-attributes.ts +14 -11
  61. package/src/i3s-converter/helpers/geometry-attributes.ts +80 -50
  62. package/src/i3s-converter/helpers/geometry-converter.ts +41 -177
  63. package/src/i3s-converter/helpers/preprocess-3d-tiles.ts +30 -9
  64. package/src/i3s-converter/i3s-converter.ts +0 -2
  65. package/src/i3s-converter/types.ts +14 -5
  66. package/src/pgm-loader.ts +15 -7
  67. package/dist/3d-tiles-converter/3d-tiles-converter.js +0 -279
  68. package/dist/3d-tiles-converter/helpers/b3dm-converter.js +0 -271
  69. package/dist/3d-tiles-converter/helpers/i3s-obb-to-3d-tiles-obb.js +0 -23
  70. package/dist/3d-tiles-converter/helpers/load-i3s.js +0 -42
  71. package/dist/3d-tiles-converter/helpers/texture-atlas.js +0 -54
  72. package/dist/3d-tiles-converter/json-templates/tileset.js +0 -43
  73. package/dist/bundle.js +0 -5
  74. package/dist/constants.js +0 -4
  75. package/dist/converter-cli.js +0 -222
  76. package/dist/deps-installer/deps-installer.js +0 -89
  77. package/dist/i3s-converter/helpers/batch-ids-extensions.js +0 -179
  78. package/dist/i3s-converter/helpers/coordinate-converter.js +0 -122
  79. package/dist/i3s-converter/helpers/create-scene-server-path.js +0 -28
  80. package/dist/i3s-converter/helpers/feature-attributes.js +0 -218
  81. package/dist/i3s-converter/helpers/geometry-attributes.js +0 -203
  82. package/dist/i3s-converter/helpers/geometry-converter.js +0 -1321
  83. package/dist/i3s-converter/helpers/gltf-attributes.js +0 -129
  84. package/dist/i3s-converter/helpers/load-3d-tiles.js +0 -99
  85. package/dist/i3s-converter/helpers/node-debug.js +0 -120
  86. package/dist/i3s-converter/helpers/node-index-document.js +0 -271
  87. package/dist/i3s-converter/helpers/node-pages.js +0 -316
  88. package/dist/i3s-converter/helpers/preprocess-3d-tiles.js +0 -100
  89. package/dist/i3s-converter/helpers/tileset-traversal.js +0 -29
  90. package/dist/i3s-converter/i3s-converter.js +0 -964
  91. package/dist/i3s-converter/json-templates/geometry-definitions.js +0 -87
  92. package/dist/i3s-converter/json-templates/layers.js +0 -139
  93. package/dist/i3s-converter/json-templates/metadata.js +0 -25
  94. package/dist/i3s-converter/json-templates/node.js +0 -89
  95. package/dist/i3s-converter/json-templates/scene-server.js +0 -31
  96. package/dist/i3s-converter/json-templates/shared-resources.js +0 -129
  97. package/dist/i3s-converter/json-templates/store.js +0 -103
  98. package/dist/i3s-converter/types.js +0 -17
  99. package/dist/i3s-server/app.js +0 -29
  100. package/dist/i3s-server/bin/www.js +0 -37
  101. package/dist/i3s-server/controllers/index-controller.js +0 -31
  102. package/dist/i3s-server/controllers/slpk-controller.js +0 -33
  103. package/dist/i3s-server/routes/index.js +0 -20
  104. package/dist/i3s-server/routes/slpk-router.js +0 -34
  105. package/dist/i3s-server/utils/create-scene-server.js +0 -22
  106. package/dist/i3s-server/utils/server-utils.js +0 -66
  107. package/dist/index.js +0 -10
  108. package/dist/lib/utils/cli-utils.js +0 -82
  109. package/dist/lib/utils/compress-util.js +0 -257
  110. package/dist/lib/utils/file-utils.js +0 -139
  111. package/dist/lib/utils/geometry-utils.js +0 -18
  112. package/dist/lib/utils/lod-conversion-utils.js +0 -76
  113. package/dist/lib/utils/queue.js +0 -18
  114. package/dist/lib/utils/statistic-utills.js +0 -64
  115. package/dist/lib/utils/write-queue.js +0 -80
  116. package/dist/pgm-loader.js +0 -24
  117. package/dist/slpk-extractor/slpk-extractor.js +0 -75
  118. package/dist/slpk-extractor-cli.js +0 -102
@@ -1,964 +0,0 @@
1
- "use strict";
2
- // loaders.gl, MIT license
3
- var __createBinding = (this && this.__createBinding) || (Object.create ? (function(o, m, k, k2) {
4
- if (k2 === undefined) k2 = k;
5
- var desc = Object.getOwnPropertyDescriptor(m, k);
6
- if (!desc || ("get" in desc ? !m.__esModule : desc.writable || desc.configurable)) {
7
- desc = { enumerable: true, get: function() { return m[k]; } };
8
- }
9
- Object.defineProperty(o, k2, desc);
10
- }) : (function(o, m, k, k2) {
11
- if (k2 === undefined) k2 = k;
12
- o[k2] = m[k];
13
- }));
14
- var __setModuleDefault = (this && this.__setModuleDefault) || (Object.create ? (function(o, v) {
15
- Object.defineProperty(o, "default", { enumerable: true, value: v });
16
- }) : function(o, v) {
17
- o["default"] = v;
18
- });
19
- var __importStar = (this && this.__importStar) || function (mod) {
20
- if (mod && mod.__esModule) return mod;
21
- var result = {};
22
- if (mod != null) for (var k in mod) if (k !== "default" && Object.prototype.hasOwnProperty.call(mod, k)) __createBinding(result, mod, k);
23
- __setModuleDefault(result, mod);
24
- return result;
25
- };
26
- var __importDefault = (this && this.__importDefault) || function (mod) {
27
- return (mod && mod.__esModule) ? mod : { "default": mod };
28
- };
29
- Object.defineProperty(exports, "__esModule", { value: true });
30
- const core_1 = require("@loaders.gl/core");
31
- const _3d_tiles_1 = require("@loaders.gl/3d-tiles");
32
- const path_1 = require("path");
33
- const uuid_1 = require("uuid");
34
- const process_1 = __importDefault(require("process"));
35
- const json_map_transform_1 = __importDefault(require("json-map-transform"));
36
- const md5_1 = __importDefault(require("md5"));
37
- const node_pages_1 = __importDefault(require("./helpers/node-pages"));
38
- const file_utils_1 = require("../lib/utils/file-utils");
39
- const compress_util_1 = require("../lib/utils/compress-util");
40
- const statistic_utills_1 = require("../lib/utils/statistic-utills");
41
- const geometry_converter_1 = __importStar(require("./helpers/geometry-converter"));
42
- const coordinate_converter_1 = require("./helpers/coordinate-converter");
43
- const create_scene_server_path_1 = require("./helpers/create-scene-server-path");
44
- const lod_conversion_utils_1 = require("../lib/utils/lod-conversion-utils");
45
- const pgm_loader_1 = require("../pgm-loader");
46
- const layers_1 = require("./json-templates/layers");
47
- const geometry_definitions_1 = require("./json-templates/geometry-definitions");
48
- const shared_resources_1 = require("./json-templates/shared-resources");
49
- const node_debug_1 = require("./helpers/node-debug");
50
- const textures_1 = require("@loaders.gl/textures");
51
- const images_1 = require("@loaders.gl/images");
52
- const types_1 = require("./types");
53
- const worker_utils_1 = require("@loaders.gl/worker-utils");
54
- const write_queue_1 = __importDefault(require("../lib/utils/write-queue"));
55
- const constants_1 = require("../constants");
56
- const feature_attributes_1 = require("./helpers/feature-attributes");
57
- const node_index_document_1 = require("./helpers/node-index-document");
58
- const load_3d_tiles_1 = require("./helpers/load-3d-tiles");
59
- const core_2 = require("@math.gl/core");
60
- const tiles_1 = require("@loaders.gl/tiles");
61
- const tileset_traversal_1 = require("./helpers/tileset-traversal");
62
- const preprocess_3d_tiles_1 = require("./helpers/preprocess-3d-tiles");
63
- const ION_DEFAULT_TOKEN = process_1.default.env?.IonToken;
64
- const HARDCODED_NODES_PER_PAGE = 64;
65
- const _3D_TILES = '3DTILES';
66
- const _3D_OBJECT_LAYER_TYPE = '3DObject';
67
- const REFRESH_TOKEN_TIMEOUT = 1800; // 30 minutes in seconds
68
- const CESIUM_DATASET_PREFIX = 'https://';
69
- // const FS_FILE_TOO_LARGE = 'ERR_FS_FILE_TOO_LARGE';
70
- /**
71
- * Converter from 3d-tiles tileset to i3s layer
72
- */
73
- class I3SConverter {
74
- constructor() {
75
- this.boundingVolumeWarnings = [];
76
- this.conversionStartTime = [0, 0];
77
- this.refreshTokenTime = [0, 0];
78
- this.sourceTileset = null;
79
- this.loadOptions = {
80
- _nodeWorkers: true,
81
- reuseWorkers: true,
82
- useLocalLibraries: true,
83
- basis: {
84
- format: 'rgba32',
85
- // We need to load local fs workers because nodejs can't load workers from the Internet
86
- workerUrl: './modules/textures/dist/basis-worker-node.js'
87
- },
88
- // We need to load local fs workers because nodejs can't load workers from the Internet
89
- draco: { workerUrl: './modules/draco/dist/draco-worker-node.js' },
90
- fetch: {},
91
- modules: {}
92
- };
93
- this.geoidHeightModel = null;
94
- this.Loader = _3d_tiles_1.Tiles3DLoader;
95
- this.workerSource = {};
96
- this.writeQueue = new write_queue_1.default();
97
- this.compressList = null;
98
- this.preprocessData = {
99
- meshTopologyTypes: new Set(),
100
- metadataClasses: new Set()
101
- };
102
- this.nodePages = new node_pages_1.default(file_utils_1.writeFile, HARDCODED_NODES_PER_PAGE, this);
103
- this.options = {};
104
- this.layers0Path = '';
105
- this.materialMap = new Map();
106
- this.materialDefinitions = [];
107
- this.geometryMap = new Map();
108
- this.geometryConfigs = [];
109
- this.vertexCounter = 0;
110
- this.layers0 = null;
111
- this.featuresHashArray = [];
112
- this.refinementCounter = {
113
- tilesCount: 0,
114
- tilesWithAddRefineCount: 0
115
- };
116
- this.validate = false;
117
- this.generateTextures = false;
118
- this.generateBoundingVolumes = false;
119
- this.layersHasTexture = false;
120
- this.compressList = null;
121
- }
122
- /**
123
- * Convert a 3d tileset
124
- * @param options
125
- * @param options.inputUrl the url to read the tileset from
126
- * @param options.outputPath the output filename
127
- * @param options.tilesetName the output name of the tileset
128
- * @param options.maxDepth The max tree depth of conversion
129
- * @param options.slpk Generate slpk (Scene Layer Packages) output file
130
- * @param options.sevenZipExe Location of 7z.exe archiver to create slpk on Windows
131
- * @param options.egmFilePath location of *.pgm file to convert heights from ellipsoidal to gravity-related format
132
- * @param options.token Token for Cesium ION tilesets authentication
133
- * @param options.draco Generate I3S 1.7 draco compressed geometries
134
- * @param options.validate -enable validation
135
- * @param options.generateTextures - generate alternative type of textures (to have non-compressed jpeg/png and compressed ktx2)
136
- * @param options.generateBoundingVolumes - generate bounding volumes from vertices coordinates instead of source tiles bounding volumes
137
- * @param options.instantNodeWriting - Keep created 3DNodeIndexDocument files on disk instead of memory. This option reduce memory usage but decelerates conversion speed
138
- */
139
- async convert(options) {
140
- if (core_1.isBrowser) {
141
- console.log(constants_1.BROWSER_ERROR_MESSAGE);
142
- return constants_1.BROWSER_ERROR_MESSAGE;
143
- }
144
- this.conversionStartTime = process_1.default.hrtime();
145
- const { tilesetName, slpk, egmFilePath, inputUrl, validate, outputPath, draco = true, sevenZipExe, maxDepth, token, generateTextures, generateBoundingVolumes, instantNodeWriting = false, mergeMaterials = true, inquirer, metadataClass } = options;
146
- this.options = {
147
- maxDepth,
148
- slpk,
149
- sevenZipExe,
150
- egmFilePath,
151
- draco,
152
- token,
153
- inputUrl,
154
- instantNodeWriting,
155
- mergeMaterials,
156
- inquirer,
157
- metadataClass
158
- };
159
- this.compressList = (this.options.instantNodeWriting && []) || null;
160
- this.validate = Boolean(validate);
161
- this.Loader = inputUrl.indexOf(CESIUM_DATASET_PREFIX) !== -1 ? _3d_tiles_1.CesiumIonLoader : _3d_tiles_1.Tiles3DLoader;
162
- this.generateTextures = Boolean(generateTextures);
163
- this.generateBoundingVolumes = Boolean(generateBoundingVolumes);
164
- this.writeQueue = new write_queue_1.default();
165
- this.writeQueue.startListening();
166
- console.log('Loading egm file...'); // eslint-disable-line
167
- this.geoidHeightModel = await (0, core_1.load)(egmFilePath, pgm_loader_1.PGMLoader);
168
- console.log('Loading egm file completed!'); // eslint-disable-line
169
- if (slpk) {
170
- this.nodePages.useWriteFunction(file_utils_1.writeFileForSlpk);
171
- }
172
- try {
173
- const preloadOptions = await this._fetchPreloadOptions();
174
- let tilesetUrl = inputUrl;
175
- if (preloadOptions.url) {
176
- tilesetUrl = preloadOptions.url;
177
- }
178
- if (preloadOptions.headers) {
179
- this.loadOptions.fetch = { headers: preloadOptions.headers };
180
- }
181
- this.sourceTileset = await (0, load_3d_tiles_1.loadFromArchive)(tilesetUrl, this.Loader, this.loadOptions);
182
- const preprocessResult = await this.preprocessConversion();
183
- if (preprocessResult) {
184
- await this._createAndSaveTileset(outputPath, tilesetName);
185
- await this._finishConversion({ slpk: Boolean(slpk), outputPath, tilesetName });
186
- }
187
- }
188
- catch (error) {
189
- throw error;
190
- }
191
- finally {
192
- await this.writeQueue.finalize();
193
- // Clean up worker pools
194
- const workerFarm = worker_utils_1.WorkerFarm.getWorkerFarm({});
195
- workerFarm.destroy();
196
- }
197
- return 'success';
198
- }
199
- /**
200
- * Preprocess stage of the tile converter. Traverse all the tiles tree and
201
- * check a tile content to be sure that the data is supported
202
- * @returns true - the conversion is possible, false - the tileset's content is not supported
203
- */
204
- async preprocessConversion() {
205
- console.log(`Analyze source tileset`);
206
- const sourceRootTile = this.sourceTileset.root;
207
- await (0, tileset_traversal_1.traverseDatasetWith)(sourceRootTile, null, this.analyzeTile.bind(this), undefined, this.options.maxDepth);
208
- const { meshTopologyTypes, metadataClasses } = this.preprocessData;
209
- console.log(`------------------------------------------------`);
210
- console.log(`Preprocess results:`);
211
- console.log(`glTF mesh topology types: ${Array.from(meshTopologyTypes).join(', ')}`);
212
- if (metadataClasses.size) {
213
- console.log(`Feature metadata classes have been found: ${Array.from(metadataClasses).join(', ')}`);
214
- }
215
- else {
216
- console.log('Feature metadata classes have not been found');
217
- }
218
- console.log(`------------------------------------------------`);
219
- if (!meshTopologyTypes.has(types_1.GLTFPrimitiveModeString.TRIANGLES) &&
220
- !meshTopologyTypes.has(types_1.GLTFPrimitiveModeString.TRIANGLE_STRIP)) {
221
- console.log('The tileset is of unsupported mesh topology types. The conversion will be interrupted.');
222
- console.log(`------------------------------------------------`);
223
- return false;
224
- }
225
- if (metadataClasses.size > 1) {
226
- if (this.options.metadataClass?.length) {
227
- console.log(`${this.options.metadataClass} has been selected`);
228
- }
229
- else if (this.options.inquirer) {
230
- const result = await this.options.inquirer.prompt([
231
- {
232
- name: 'metadataClass',
233
- type: 'list',
234
- message: 'Select feature metadata data class to convert...',
235
- choices: Array.from(metadataClasses)
236
- }
237
- ]);
238
- this.options.metadataClass = result.metadataClass;
239
- console.log(`${result.metadataClass} has been selected`);
240
- }
241
- else {
242
- console.log(`A feature metadata class has not been selected. Start the converter with option "--metadata-class". For example, "npx tile-converter ... --metadata-class ${Array.from(metadataClasses)[0]}"`);
243
- console.log(`------------------------------------------------`);
244
- return false;
245
- }
246
- }
247
- return true;
248
- }
249
- /**
250
- * Analyze a tile content. The callback for preprocess stage.
251
- * @param sourceTile - 3DTiles tile JSON metadata
252
- * @param traversalProps - mandatory argument but it is not used for the preprocess stage
253
- * @returns - nothing
254
- */
255
- async analyzeTile(sourceTile, traversalProps) {
256
- const isTileset = (0, load_3d_tiles_1.isNestedTileset)(sourceTile);
257
- if (isTileset) {
258
- await (0, load_3d_tiles_1.loadNestedTileset)(this.sourceTileset, sourceTile, this.loadOptions);
259
- return null;
260
- }
261
- if (sourceTile.id) {
262
- console.log(`[analyze]: ${sourceTile.id}`); // eslint-disable-line
263
- }
264
- let tileContent = null;
265
- try {
266
- tileContent = await (0, load_3d_tiles_1.loadTile3DContent)(this.sourceTileset, sourceTile, {
267
- ...this.loadOptions,
268
- '3d-tiles': { ...this.loadOptions['3d-tiles'], loadGLTF: false }
269
- });
270
- }
271
- catch (error) {
272
- console.log(`[warning]: Failed to load ${sourceTile.contentUrl}. An I3S tile with empty content will be added to the output tileset`);
273
- }
274
- const tilePreprocessData = await (0, preprocess_3d_tiles_1.analyzeTileContent)(tileContent);
275
- (0, preprocess_3d_tiles_1.mergePreprocessData)(this.preprocessData, tilePreprocessData);
276
- return null;
277
- }
278
- /**
279
- * Convert and save the layer and embedded tiles
280
- * @param outputPath - path to save output data
281
- * @param tilesetName - new tileset path
282
- */
283
- async _createAndSaveTileset(outputPath, tilesetName) {
284
- const tilesetPath = (0, path_1.join)(`${outputPath}`, `${tilesetName}`);
285
- // Removing the tilesetPath needed to exclude erroneous files after conversion
286
- try {
287
- await (0, file_utils_1.removeDir)(tilesetPath);
288
- }
289
- catch (e) {
290
- // do nothing
291
- }
292
- this.layers0Path = (0, path_1.join)(tilesetPath, 'SceneServer', 'layers', '0');
293
- this.materialDefinitions = [];
294
- this.materialMap = new Map();
295
- const sourceRootTile = this.sourceTileset.root;
296
- const sourceBoundingVolume = (0, tiles_1.createBoundingVolume)(sourceRootTile.boundingVolume, new core_2.Matrix4(sourceRootTile.transform), null);
297
- this._formLayers0(tilesetName, sourceBoundingVolume, this.sourceTileset?.root?.boundingVolume?.region);
298
- const boundingVolumes = (0, coordinate_converter_1.createBoundingVolumes)(sourceBoundingVolume, this.geoidHeightModel);
299
- await this.nodePages.push({
300
- index: 0,
301
- lodThreshold: 0,
302
- obb: boundingVolumes.obb,
303
- children: []
304
- });
305
- const rootNode = await node_index_document_1.NodeIndexDocument.createRootNode(boundingVolumes, this);
306
- await (0, tileset_traversal_1.traverseDatasetWith)(sourceRootTile, {
307
- transform: new core_2.Matrix4(sourceRootTile.transform),
308
- parentNodes: [rootNode]
309
- }, this.convertTile.bind(this), this.finalizeTile.bind(this), this.options.maxDepth);
310
- this.layers0.materialDefinitions = this.materialDefinitions;
311
- // @ts-ignore
312
- this.layers0.geometryDefinitions = (0, json_map_transform_1.default)(this.geometryConfigs.map((config) => ({
313
- geometryConfig: { ...config, draco: this.options.draco }
314
- })), (0, geometry_definitions_1.GEOMETRY_DEFINITION)());
315
- if (this.layersHasTexture === false) {
316
- this.layers0.store.defaultGeometrySchema.ordering =
317
- this.layers0.store.defaultGeometrySchema.ordering.filter((attribute) => attribute !== 'uv0');
318
- }
319
- await this._writeLayers0();
320
- (0, create_scene_server_path_1.createSceneServerPath)(tilesetName, this.layers0, tilesetPath);
321
- for (const filePath of this.compressList || []) {
322
- await (0, compress_util_1.compressFileWithGzip)(filePath);
323
- await (0, file_utils_1.removeFile)(filePath);
324
- }
325
- await this.nodePages.save();
326
- await this.writeQueue.finalize();
327
- await this._createSlpk(tilesetPath);
328
- }
329
- /**
330
- * Form object of 3DSceneLayer https://github.com/Esri/i3s-spec/blob/master/docs/1.7/3DSceneLayer.cmn.md
331
- * @param tilesetName - Name of layer
332
- * @param sourceBoundingVolume - initialized bounding volume of the source root tile
333
- * @param boundingVolumeRegion - region bounding volume of the source root tile
334
- */
335
- _formLayers0(tilesetName, sourceBoundingVolume, boundingVolumeRegion) {
336
- if (!this.sourceTileset?.root) {
337
- return;
338
- }
339
- const fullExtent = (0, coordinate_converter_1.convertBoundingVolumeToI3SFullExtent)(sourceBoundingVolume);
340
- if (boundingVolumeRegion) {
341
- fullExtent.zmin = boundingVolumeRegion[4];
342
- fullExtent.zmax = boundingVolumeRegion[5];
343
- }
344
- const extent = [fullExtent.xmin, fullExtent.ymin, fullExtent.xmax, fullExtent.ymax];
345
- const layers0data = {
346
- version: `{${(0, uuid_1.v4)().toUpperCase()}}`,
347
- id: 0,
348
- name: tilesetName,
349
- href: './layers/0',
350
- store: {
351
- id: `{${(0, uuid_1.v4)().toUpperCase()}}`,
352
- extent
353
- },
354
- nodePages: {
355
- nodesPerPage: HARDCODED_NODES_PER_PAGE
356
- },
357
- compressGeometry: this.options.draco,
358
- fullExtent
359
- };
360
- this.layers0 = (0, json_map_transform_1.default)(layers0data, (0, layers_1.LAYERS)());
361
- }
362
- /**
363
- * Write 3DSceneLayer https://github.com/Esri/i3s-spec/blob/master/docs/1.7/3DSceneLayer.cmn.md in file
364
- */
365
- async _writeLayers0() {
366
- if (this.options.slpk) {
367
- await this.writeQueue.enqueue({
368
- archiveKey: '3dSceneLayer.json.gz',
369
- writePromise: () => (0, file_utils_1.writeFileForSlpk)(this.layers0Path, JSON.stringify(this.layers0), '3dSceneLayer.json')
370
- });
371
- }
372
- else {
373
- await this.writeQueue.enqueue({
374
- writePromise: () => (0, file_utils_1.writeFile)(this.layers0Path, JSON.stringify(this.layers0))
375
- });
376
- }
377
- }
378
- /**
379
- * Pack files into *.slpk archive
380
- * @param tilesetPath - Path to save file
381
- */
382
- async _createSlpk(tilesetPath) {
383
- if (this.options.slpk) {
384
- const slpkTilesetPath = (0, path_1.join)(tilesetPath, 'SceneServer', 'layers', '0');
385
- const slpkFileName = `${tilesetPath}.slpk`;
386
- await (0, compress_util_1.compressWithChildProcess)(slpkTilesetPath, slpkFileName, 0, '.', this.options.sevenZipExe);
387
- // TODO: `addFileToZip` corrupts archive so it can't be validated with windows i3s_converter.exe
388
- // const fileHash128Path = `${tilesetPath}/@specialIndexFileHASH128@`;
389
- // try {
390
- // await generateHash128FromZip(slpkFileName, fileHash128Path);
391
- // await addFileToZip(
392
- // tilesetPath,
393
- // '@specialIndexFileHASH128@',
394
- // slpkFileName,
395
- // this.options.sevenZipExe
396
- // );
397
- // } catch (error) {
398
- // if (error.code === FS_FILE_TOO_LARGE) {
399
- // console.warn(`${slpkFileName} file is too big to generate a hash`); // eslint-disable-line
400
- // } else {
401
- // console.error(error); // eslint-disable-line
402
- // }
403
- // }
404
- // All converted files are contained in slpk now they can be deleted
405
- try {
406
- await (0, file_utils_1.removeDir)(tilesetPath);
407
- }
408
- catch (e) {
409
- // do nothing
410
- }
411
- }
412
- }
413
- /**
414
- * Convert the specific 3DTiles tile to I3S nodes.
415
- * This is callback function for the traversal generic function
416
- * @param sourceTile - current 3DTiles tile JSON metadata
417
- * @param traversalProps - traversal properties calculated recursively
418
- * @returns - traversal properties for the child tiles
419
- */
420
- async convertTile(sourceTile, traversalProps) {
421
- const isTileset = (0, load_3d_tiles_1.isNestedTileset)(sourceTile);
422
- if (isTileset || sourceTile.type === 'empty') {
423
- if (isTileset) {
424
- if (sourceTile.id) {
425
- console.log(`[load]: ${sourceTile.id}`); // eslint-disable-line
426
- }
427
- await (0, load_3d_tiles_1.loadNestedTileset)(this.sourceTileset, sourceTile, this.loadOptions);
428
- }
429
- return traversalProps;
430
- }
431
- if (sourceTile.id) {
432
- console.log(`[convert]: ${sourceTile.id}`); // eslint-disable-line
433
- }
434
- const { parentNodes, transform } = traversalProps;
435
- let transformationMatrix = transform.clone();
436
- if (sourceTile.transform) {
437
- transformationMatrix = transformationMatrix.multiplyRight(sourceTile.transform);
438
- }
439
- const parentNode = parentNodes[0];
440
- const childNodes = await this._createNode(parentNode, sourceTile, transformationMatrix);
441
- await parentNode.addChildren(childNodes);
442
- const newTraversalProps = {
443
- transform: transformationMatrix,
444
- parentNodes: childNodes
445
- };
446
- return newTraversalProps;
447
- }
448
- /**
449
- * Do final action with nodes after the current node and all child nodes been converted.
450
- * @param conversionResults - array of conversion results of the current node
451
- * @param currentTraversalProps - traversal properties of the current node
452
- */
453
- async finalizeTile(conversionResults, currentTraversalProps) {
454
- for (const result of conversionResults) {
455
- for (const node of result.parentNodes) {
456
- await node.addNeighbors();
457
- }
458
- }
459
- for (const node of currentTraversalProps.parentNodes) {
460
- await node.save();
461
- }
462
- }
463
- /**
464
- * Convert tile to one or more I3S nodes
465
- * @param parentNode - 3DNodeIndexDocument of parent node
466
- * @param sourceTile - source 3DTile data
467
- * @param transformationMatrix - transformation matrix of the current tile, calculated recursively multiplying
468
- * transform of all parent tiles and transform of the current tile
469
- * @param level - tree level
470
- */
471
- async _createNode(parentNode, sourceTile, transformationMatrix) {
472
- this._checkAddRefinementTypeForTile(sourceTile);
473
- await this._updateTilesetOptions();
474
- let tileContent = null;
475
- try {
476
- tileContent = await (0, load_3d_tiles_1.loadTile3DContent)(this.sourceTileset, sourceTile, this.loadOptions);
477
- }
478
- catch (error) {
479
- console.log(`[warning]: Failed to load ${sourceTile.contentUrl}`);
480
- }
481
- const sourceBoundingVolume = (0, tiles_1.createBoundingVolume)(sourceTile.boundingVolume, transformationMatrix, null);
482
- let boundingVolumes = (0, coordinate_converter_1.createBoundingVolumes)(sourceBoundingVolume, this.geoidHeightModel);
483
- const propertyTable = (0, geometry_converter_1.getPropertyTable)(tileContent, this.options.metadataClass);
484
- if (propertyTable) {
485
- /*
486
- Call the convertion procedure even if the node attributes have been already created.
487
- We will append new attributes only in case the property table is updated.
488
- According to ver 1.9 (see https://github.com/Esri/i3s-spec/blob/master/docs/1.9/attributeStorageInfo.cmn.md):
489
- "The attributeStorageInfo object describes the structure of the binary attribute data resource of a layer, which is the same for every node in the layer."
490
- But the specification of ver 2.1 doesn't have such a requirement ("...the same for every node...")
491
- */
492
- this._convertPropertyTableToNodeAttributes(propertyTable);
493
- }
494
- const resourcesData = await this._convertResources(sourceTile, transformationMatrix, sourceBoundingVolume, tileContent, parentNode.inPageId, propertyTable);
495
- const nodes = [];
496
- const nodeIds = [];
497
- const nodesInPage = [];
498
- const emptyResources = {
499
- geometry: null,
500
- compressedGeometry: null,
501
- texture: null,
502
- hasUvRegions: false,
503
- sharedResources: null,
504
- meshMaterial: null,
505
- vertexCount: null,
506
- attributes: null,
507
- featureCount: null,
508
- boundingVolumes: null
509
- };
510
- for (const resources of resourcesData || [emptyResources]) {
511
- this.layersHasTexture = this.layersHasTexture || Boolean(resources.texture);
512
- if (this.generateBoundingVolumes && resources.boundingVolumes) {
513
- boundingVolumes = resources.boundingVolumes;
514
- }
515
- const lodSelection = (0, lod_conversion_utils_1.convertGeometricErrorToScreenThreshold)(sourceTile, boundingVolumes);
516
- const maxScreenThresholdSQ = lodSelection.find((val) => val.metricType === 'maxScreenThresholdSQ') || { maxError: 0 };
517
- const nodeInPage = await this._updateNodeInNodePages(maxScreenThresholdSQ, boundingVolumes, sourceTile, parentNode.inPageId, resources);
518
- const nodeData = await node_index_document_1.NodeIndexDocument.createNodeIndexDocument(parentNode, boundingVolumes, lodSelection, nodeInPage, resources);
519
- const node = await new node_index_document_1.NodeIndexDocument(nodeInPage.index, this).addData(nodeData);
520
- nodes.push(node);
521
- if (nodeInPage.mesh) {
522
- await this._writeResources(resources, node.id);
523
- }
524
- if (this.validate) {
525
- this.boundingVolumeWarnings = (0, node_debug_1.validateNodeBoundingVolumes)(nodeData);
526
- if (this.boundingVolumeWarnings && this.boundingVolumeWarnings.length) {
527
- console.warn('Bounding Volume Warnings: ', ...this.boundingVolumeWarnings); //eslint-disable-line
528
- }
529
- }
530
- nodeIds.push(nodeInPage.index);
531
- nodesInPage.push(nodeInPage);
532
- }
533
- return nodes;
534
- }
535
- /**
536
- * Convert tile to one or more I3S nodes
537
- * @param sourceTile - source tile (3DTile)
538
- * @param transformationMatrix - transformation matrix of the current tile, calculated recursively multiplying
539
- * transform of all parent tiles and transform of the current tile
540
- * @param boundingVolume - initialized bounding volume of the source tile
541
- * @param tileContent - content of the source tile
542
- * @param parentId - id of parent node in node pages
543
- * @param propertyTable - batch table from b3dm / feature properties from EXT_FEATURE_METADATA, EXT_MESH_FEATURES or EXT_STRUCTURAL_METADATA
544
- * @returns - converted node resources
545
- */
546
- async _convertResources(sourceTile, transformationMatrix, boundingVolume, tileContent, parentId, propertyTable) {
547
- if (!this.isContentSupported(sourceTile) || !tileContent) {
548
- return null;
549
- }
550
- const draftObb = {
551
- center: [],
552
- halfSize: [],
553
- quaternion: []
554
- };
555
- const resourcesData = await (0, geometry_converter_1.default)(tileContent, transformationMatrix, boundingVolume, async () => (await this.nodePages.push({ index: 0, obb: draftObb }, parentId)).index, propertyTable, this.featuresHashArray, this.layers0?.attributeStorageInfo, this.options.draco, this.generateBoundingVolumes, this.options.mergeMaterials, this.geoidHeightModel, this.loadOptions.modules, this.options.metadataClass);
556
- return resourcesData;
557
- }
558
- /**
559
- * Update node object (https://github.com/Esri/i3s-spec/blob/master/docs/1.7/node.cmn.md)
560
- * in node pages (https://github.com/Esri/i3s-spec/blob/master/docs/1.7/nodePage.cmn.md)
561
- * @param maxScreenThresholdSQ - Level of Details (LOD) metric
562
- * @param boundingVolumes - Bounding volumes
563
- * @param sourceTile - source tile (3DTile)
564
- * @param parentId - id of parent node in node pages
565
- * @param resources - the node resources data
566
- * @param resources.meshMaterial - PBR-like material object
567
- * @param resources.texture - texture image
568
- * @param resources.vertexCount - number of vertices in geometry
569
- * @param resources.featureCount - number of features
570
- * @param resources.geometry - Uint8Array with geometry attributes
571
- * @return the node object in node pages
572
- */
573
- async _updateNodeInNodePages(maxScreenThresholdSQ, boundingVolumes, sourceTile, parentId, resources) {
574
- const { meshMaterial, texture, vertexCount, featureCount, geometry, hasUvRegions } = resources;
575
- const nodeInPage = {
576
- index: 0,
577
- lodThreshold: maxScreenThresholdSQ.maxError,
578
- obb: boundingVolumes.obb,
579
- children: []
580
- };
581
- if (geometry && this.isContentSupported(sourceTile)) {
582
- nodeInPage.mesh = {
583
- geometry: {
584
- definition: this.findOrCreateGeometryDefinition(Boolean(texture), hasUvRegions),
585
- resource: 0
586
- },
587
- attribute: {
588
- resource: 0
589
- },
590
- material: {
591
- definition: 0
592
- }
593
- };
594
- }
595
- let nodeId = resources.nodeId;
596
- let node;
597
- if (!nodeId) {
598
- node = await this.nodePages.push(nodeInPage, parentId);
599
- }
600
- else {
601
- node = await this.nodePages.getNodeById(nodeId);
602
- }
603
- if (!nodeInPage.mesh) {
604
- console.log(`[warning]: node ${node.index} is created with empty content`);
605
- }
606
- node_pages_1.default.updateAll(node, nodeInPage);
607
- if (meshMaterial) {
608
- node_pages_1.default.updateMaterialByNodeId(node, this._findOrCreateMaterial(meshMaterial));
609
- }
610
- if (texture) {
611
- const texelCountHint = texture.image.height * texture.image.width;
612
- node_pages_1.default.updateTexelCountHintByNodeId(node, texelCountHint);
613
- }
614
- if (vertexCount) {
615
- this.vertexCounter += vertexCount;
616
- node_pages_1.default.updateVertexCountByNodeId(node, vertexCount);
617
- }
618
- node_pages_1.default.updateNodeAttributeByNodeId(node);
619
- if (featureCount) {
620
- node_pages_1.default.updateFeatureCountByNodeId(node, featureCount);
621
- }
622
- this.nodePages.saveNode(node);
623
- return node;
624
- }
625
- /**
626
- * Write node resources in files
627
- * @param resources - source tile (3DTile)
628
- * @param resources.geometry - Uint8Array with geometry attributes
629
- * @param resources.compressedGeometry - Uint8Array with compressed (draco) geometry
630
- * @param resources.texture - texture image
631
- * @param resources.sharedResources - shared resource data object
632
- * @param resources.attributes - feature attributes
633
- * @return {Promise<void>}
634
- */
635
- async _writeResources(resources, nodePath) {
636
- const { geometry: geometryBuffer, compressedGeometry, texture, sharedResources, attributes } = resources;
637
- const childPath = (0, path_1.join)(this.layers0Path, 'nodes', nodePath);
638
- const slpkChildPath = (0, path_1.join)('nodes', nodePath);
639
- await this._writeGeometries(geometryBuffer, compressedGeometry, childPath, slpkChildPath);
640
- await this._writeShared(sharedResources, childPath, slpkChildPath, nodePath);
641
- await this._writeTexture(texture, childPath, slpkChildPath);
642
- await this._writeAttributes(attributes, childPath, slpkChildPath);
643
- }
644
- /**
645
- * Write non-compressed and compressed geometries in files
646
- * @param geometryBuffer - Uint8Array with geometry attributes
647
- * @param compressedGeometry - Uint8Array with compressed (draco) geometry
648
- * @param childPath - a child path to write resources
649
- * @param slpkChildPath - resource path inside *slpk file
650
- */
651
- async _writeGeometries(geometryBuffer, compressedGeometry, childPath, slpkChildPath) {
652
- if (this.options.slpk) {
653
- const slpkGeometryPath = (0, path_1.join)(childPath, 'geometries');
654
- await this.writeQueue.enqueue({
655
- archiveKey: `${slpkChildPath}/geometries/0.bin.gz`,
656
- writePromise: () => (0, file_utils_1.writeFileForSlpk)(slpkGeometryPath, geometryBuffer, '0.bin')
657
- });
658
- }
659
- else {
660
- const geometryPath = (0, path_1.join)(childPath, 'geometries/0/');
661
- await this.writeQueue.enqueue({
662
- writePromise: () => (0, file_utils_1.writeFile)(geometryPath, geometryBuffer, 'index.bin')
663
- });
664
- }
665
- if (this.options.draco) {
666
- if (this.options.slpk) {
667
- const slpkCompressedGeometryPath = (0, path_1.join)(childPath, 'geometries');
668
- await this.writeQueue.enqueue({
669
- archiveKey: `${slpkChildPath}/geometries/1.bin.gz`,
670
- writePromise: () => (0, file_utils_1.writeFileForSlpk)(slpkCompressedGeometryPath, compressedGeometry, '1.bin')
671
- });
672
- }
673
- else {
674
- const compressedGeometryPath = (0, path_1.join)(childPath, 'geometries/1/');
675
- await this.writeQueue.enqueue({
676
- writePromise: () => (0, file_utils_1.writeFile)(compressedGeometryPath, compressedGeometry, 'index.bin')
677
- });
678
- }
679
- }
680
- }
681
- /**
682
- * Write shared resources in a file
683
- * @param sharedResources - shared resource data object
684
- * @param childPath - a child path to write resources
685
- * @param slpkChildPath - resource path inside *slpk file
686
- * @param nodePath - a node path
687
- */
688
- async _writeShared(sharedResources, childPath, slpkChildPath, nodePath) {
689
- if (!sharedResources) {
690
- return;
691
- }
692
- sharedResources.nodePath = nodePath;
693
- const sharedData = (0, json_map_transform_1.default)(sharedResources, (0, shared_resources_1.SHARED_RESOURCES)());
694
- const sharedDataStr = JSON.stringify(sharedData);
695
- if (this.options.slpk) {
696
- const slpkSharedPath = (0, path_1.join)(childPath, 'shared');
697
- await this.writeQueue.enqueue({
698
- archiveKey: `${slpkChildPath}/shared/sharedResource.json.gz`,
699
- writePromise: () => (0, file_utils_1.writeFileForSlpk)(slpkSharedPath, sharedDataStr, 'sharedResource.json')
700
- });
701
- }
702
- else {
703
- const sharedPath = (0, path_1.join)(childPath, 'shared/');
704
- await this.writeQueue.enqueue({ writePromise: () => (0, file_utils_1.writeFile)(sharedPath, sharedDataStr) });
705
- }
706
- }
707
- /**
708
- * Generates textures based on texture mime type and fill in textureSetDefinitions data.
709
- * @param texture - the texture image
710
- * @param childPath - a child path to write resources
711
- * @param slpkChildPath - the resource path inside *slpk file
712
- */
713
- async _writeTexture(texture, childPath, slpkChildPath) {
714
- if (texture) {
715
- const format = this._getFormatByMimeType(texture?.mimeType);
716
- const formats = [];
717
- const textureData = texture.bufferView.data;
718
- switch (format) {
719
- case 'jpg':
720
- case 'png': {
721
- formats.push({ name: '0', format });
722
- await this.writeTextureFile(textureData, '0', format, childPath, slpkChildPath);
723
- if (this.generateTextures) {
724
- formats.push({ name: '1', format: 'ktx2' });
725
- // For Node.js texture.image.data is type of Buffer
726
- const copyArrayBuffer = texture.image.data.subarray();
727
- const arrayToEncode = new Uint8Array(copyArrayBuffer);
728
- const ktx2TextureData = (0, core_1.encode)({ ...texture.image, data: arrayToEncode }, textures_1.KTX2BasisWriterWorker, {
729
- ...textures_1.KTX2BasisWriterWorker.options,
730
- ['ktx2-basis-writer']: {
731
- // We need to load local fs workers because nodejs can't load workers from the Internet
732
- workerUrl: './modules/textures/dist/ktx2-basis-writer-worker-node.js'
733
- },
734
- reuseWorkers: true,
735
- _nodeWorkers: true,
736
- useLocalLibraries: true
737
- });
738
- await this.writeTextureFile(ktx2TextureData, '1', 'ktx2', childPath, slpkChildPath);
739
- }
740
- break;
741
- }
742
- case 'ktx2': {
743
- formats.push({ name: '1', format });
744
- await this.writeTextureFile(textureData, '1', format, childPath, slpkChildPath);
745
- if (this.generateTextures) {
746
- formats.push({ name: '0', format: 'jpg' });
747
- const decodedFromKTX2TextureData = (0, core_1.encode)(texture.image.data[0], images_1.ImageWriter);
748
- await this.writeTextureFile(decodedFromKTX2TextureData, '0', 'jpg', childPath, slpkChildPath);
749
- }
750
- }
751
- }
752
- if (!this.layers0.textureSetDefinitions.length) {
753
- this.layers0.textureSetDefinitions.push({ formats });
754
- this.layers0.textureSetDefinitions.push({ formats, atlas: true });
755
- }
756
- }
757
- }
758
- /**
759
- * Write the texture image in a file
760
- * @param textureData
761
- * @param name
762
- * @param format
763
- * @param childPath
764
- * @param slpkChildPath
765
- */
766
- async writeTextureFile(textureData, name, format, childPath, slpkChildPath) {
767
- if (this.options.slpk) {
768
- const slpkTexturePath = (0, path_1.join)(childPath, 'textures');
769
- const compress = false;
770
- await this.writeQueue.enqueue({
771
- archiveKey: `${slpkChildPath}/textures/${name}.${format}`,
772
- writePromise: () => (0, file_utils_1.writeFileForSlpk)(slpkTexturePath, textureData, `${name}.${format}`, compress)
773
- });
774
- }
775
- else {
776
- const texturePath = (0, path_1.join)(childPath, `textures/${name}/`);
777
- await this.writeQueue.enqueue({
778
- writePromise: () => (0, file_utils_1.writeFile)(texturePath, textureData, `index.${format}`)
779
- });
780
- }
781
- }
782
- /**
783
- * Write feature attributes in files
784
- * @param attributes - feature attributes
785
- * @param childPath - a child path to write resources
786
- * @param slpkChildPath - the resource path inside *slpk file
787
- */
788
- async _writeAttributes(attributes = [], childPath, slpkChildPath) {
789
- if (attributes?.length && this.layers0?.attributeStorageInfo?.length) {
790
- const minimumLength = attributes.length < this.layers0.attributeStorageInfo.length
791
- ? attributes.length
792
- : this.layers0.attributeStorageInfo.length;
793
- for (let index = 0; index < minimumLength; index++) {
794
- const folderName = this.layers0.attributeStorageInfo[index].key;
795
- const fileBuffer = new Uint8Array(attributes[index]);
796
- if (this.options.slpk) {
797
- const slpkAttributesPath = (0, path_1.join)(childPath, 'attributes', folderName);
798
- await this.writeQueue.enqueue({
799
- archiveKey: `${slpkChildPath}/attributes/${folderName}.bin.gz`,
800
- writePromise: () => (0, file_utils_1.writeFileForSlpk)(slpkAttributesPath, fileBuffer, '0.bin')
801
- });
802
- }
803
- else {
804
- const attributesPath = (0, path_1.join)(childPath, `attributes/${folderName}/0`);
805
- await this.writeQueue.enqueue({
806
- writePromise: () => (0, file_utils_1.writeFile)(attributesPath, fileBuffer, 'index.bin')
807
- });
808
- }
809
- }
810
- }
811
- }
812
- /**
813
- * Return file format by its MIME type
814
- * @param mimeType - feature attributes
815
- */
816
- _getFormatByMimeType(mimeType) {
817
- switch (mimeType) {
818
- case 'image/jpeg':
819
- return 'jpg';
820
- case 'image/png':
821
- return 'png';
822
- case 'image/ktx2':
823
- return 'ktx2';
824
- default:
825
- return 'jpg';
826
- }
827
- }
828
- /**
829
- * Find or create material in materialDefinitions array
830
- * @param material - end-to-end index of the node
831
- * @return material id
832
- */
833
- _findOrCreateMaterial(material) {
834
- const hash = (0, md5_1.default)(JSON.stringify(material));
835
- if (this.materialMap.has(hash)) {
836
- return this.materialMap.get(hash) || 0;
837
- }
838
- const newMaterialId = this.materialDefinitions.push(material) - 1;
839
- this.materialMap.set(hash, newMaterialId);
840
- return newMaterialId;
841
- }
842
- /**
843
- * Get unique geometry configuration index
844
- * In the end of conversion configurations will be transformed to geometryDefinitions array
845
- * @param hasTexture
846
- * @param hasUvRegions
847
- * @returns
848
- */
849
- findOrCreateGeometryDefinition(hasTexture, hasUvRegions) {
850
- const geometryConfig = { hasTexture, hasUvRegions };
851
- const hash = (0, md5_1.default)(JSON.stringify(geometryConfig));
852
- if (this.geometryMap.has(hash)) {
853
- return this.geometryMap.get(hash) || 0;
854
- }
855
- const newGeometryId = this.geometryConfigs.push(geometryConfig) - 1;
856
- this.geometryMap.set(hash, newGeometryId);
857
- return newGeometryId;
858
- }
859
- /**
860
- * Do conversion of 3DTiles property table to I3s node attributes.
861
- * @param propertyTable - Table with layer meta data.
862
- */
863
- _convertPropertyTableToNodeAttributes(propertyTable) {
864
- let attributeIndex = 0;
865
- const propertyTableWithObjectId = {
866
- OBJECTID: [0],
867
- ...propertyTable
868
- };
869
- for (const key in propertyTableWithObjectId) {
870
- /*
871
- We will append new attributes only in case the property table is updated.
872
- According to ver 1.9 (see https://github.com/Esri/i3s-spec/blob/master/docs/1.9/attributeStorageInfo.cmn.md):
873
- "The attributeStorageInfo object describes the structure of the binary attribute data resource of a layer, which is the same for every node in the layer."
874
- But the specification of ver 2.1 doesn't have such a requirement ("...the same for every node...")
875
- */
876
- const found = this.layers0.attributeStorageInfo.find((element) => element.name === key);
877
- if (!found) {
878
- const firstAttribute = propertyTableWithObjectId[key][0];
879
- const attributeType = (0, feature_attributes_1.getAttributeType)(key, firstAttribute);
880
- const storageAttribute = (0, feature_attributes_1.createdStorageAttribute)(attributeIndex, key, attributeType);
881
- const fieldAttributeType = (0, feature_attributes_1.getFieldAttributeType)(attributeType);
882
- const fieldAttribute = (0, feature_attributes_1.createFieldAttribute)(key, fieldAttributeType);
883
- const popupInfo = (0, feature_attributes_1.createPopupInfo)(propertyTableWithObjectId);
884
- this.layers0.attributeStorageInfo.push(storageAttribute);
885
- this.layers0.fields.push(fieldAttribute);
886
- this.layers0.popupInfo = popupInfo;
887
- this.layers0.layerType = _3D_OBJECT_LAYER_TYPE;
888
- }
889
- attributeIndex += 1;
890
- }
891
- }
892
- /**
893
- * Print statistics in the end of conversion
894
- * @param params - output files data
895
- */
896
- async _finishConversion(params) {
897
- const { tilesCount, tilesWithAddRefineCount } = this.refinementCounter;
898
- const addRefinementPercentage = tilesWithAddRefineCount
899
- ? (tilesWithAddRefineCount / tilesCount) * 100
900
- : 0;
901
- const filesSize = await (0, statistic_utills_1.calculateFilesSize)(params);
902
- const diff = process_1.default.hrtime(this.conversionStartTime);
903
- const conversionTime = (0, statistic_utills_1.timeConverter)(diff);
904
- console.log(`------------------------------------------------`); // eslint-disable-line no-undef, no-console
905
- console.log(`Finishing conversion of ${_3D_TILES}`); // eslint-disable-line no-undef, no-console
906
- console.log(`Total conversion time: ${conversionTime}`); // eslint-disable-line no-undef, no-console
907
- console.log(`Vertex count: `, this.vertexCounter); // eslint-disable-line no-undef, no-console
908
- console.log(`File(s) size: `, filesSize, ' bytes'); // eslint-disable-line no-undef, no-console
909
- console.log(`Percentage of tiles with "ADD" refinement type:`, addRefinementPercentage, '%'); // eslint-disable-line no-undef, no-console
910
- console.log(`------------------------------------------------`); // eslint-disable-line no-undef, no-console
911
- }
912
- /**
913
- * Fetch preload options for ION tileset
914
- */
915
- async _fetchPreloadOptions() {
916
- if (!this.Loader.preload) {
917
- return {};
918
- }
919
- const options = {
920
- 'cesium-ion': { accessToken: this.options.token || ION_DEFAULT_TOKEN }
921
- };
922
- const preloadOptions = await this.Loader.preload(this.options.inputUrl, options);
923
- this.refreshTokenTime = process_1.default.hrtime();
924
- return { ...options, ...preloadOptions };
925
- }
926
- /**
927
- * Update options of source tileset
928
- */
929
- async _updateTilesetOptions() {
930
- const diff = process_1.default.hrtime(this.refreshTokenTime);
931
- if (diff[0] < REFRESH_TOKEN_TIMEOUT) {
932
- return;
933
- }
934
- this.refreshTokenTime = process_1.default.hrtime();
935
- const preloadOptions = await this._fetchPreloadOptions();
936
- if (preloadOptions.headers) {
937
- this.loadOptions.fetch = {
938
- ...this.loadOptions.fetch,
939
- headers: preloadOptions.headers
940
- };
941
- console.log('Authorization Bearer token has been updated'); // eslint-disable-line no-undef, no-console
942
- }
943
- }
944
- /** Do calculations of all tiles and tiles with "ADD" type of refinement.
945
- * @param tile
946
- */
947
- _checkAddRefinementTypeForTile(tile) {
948
- const ADD_TILE_REFINEMENT = 1;
949
- if (tile.refine === ADD_TILE_REFINEMENT) {
950
- this.refinementCounter.tilesWithAddRefineCount += 1;
951
- console.warn('This tile uses "ADD" type of refinement'); // eslint-disable-line
952
- }
953
- this.refinementCounter.tilesCount += 1;
954
- }
955
- /**
956
- * Check if the tile's content format is supported by the converter
957
- * @param sourceTile
958
- * @returns
959
- */
960
- isContentSupported(sourceTile) {
961
- return ['b3dm', 'glTF', 'scenegraph'].includes(sourceTile.type || '');
962
- }
963
- }
964
- exports.default = I3SConverter;