kepler.gl 3.1.9 → 3.2.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.
Files changed (98) hide show
  1. package/README.md +21 -21
  2. package/dist/src/components/src/common/link-renderer.d.ts +6 -2
  3. package/dist/src/components/src/index.d.ts +1 -0
  4. package/dist/src/components/src/map-container.d.ts +1 -0
  5. package/dist/src/components/src/plot-container.d.ts +1 -0
  6. package/dist/src/components/src/side-panel/layer-manager.d.ts +1 -0
  7. package/dist/src/duckdb/src/index.d.ts +1 -0
  8. package/dist/src/duckdb/src/processors/data-processor.d.ts +8 -0
  9. package/dist/src/utils/src/data-utils.d.ts +1 -0
  10. package/package.json +2 -2
  11. package/src/actions/package.json +8 -8
  12. package/src/ai-assistant/package.json +7 -7
  13. package/src/cloud-providers/package.json +2 -2
  14. package/src/common-utils/package.json +3 -3
  15. package/src/components/dist/common/link-renderer.d.ts +6 -2
  16. package/src/components/dist/common/link-renderer.js +1 -1
  17. package/src/components/dist/index.d.ts +1 -0
  18. package/src/components/dist/index.js +44 -1
  19. package/src/components/dist/map-container.d.ts +1 -0
  20. package/src/components/dist/map-container.js +2 -1
  21. package/src/components/dist/modal-container.js +3 -3
  22. package/src/components/dist/modals/tilesets-modals/tileset-raster-form.js +38 -9
  23. package/src/components/dist/plot-container.d.ts +1 -0
  24. package/src/components/dist/plot-container.js +7 -4
  25. package/src/components/dist/side-panel/layer-manager.d.ts +1 -0
  26. package/src/components/dist/side-panel/layer-manager.js +5 -3
  27. package/src/components/dist/side-panel/layer-panel/color-scale-selector.js +16 -9
  28. package/src/components/dist/side-panel/layer-panel/raster-tile-layer-configurator.js +3 -3
  29. package/src/components/package.json +16 -16
  30. package/src/components/src/common/link-renderer.tsx +7 -2
  31. package/src/components/src/index.ts +8 -0
  32. package/src/components/src/map-container.tsx +4 -0
  33. package/src/components/src/modal-container.tsx +6 -2
  34. package/src/components/src/modals/tilesets-modals/tileset-raster-form.tsx +60 -4
  35. package/src/components/src/plot-container.tsx +9 -3
  36. package/src/components/src/side-panel/layer-manager.tsx +4 -2
  37. package/src/components/src/side-panel/layer-panel/color-scale-selector.tsx +22 -9
  38. package/src/components/src/side-panel/layer-panel/raster-tile-layer-configurator.tsx +2 -1
  39. package/src/constants/dist/default-settings.js +1 -1
  40. package/src/constants/node_modules/.cache/terser-webpack-plugin/content-v2/sha512/56/48/71c2c5381c01c063c164e15fa04b4b1f3faae02e6727f2f2daf483e15205feca68334bad721af0c6f31abb20b56d60504623838127764aab6883912e7f71 +1 -0
  41. package/src/constants/node_modules/.cache/terser-webpack-plugin/content-v2/sha512/e4/a7/04d0f645deb094cab21e70c2457f3b87750bee1dfea9b3ad1d4f746ae10ff013d960af8a46b1f87a10409989d6a352c21e29a8784a161a84f83a310955e2 +1 -0
  42. package/src/constants/node_modules/.cache/terser-webpack-plugin/index-v5/b2/5f/6daa692cacd1234803d6a5abefef0c49aae2d1403e4d8e53140088950a04 +2 -0
  43. package/src/constants/node_modules/.cache/terser-webpack-plugin/index-v5/bf/88/ea60c7619185516464a4ae21f51394966047bfdf190a5463de651c61931c +2 -0
  44. package/src/constants/package.json +2 -2
  45. package/src/constants/umd/keplergl.min.js +1 -1
  46. package/src/deckgl-arrow-layers/package.json +2 -2
  47. package/src/deckgl-layers/package.json +4 -4
  48. package/src/duckdb/dist/index.d.ts +1 -0
  49. package/src/duckdb/dist/index.js +24 -2
  50. package/src/duckdb/dist/processors/data-processor.d.ts +8 -0
  51. package/src/duckdb/dist/processors/data-processor.js +23 -1
  52. package/src/duckdb/package.json +6 -6
  53. package/src/duckdb/src/index.ts +5 -0
  54. package/src/duckdb/src/processors/data-processor.ts +25 -1
  55. package/src/effects/package.json +5 -5
  56. package/src/layers/dist/raster-tile/gpu-utils.d.ts +11 -10
  57. package/src/layers/dist/raster-tile/gpu-utils.js +7 -6
  58. package/src/layers/dist/raster-tile/image.d.ts +4 -0
  59. package/src/layers/dist/raster-tile/image.js +30 -16
  60. package/src/layers/dist/raster-tile/raster-tile-layer.d.ts +4 -0
  61. package/src/layers/dist/raster-tile/raster-tile-layer.js +18 -6
  62. package/src/layers/dist/raster-tile/request-throttle.d.ts +1 -1
  63. package/src/layers/dist/raster-tile/request-throttle.js +17 -16
  64. package/src/layers/dist/raster-tile/types.d.ts +16 -0
  65. package/src/layers/dist/raster-tile/types.js +1 -1
  66. package/src/layers/dist/raster-tile/url.d.ts +1 -1
  67. package/src/layers/dist/raster-tile/url.js +5 -4
  68. package/src/layers/package.json +9 -9
  69. package/src/layers/src/raster-tile/gpu-utils.ts +96 -78
  70. package/src/layers/src/raster-tile/image.ts +35 -9
  71. package/src/layers/src/raster-tile/raster-tile-layer.ts +23 -5
  72. package/src/layers/src/raster-tile/request-throttle.ts +10 -5
  73. package/src/layers/src/raster-tile/types.ts +19 -1
  74. package/src/layers/src/raster-tile/url.ts +10 -3
  75. package/src/localization/package.json +1 -1
  76. package/src/processors/package.json +7 -7
  77. package/src/reducers/package.json +16 -16
  78. package/src/schemas/package.json +7 -7
  79. package/src/styles/node_modules/.cache/terser-webpack-plugin/content-v2/sha512/21/4a/50f77069dbf70c6792dd82d4e149375a888e25ddddd3209e2db013be62091f79082291fa06eb518cccb111b09a778e2946c1fb7a27bb6e32f72da0ccf250 +1 -0
  80. package/src/styles/node_modules/.cache/terser-webpack-plugin/content-v2/sha512/41/e3/d873c0efff87c4438d34d09be942ff6877947926c2309949e1eea8a58722c9ebf1e9451b44f62996d3590f398318e2fd17f19613445810226c8354b7d746 +1 -0
  81. package/src/styles/node_modules/.cache/terser-webpack-plugin/index-v5/9d/95/efc8c9f0bdd09b105646cf5e34b067e621726ceeb296771ff7da9e2e03b4 +2 -0
  82. package/src/styles/node_modules/.cache/terser-webpack-plugin/index-v5/c5/b6/9c4bdb089acc30ac8a16a6b9b93898757537f7017b2f725ddb832b4614d6 +2 -0
  83. package/src/styles/package.json +2 -2
  84. package/src/styles/umd/keplergl.min.js +1 -1
  85. package/src/table/dist/tileset/vector-tile-utils.js +9 -5
  86. package/src/table/package.json +5 -5
  87. package/src/table/src/tileset/vector-tile-utils.ts +9 -6
  88. package/src/tasks/package.json +2 -2
  89. package/src/types/package.json +1 -1
  90. package/src/utils/dist/application-config.js +5 -5
  91. package/src/utils/dist/data-scale-utils.js +35 -16
  92. package/src/utils/dist/data-utils.d.ts +1 -0
  93. package/src/utils/dist/data-utils.js +2 -1
  94. package/src/utils/package.json +4 -4
  95. package/src/utils/src/application-config.ts +4 -6
  96. package/src/utils/src/data-scale-utils.ts +34 -21
  97. package/src/utils/src/data-utils.ts +1 -0
  98. package/umd/keplergl.min.js +1240 -1220
@@ -43,4 +43,4 @@ var BandCombination = exports.BandCombination = /*#__PURE__*/function (BandCombi
43
43
  /** Properties provided to getTileData by the deck.gl TileLayer */
44
44
  /** Properties passed into the deck.gl TileLayer getTileData callback */
45
45
  /** Required information on how to render each preset */
46
- //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"names":["BandCombination","exports"],"sources":["../../src/raster-tile/types.ts"],"sourcesContent":["// SPDX-License-Identifier: MIT\n// Copyright contributors to the kepler.gl project\n\nimport {_Tile2DHeader} from '@deck.gl/geo-layers/typed';\nimport {TypedArray} from '@loaders.gl/loader-utils/src/types';\nimport {Texture2DProps} from '@luma.gl/webgl';\n\nimport {KeplerTable as KeplerDataset} from '@kepler.gl/table';\nimport type {ColorMap, StacTypes} from '@kepler.gl/types';\n\nexport type STACObject = StacTypes.STACObject;\nexport type CompleteSTACItem = StacTypes.CompleteSTACItem;\nexport type CompleteSTACCollection = StacTypes.CompleteSTACCollection;\nexport type CompleteSTACObject = StacTypes.CompleteSTACObject;\n\ntype DataTypeOfTheBand = StacTypes.DataTypeOfTheBand;\n\nexport type CompleteSTACAssetLinks =\n  | CompleteSTACItem['assets']\n  | CompleteSTACCollection['item_assets'];\n\nexport enum BandCombination {\n  /** Render a single raster band. */\n  Single = 'single',\n\n  /** Render three bands together on the GPU.\n   * Note that this could be \"true color\" if the first band is red, the second is green, and the\n   * third is blue. But it could also be \"false color\" if the three bands chosen are not RGB.\n   */\n  Rgb = 'rgb',\n\n  /** Calculate the Normalized Difference Vegetation Index (NDVI) on two bands. */\n  NormalizedDifference = 'normalizedDifference',\n\n  /** Calculate the EVI on two bands. */\n  EnhancedVegetationIndex = 'enhancedVegetationIndex',\n\n  /** Calculate SAVI on two bands. */\n  SoilAdjustedVegetationIndex = 'soilAdjustedVegetationIndex',\n\n  /** Calculate MSAVI on two bands. */\n  ModifiedSoilAdjustedVegetationIndex = 'modifiedSoilAdjustedVegetationIndex'\n}\n\n/**\n * Identifiers of STAC assets to load\n *\n * These identifiers should match the keys of the STAC assets object. Refer to\n * https://github.com/radiantearth/stac-spec/blob/master/item-spec/item-spec.md#assets\n */\nexport type AssetIds = string[];\n\n/**\n * Indexes of bands within each asset that should be loaded.\n *\n * Each asset refers to a single Cloud-Optimized GeoTIFF file on disk. Each COG has a width, height,\n * and one or more bands.\n *\n * This array of indexes must be the same length as the list of AssetIds. The nth band index\n * describes the which index from the nth asset id to load.\n */\nexport type BandIndexes = number[];\n\n/**\n * An array of integers representing how bands should be reordered on the GPU. This allows for\n * faster changing of bands for single-asset raster data, such as NAIP. Will be null for multi-asset\n * STAC items.\n */\nexport type RenderBandIndexes = BandIndexes | null;\n\nexport type AssetRequestInfo = {\n  loadAssetIds: AssetIds;\n  loadBandIndexes: BandIndexes;\n  renderBandIndexes: RenderBandIndexes;\n};\n\n/** User-provided information for how to render a preset */\nexport type PresetOption = {\n  singleBand?: {\n    assetId: string;\n    bandIndex?: number;\n  };\n};\n\nexport type DataSourceParams = AssetRequestInfo & {\n  minZoom: number;\n  maxZoom: number;\n  minPixelValue: number;\n  maxPixelValue: number;\n  minCategoricalBandValue?: number;\n  maxCategoricalBandValue?: number;\n  globalBounds: number[] | null;\n  dataType: DataTypeOfTheBand;\n};\n\nexport type CategoricalColormapOptions = {\n  colorMap?: ColorMap;\n  minValue?: number;\n  maxValue?: number;\n};\n\nexport type ExtendedKeplerSTAC = {rasterTileServerUrls?: []};\n\n/**\n * Custom fields we pass on to the getTileData callback\n * Anything required to know _what data to load_ should be passed in here.\n */\nexport type GetTileDataCustomProps = Pick<AssetRequestInfo, 'loadAssetIds' | 'loadBandIndexes'> & {\n  stac: CompleteSTACObject & ExtendedKeplerSTAC;\n  colormapId: string;\n  categoricalColorMap: ColorMap;\n  minCategoricalBandValue?: number;\n  maxCategoricalBandValue?: number;\n\n  /** If true, fetch terrain data as well as texture data. */\n  shouldLoadTerrain: boolean;\n  globalBounds: number[] | null;\n  useSTACSearching: boolean;\n  stacSearchProvider: string;\n  startDate: string;\n  endDate: string;\n\n  /* Stringified JSON representing a STAC API Search Query the backend should request from a server. */\n  _stacQuery?: string;\n};\n\n/** Properties provided to getTileData by the deck.gl TileLayer */\nexport interface GetTileDataDefaultProps {\n  index: {x: number; y: number; z: number};\n  url?: string;\n\n  /** Bounding box of the current web mercator tile. */\n  bbox: {west: number; north: number; east: number; south: number};\n\n  /** AbortSignal used for cancelling requests */\n  signal: AbortSignal;\n}\n\n/** Properties passed into the deck.gl TileLayer getTileData callback */\nexport type GetTileDataProps = GetTileDataCustomProps & GetTileDataDefaultProps;\n\nexport type GetTileDataOutput = any | null;\n\nexport type RenderSubLayersCustomProps = ColorRescaling &\n  Pick<AssetRequestInfo, 'renderBandIndexes'> & {\n    opacity: number;\n    linearRescaling: boolean;\n    linearRescalingFactor: [number, number];\n    nonLinearRescaling: boolean;\n    minPixelValue: number;\n    maxPixelValue: number;\n    bandCombination: BandCombination;\n    filterEnabled: boolean;\n    filterRange: [number, number];\n    currentTime: number;\n    dataType: DataTypeOfTheBand;\n    minCategoricalBandValue?: number;\n    maxCategoricalBandValue?: number;\n    hasCategoricalColorMap: boolean;\n    hasShadowEffect?: boolean;\n  };\n\nexport interface RenderSubLayersDefaultProps {\n  id: string;\n  tile: _Tile2DHeader<GetTileDataOutput>;\n  data: GetTileDataOutput;\n}\n\nexport type RenderSubLayersProps = RenderSubLayersCustomProps & RenderSubLayersDefaultProps;\n\n/** Required information on how to render each preset */\nexport interface PresetData {\n  label: string;\n  id: string;\n  bandCombination: BandCombination;\n  commonNames?: string[];\n  description?: string;\n  infoUrl?: string;\n}\n\nexport interface ColorRescaling {\n  gammaContrastFactor: number;\n  sigmoidalContrastFactor: number;\n  sigmoidalBiasFactor: number;\n  saturationValue: number;\n}\n\nexport interface ConfigOption {\n  label: string;\n  id: string;\n}\n\nexport type ColormapImageData = Texture2DProps;\n\nexport interface ImageData {\n  imageBands: Texture2DProps[] | null;\n  imageColormap?: Texture2DProps | null;\n  imageMask?: Texture2DProps | null;\n  imagePan?: Texture2DProps | null;\n  imageRgba?: Texture2DProps | null;\n}\n\nexport type AssetRequestData = {\n  url: string;\n  rasterServerUrl: string;\n  options: RequestInit;\n  useMask: boolean;\n  /** Pass this property through the request to pick specific bands from the response */\n  responseRequiredBandIndices?: number[] | null;\n};\n\nexport type NPYLoaderDataTypes =\n  | Uint8Array\n  | Uint8ClampedArray\n  | Int8Array\n  | Uint16Array\n  | Int16Array\n  | Uint32Array\n  | Int32Array\n  | Float32Array\n  | Float64Array;\n\nexport interface NPYLoaderResponse {\n  data: NPYLoaderDataTypes;\n  header: {\n    descr: string;\n    shape: number[];\n  };\n}\n\nexport interface TerrainData {\n  header: {\n    vertexCount: number;\n    boundingBox: [[number, number, number], [number, number, number]];\n  };\n  mode: 4;\n  indices: {\n    value: TypedArray;\n    size: 1;\n  };\n  attributes: {\n    POSITION: {\n      value: TypedArray;\n      size: 3;\n    };\n    TEXCOORD_0: {\n      value: TypedArray;\n      size: 2;\n    };\n  };\n}\n\nexport type KeplerRasterDataset = KeplerDataset & {\n  metadata: CompleteSTACObject;\n};\n\nexport interface Tile2DHeader {\n  content?: {\n    terrain?: TerrainData;\n    images: GetTileDataOutput;\n  } | null;\n  get data(): GetTileDataOutput;\n}\n"],"mappings":";;;;;;AAAA;AACA;AAAA,IAoBYA,eAAe,GAAAC,OAAA,CAAAD,eAAA,0BAAfA,eAAe;EAAfA,eAAe;EAAfA,eAAe;EAAfA,eAAe;EAAfA,eAAe;EAAfA,eAAe;EAAfA,eAAe;EAAA,OAAfA,eAAe;AAAA;AAuB3B;AACA;AACA;AACA;AACA;AACA;AAGA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AAGA;AACA;AACA;AACA;AACA;AASA;AA2BA;AACA;AACA;AACA;AAoBA;AAYA;AAgCA","ignoreList":[]}
46
+ //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"names":["BandCombination","exports"],"sources":["../../src/raster-tile/types.ts"],"sourcesContent":["// SPDX-License-Identifier: MIT\n// Copyright contributors to the kepler.gl project\n\nimport {_Tile2DHeader} from '@deck.gl/geo-layers/typed';\nimport {TypedArray} from '@loaders.gl/loader-utils/src/types';\nimport {Texture2DProps} from '@luma.gl/webgl';\n\nimport {KeplerTable as KeplerDataset} from '@kepler.gl/table';\nimport type {ColorMap, StacTypes} from '@kepler.gl/types';\n\nexport type STACObject = StacTypes.STACObject;\nexport type CompleteSTACItem = StacTypes.CompleteSTACItem;\nexport type CompleteSTACCollection = StacTypes.CompleteSTACCollection;\nexport type CompleteSTACObject = StacTypes.CompleteSTACObject;\n\ntype DataTypeOfTheBand = StacTypes.DataTypeOfTheBand;\n\nexport type CompleteSTACAssetLinks =\n  | CompleteSTACItem['assets']\n  | CompleteSTACCollection['item_assets'];\n\nexport enum BandCombination {\n  /** Render a single raster band. */\n  Single = 'single',\n\n  /** Render three bands together on the GPU.\n   * Note that this could be \"true color\" if the first band is red, the second is green, and the\n   * third is blue. But it could also be \"false color\" if the three bands chosen are not RGB.\n   */\n  Rgb = 'rgb',\n\n  /** Calculate the Normalized Difference Vegetation Index (NDVI) on two bands. */\n  NormalizedDifference = 'normalizedDifference',\n\n  /** Calculate the EVI on two bands. */\n  EnhancedVegetationIndex = 'enhancedVegetationIndex',\n\n  /** Calculate SAVI on two bands. */\n  SoilAdjustedVegetationIndex = 'soilAdjustedVegetationIndex',\n\n  /** Calculate MSAVI on two bands. */\n  ModifiedSoilAdjustedVegetationIndex = 'modifiedSoilAdjustedVegetationIndex'\n}\n\n/**\n * Identifiers of STAC assets to load\n *\n * These identifiers should match the keys of the STAC assets object. Refer to\n * https://github.com/radiantearth/stac-spec/blob/master/item-spec/item-spec.md#assets\n */\nexport type AssetIds = string[];\n\n/**\n * Indexes of bands within each asset that should be loaded.\n *\n * Each asset refers to a single Cloud-Optimized GeoTIFF file on disk. Each COG has a width, height,\n * and one or more bands.\n *\n * This array of indexes must be the same length as the list of AssetIds. The nth band index\n * describes the which index from the nth asset id to load.\n */\nexport type BandIndexes = number[];\n\n/**\n * An array of integers representing how bands should be reordered on the GPU. This allows for\n * faster changing of bands for single-asset raster data, such as NAIP. Will be null for multi-asset\n * STAC items.\n */\nexport type RenderBandIndexes = BandIndexes | null;\n\nexport type AssetRequestInfo = {\n  loadAssetIds: AssetIds;\n  loadBandIndexes: BandIndexes;\n  renderBandIndexes: RenderBandIndexes;\n};\n\n/** User-provided information for how to render a preset */\nexport type PresetOption = {\n  singleBand?: {\n    assetId: string;\n    bandIndex?: number;\n  };\n};\n\nexport type DataSourceParams = AssetRequestInfo & {\n  minZoom: number;\n  maxZoom: number;\n  minPixelValue: number;\n  maxPixelValue: number;\n  minCategoricalBandValue?: number;\n  maxCategoricalBandValue?: number;\n  globalBounds: number[] | null;\n  dataType: DataTypeOfTheBand;\n};\n\nexport type CategoricalColormapOptions = {\n  colorMap?: ColorMap;\n  minValue?: number;\n  maxValue?: number;\n};\n\nexport type ExtendedKeplerSTAC = {\n  rasterTileServerUrls?: [];\n  /** Optional per-layer override for max retries when fetching raster data */\n  rasterServerMaxRetries?: number;\n  /** Optional per-layer override for retry delay between attempts (ms) */\n  rasterServerRetryDelay?: number;\n  /** Optional per-layer override for which server errors are retried */\n  rasterServerServerErrorsToRetry?: number[];\n  /** Optional per-layer override for max concurrent requests per server */\n  rasterServerMaxPerServerRequests?: number;\n};\n\n/**\n * Custom fields we pass on to the getTileData callback\n * Anything required to know _what data to load_ should be passed in here.\n */\nexport type GetTileDataCustomProps = Pick<AssetRequestInfo, 'loadAssetIds' | 'loadBandIndexes'> & {\n  stac: CompleteSTACObject & ExtendedKeplerSTAC;\n  colormapId: string;\n  categoricalColorMap: ColorMap;\n  minCategoricalBandValue?: number;\n  maxCategoricalBandValue?: number;\n\n  /** If true, fetch terrain data as well as texture data. */\n  shouldLoadTerrain: boolean;\n  globalBounds: number[] | null;\n  useSTACSearching: boolean;\n  stacSearchProvider: string;\n  startDate: string;\n  endDate: string;\n\n  /* Stringified JSON representing a STAC API Search Query the backend should request from a server. */\n  _stacQuery?: string;\n};\n\n/** Properties provided to getTileData by the deck.gl TileLayer */\nexport interface GetTileDataDefaultProps {\n  index: {x: number; y: number; z: number};\n  url?: string;\n\n  /** Bounding box of the current web mercator tile. */\n  bbox: {west: number; north: number; east: number; south: number};\n\n  /** AbortSignal used for cancelling requests */\n  signal: AbortSignal;\n}\n\n/** Properties passed into the deck.gl TileLayer getTileData callback */\nexport type GetTileDataProps = GetTileDataCustomProps & GetTileDataDefaultProps;\n\nexport type GetTileDataOutput = any | null;\n\nexport type RenderSubLayersCustomProps = ColorRescaling &\n  Pick<AssetRequestInfo, 'renderBandIndexes'> & {\n    opacity: number;\n    linearRescaling: boolean;\n    linearRescalingFactor: [number, number];\n    nonLinearRescaling: boolean;\n    minPixelValue: number;\n    maxPixelValue: number;\n    bandCombination: BandCombination;\n    filterEnabled: boolean;\n    filterRange: [number, number];\n    currentTime: number;\n    dataType: DataTypeOfTheBand;\n    minCategoricalBandValue?: number;\n    maxCategoricalBandValue?: number;\n    hasCategoricalColorMap: boolean;\n    hasShadowEffect?: boolean;\n  };\n\nexport interface RenderSubLayersDefaultProps {\n  id: string;\n  tile: _Tile2DHeader<GetTileDataOutput>;\n  data: GetTileDataOutput;\n}\n\nexport type RenderSubLayersProps = RenderSubLayersCustomProps & RenderSubLayersDefaultProps;\n\n/** Required information on how to render each preset */\nexport interface PresetData {\n  label: string;\n  id: string;\n  bandCombination: BandCombination;\n  commonNames?: string[];\n  description?: string;\n  infoUrl?: string;\n}\n\nexport interface ColorRescaling {\n  gammaContrastFactor: number;\n  sigmoidalContrastFactor: number;\n  sigmoidalBiasFactor: number;\n  saturationValue: number;\n}\n\nexport interface ConfigOption {\n  label: string;\n  id: string;\n}\n\nexport type ColormapImageData = Texture2DProps;\n\nexport interface ImageData {\n  imageBands: Texture2DProps[] | null;\n  imageColormap?: Texture2DProps | null;\n  imageMask?: Texture2DProps | null;\n  imagePan?: Texture2DProps | null;\n  imageRgba?: Texture2DProps | null;\n}\n\nexport type AssetRequestData = {\n  url: string;\n  rasterServerUrl: string;\n  options: RequestInit;\n  useMask: boolean;\n  /** Pass this property through the request to pick specific bands from the response */\n  responseRequiredBandIndices?: number[] | null;\n  /** Optional per-request override for max retries when fetching NPY arrays */\n  rasterServerMaxRetries?: number;\n  /** Optional per-request override for retry delay between attempts (ms) */\n  rasterServerRetryDelay?: number;\n  /** Optional per-request override for which server errors are retried */\n  rasterServerServerErrorsToRetry?: number[];\n  /** Optional per-request override for max concurrent requests per server */\n  rasterServerMaxPerServerRequests?: number;\n};\n\nexport type NPYLoaderDataTypes =\n  | Uint8Array\n  | Uint8ClampedArray\n  | Int8Array\n  | Uint16Array\n  | Int16Array\n  | Uint32Array\n  | Int32Array\n  | Float32Array\n  | Float64Array;\n\nexport interface NPYLoaderResponse {\n  data: NPYLoaderDataTypes;\n  header: {\n    descr: string;\n    shape: number[];\n  };\n}\n\nexport interface TerrainData {\n  header: {\n    vertexCount: number;\n    boundingBox: [[number, number, number], [number, number, number]];\n  };\n  mode: 4;\n  indices: {\n    value: TypedArray;\n    size: 1;\n  };\n  attributes: {\n    POSITION: {\n      value: TypedArray;\n      size: 3;\n    };\n    TEXCOORD_0: {\n      value: TypedArray;\n      size: 2;\n    };\n  };\n}\n\nexport type KeplerRasterDataset = KeplerDataset & {\n  metadata: CompleteSTACObject;\n};\n\nexport interface Tile2DHeader {\n  content?: {\n    terrain?: TerrainData;\n    images: GetTileDataOutput;\n  } | null;\n  get data(): GetTileDataOutput;\n}\n"],"mappings":";;;;;;AAAA;AACA;AAAA,IAoBYA,eAAe,GAAAC,OAAA,CAAAD,eAAA,0BAAfA,eAAe;EAAfA,eAAe;EAAfA,eAAe;EAAfA,eAAe;EAAfA,eAAe;EAAfA,eAAe;EAAfA,eAAe;EAAA,OAAfA,eAAe;AAAA;AAuB3B;AACA;AACA;AACA;AACA;AACA;AAGA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AAGA;AACA;AACA;AACA;AACA;AASA;AAqCA;AACA;AACA;AACA;AAoBA;AAYA;AAgCA","ignoreList":[]}
@@ -8,7 +8,7 @@ export declare function getStacApiUrlParams(options: {
8
8
  loadAssetIds: AssetIds;
9
9
  _stacQuery?: string;
10
10
  }): URLSearchParams | null;
11
- export declare function bandIndexesToURLParams(urlParams: URLSearchParams, bandIndexes: BandIndexes): URLSearchParams;
11
+ export declare function bandIndexesToURLParams(urlParams: URLSearchParams, bandIndexes: BandIndexes, useNewFormat: boolean): URLSearchParams;
12
12
  export declare function getSingleCOGUrlParams(options: {
13
13
  stac: CompleteSTACItem;
14
14
  loadAssetId: string;
@@ -89,8 +89,8 @@ function getStacApiUrlParams(options) {
89
89
  query: query
90
90
  });
91
91
  }
92
- function bandIndexesToURLParams(urlParams, bandIndexes) {
93
- if ((0, _utils.getApplicationConfig)().rasterServerUseLatestTitiler) {
92
+ function bandIndexesToURLParams(urlParams, bandIndexes, useNewFormat) {
93
+ if (useNewFormat) {
94
94
  // for newer titiler versions
95
95
  bandIndexes.forEach(function (bandIndex) {
96
96
  urlParams.append('bidx', String(bandIndex + 1));
@@ -106,6 +106,7 @@ function bandIndexesToURLParams(urlParams, bandIndexes) {
106
106
  return urlParams;
107
107
  }
108
108
  function getSingleCOGUrlParams(options) {
109
+ var _stac$rasterServerUse;
109
110
  var stac = options.stac,
110
111
  loadAssetId = options.loadAssetId,
111
112
  loadBandIndexes = options.loadBandIndexes,
@@ -119,7 +120,7 @@ function getSingleCOGUrlParams(options) {
119
120
  return_mask: String(mask),
120
121
  url: url
121
122
  });
122
- return bandIndexesToURLParams(urlParams, loadBandIndexes);
123
+ return bandIndexesToURLParams(urlParams, loadBandIndexes, Boolean((_stac$rasterServerUse = stac.rasterServerUseLatestTitiler) !== null && _stac$rasterServerUse !== void 0 ? _stac$rasterServerUse : (0, _utils.getApplicationConfig)().rasterServerUseLatestTitiler));
123
124
  }
124
125
 
125
126
  /**
@@ -194,4 +195,4 @@ var RasterLayerResources = exports.RasterLayerResources = {
194
195
  return "".concat((0, _utils.getApplicationConfig)().cdnUrl, "/raster/colormaps/").concat(colormapId, ".png");
195
196
  }
196
197
  };
197
- //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"names":["_utils","require","_config","TILE_SIZE","STAC_SEARCH_DATA","sentinelCollectionName","stacSearchUrl","constructStacApiQuery","options","_STAC_SEARCH_DATA$sta","stac","startDate","endDate","stacSearchProvider","collections","id","datetime","concat","getStacApiUrl","getStacApiUrlParams","loadAssetIds","_options$mask","mask","query","_stacQuery","JSON","stringify","searchUrl","bandIndexAssets","map","assetId","mapping","DEFAULT_BAND_MAPPINGS","bandIndex","URLSearchParams","assets","join","return_mask","String","url","bandIndexesToURLParams","urlParams","bandIndexes","getApplicationConfig","rasterServerUseLatestTitiler","forEach","append","val","getSingleCOGUrlParams","loadAssetId","loadBandIndexes","_options$mask2","href","getTitilerUrl","_stac$rasterTileServe","useSTACSearching","x","y","z","rasterTileServerUrls","length","Error","pathStem","getTitilerPathMapping","scale","domain","chooseDomain","rasterServerUrl","arguments","undefined","domains","index","Math","abs","getTerrainUrl","meshMaxError","params","mesh_max_error","toFixed","baseUrl","toString","getMeshMaxError","multiplier","RasterLayerResources","exports","rasterColorMap","colormapId","cdnUrl"],"sources":["../../src/raster-tile/url.ts"],"sourcesContent":["// SPDX-License-Identifier: MIT\n// Copyright contributors to the kepler.gl project\n\n/* Utilities for creating request urls */\n/* global URLSearchParams */\n\nimport {StacTypes} from '@kepler.gl/types';\nimport {getApplicationConfig} from '@kepler.gl/utils';\n\nimport {DEFAULT_BAND_MAPPINGS} from './config';\nimport {\n  AssetIds,\n  BandIndexes,\n  CompleteSTACItem,\n  CompleteSTACCollection,\n  GetTileDataProps\n} from './types';\n\ntype Item = StacTypes.STACItem;\ntype Collection = StacTypes.STACCollection;\n\nconst TILE_SIZE: 256 | 512 = 256;\n\ninterface StacSearchInfo {\n  sentinelCollectionName: string[];\n  stacSearchUrl: string;\n}\n\nconst STAC_SEARCH_DATA: Record<string, StacSearchInfo> = {\n  // Commenting out Microsoft for now\n  // microsoft: {\n  //   sentinelCollectionName: ['sentinel-2-l2a'],\n  //   stacSearchUrl: 'https://planetarycomputer.microsoft.com/api/stac/v1/search'\n  // },\n  'earth-search': {\n    sentinelCollectionName: ['sentinel-s2-l2a-cogs'],\n    stacSearchUrl: 'https://earth-search.aws.element84.com/v0/search'\n  }\n};\n\n/**\n * Construct query parameters to be sent to STAC API instance\n */\nfunction constructStacApiQuery(options: {\n  stac: Item | Collection;\n  startDate: string;\n  endDate: string;\n  stacSearchProvider: string;\n}): {collections: string[]; datetime: string} {\n  const {stac, startDate, endDate, stacSearchProvider} = options;\n\n  // This is a quick hack to support the same Sentinel 2 STAC object for searching both microsoft\n  // and AWS\n  const collections = STAC_SEARCH_DATA[stacSearchProvider]?.sentinelCollectionName || [stac.id];\n\n  return {\n    collections,\n    datetime: `${startDate}T00:00:00Z/${endDate}T23:59:59Z`\n  };\n}\n\n/**\n * Perform lookup to find url of desired STAC search provider\n */\nfunction getStacApiUrl(stacSearchProvider: string): string {\n  return STAC_SEARCH_DATA[stacSearchProvider].stacSearchUrl;\n}\n\nexport function getStacApiUrlParams(options: {\n  stac: CompleteSTACCollection;\n  stacSearchProvider: string;\n  startDate: string;\n  endDate: string;\n  mask?: boolean;\n  loadAssetIds: AssetIds;\n  _stacQuery?: string;\n}): URLSearchParams | null {\n  const {stac, loadAssetIds, stacSearchProvider, mask = false} = options;\n  const query = options._stacQuery || JSON.stringify(constructStacApiQuery(options));\n  const searchUrl = getStacApiUrl(stacSearchProvider);\n\n  if (!searchUrl) {\n    return null;\n  }\n\n  const bandIndexAssets = loadAssetIds.map(assetId => {\n    const mapping = DEFAULT_BAND_MAPPINGS[stac.id];\n    if (!mapping) {\n      // TODO provide a UI to setup custom band mapping\n      return assetId;\n    }\n\n    const bandIndex = mapping[assetId];\n    if (bandIndex) {\n      return bandIndex;\n    }\n\n    // This is most likely incorrect as BXX is expected, not common name\n    return assetId;\n  });\n\n  return new URLSearchParams({\n    assets: bandIndexAssets.join(','),\n    return_mask: String(mask),\n    url: searchUrl,\n    query\n  });\n}\n\nexport function bandIndexesToURLParams(\n  urlParams: URLSearchParams,\n  bandIndexes: BandIndexes\n): URLSearchParams {\n  if (getApplicationConfig().rasterServerUseLatestTitiler) {\n    // for newer titiler versions\n    bandIndexes.forEach(bandIndex => {\n      urlParams.append('bidx', String(bandIndex + 1));\n    });\n  } else {\n    // The parameter in titiler is `bands` for landsat/sentinel and `bidx` for COG\n    // GDAL/Rasterio/rio-tiler start band indexing at one\n    // older titiler versions\n    urlParams.append('bidx', bandIndexes.map(val => val + 1).join(','));\n  }\n\n  return urlParams;\n}\n\nexport function getSingleCOGUrlParams(options: {\n  stac: CompleteSTACItem;\n  loadAssetId: string;\n  loadBandIndexes: BandIndexes;\n  mask?: boolean;\n}): URLSearchParams | null {\n  const {stac, loadAssetId, loadBandIndexes, mask = false} = options;\n  const url = stac.assets[loadAssetId].href;\n\n  if (!url) {\n    return null;\n  }\n\n  const urlParams = new URLSearchParams({\n    return_mask: String(mask),\n    url\n  });\n  return bandIndexesToURLParams(urlParams, loadBandIndexes);\n}\n\n/**\n * Construct full URL to load tile from a Titiler-based backend\n */\nexport function getTitilerUrl(options: {\n  stac: GetTileDataProps['stac'];\n  useSTACSearching: boolean;\n  x: number;\n  y: number;\n  z: number;\n}): {url: string; rasterServerUrl: string} {\n  // mask Set to false for mosaics because entire image is assumed to be valid\n  const {stac, useSTACSearching, x, y, z} = options;\n\n  if (!stac.rasterTileServerUrls?.length) {\n    throw new Error('No raster tile servers');\n  }\n\n  const pathStem = getTitilerPathMapping(stac, useSTACSearching);\n  const scale = TILE_SIZE === 512 ? '@2x' : '';\n  const domain = chooseDomain(stac.rasterTileServerUrls, x, y);\n  return {\n    url: `${domain}/${pathStem}/tiles/WebMercatorQuad/${z}/${x}/${y}${scale}.npy`,\n    rasterServerUrl: domain\n  };\n}\n\nexport function getTitilerPathMapping(\n  stac: GetTileDataProps['stac'],\n  useSTACSearching = false\n): string {\n  if (useSTACSearching) {\n    return 'stac/mosaic';\n  }\n\n  return 'cog';\n}\n\n/**\n * Choose from available domains to load images from\n *\n * @param x  x tile index\n * @param y  y tile index\n *\n * @return domain\n */\nfunction chooseDomain(domains: string[], x: number, y: number): string {\n  const index = Math.abs(x + y) % domains.length;\n  return domains[index];\n}\n\nexport function getTerrainUrl(\n  rasterTileServerUrls: string[],\n  x: number,\n  y: number,\n  z: number,\n  meshMaxError: number\n): {url: string; rasterServerUrl: string} {\n  const scale = TILE_SIZE === 512 ? '@2x' : '';\n\n  const params = new URLSearchParams({\n    url: 'terrarium',\n    mesh_max_error: meshMaxError.toFixed(2)\n  });\n  const domain = chooseDomain(rasterTileServerUrls, x, y);\n  const baseUrl = `${domain}/mesh/tiles/${z}/${x}/${y}${scale}.terrain?`;\n  return {url: baseUrl + params.toString(), rasterServerUrl: domain};\n}\n\n/**\n * get mesh max error for z value\n * @param z mercator tile z coord\n * @param multiplier multipler applied to default error\n *\n * Uses suggestion from here\n * https://www.linkedin.com/pulse/fast-cesium-terrain-rendering-new-quantized-mesh-output-alvaro-huarte/\n */\nexport function getMeshMaxError(z: number, multiplier: number): number {\n  return (77067.34 / (1 << z)) * multiplier;\n}\n\nexport const RasterLayerResources = {\n  rasterColorMap: (colormapId: string) => {\n    return `${getApplicationConfig().cdnUrl}/raster/colormaps/${colormapId}.png`;\n  }\n};\n"],"mappings":"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","ignoreList":[]}
198
+ //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"names":["_utils","require","_config","TILE_SIZE","STAC_SEARCH_DATA","sentinelCollectionName","stacSearchUrl","constructStacApiQuery","options","_STAC_SEARCH_DATA$sta","stac","startDate","endDate","stacSearchProvider","collections","id","datetime","concat","getStacApiUrl","getStacApiUrlParams","loadAssetIds","_options$mask","mask","query","_stacQuery","JSON","stringify","searchUrl","bandIndexAssets","map","assetId","mapping","DEFAULT_BAND_MAPPINGS","bandIndex","URLSearchParams","assets","join","return_mask","String","url","bandIndexesToURLParams","urlParams","bandIndexes","useNewFormat","forEach","append","val","getSingleCOGUrlParams","_stac$rasterServerUse","loadAssetId","loadBandIndexes","_options$mask2","href","Boolean","rasterServerUseLatestTitiler","getApplicationConfig","getTitilerUrl","_stac$rasterTileServe","useSTACSearching","x","y","z","rasterTileServerUrls","length","Error","pathStem","getTitilerPathMapping","scale","domain","chooseDomain","rasterServerUrl","arguments","undefined","domains","index","Math","abs","getTerrainUrl","meshMaxError","params","mesh_max_error","toFixed","baseUrl","toString","getMeshMaxError","multiplier","RasterLayerResources","exports","rasterColorMap","colormapId","cdnUrl"],"sources":["../../src/raster-tile/url.ts"],"sourcesContent":["// SPDX-License-Identifier: MIT\n// Copyright contributors to the kepler.gl project\n\n/* Utilities for creating request urls */\n/* global URLSearchParams */\n\nimport {StacTypes} from '@kepler.gl/types';\nimport {getApplicationConfig} from '@kepler.gl/utils';\n\nimport {DEFAULT_BAND_MAPPINGS} from './config';\nimport {\n  AssetIds,\n  BandIndexes,\n  CompleteSTACItem,\n  CompleteSTACCollection,\n  GetTileDataProps\n} from './types';\n\ntype Item = StacTypes.STACItem;\ntype Collection = StacTypes.STACCollection;\n\nconst TILE_SIZE: 256 | 512 = 256;\n\ninterface StacSearchInfo {\n  sentinelCollectionName: string[];\n  stacSearchUrl: string;\n}\n\nconst STAC_SEARCH_DATA: Record<string, StacSearchInfo> = {\n  // Commenting out Microsoft for now\n  // microsoft: {\n  //   sentinelCollectionName: ['sentinel-2-l2a'],\n  //   stacSearchUrl: 'https://planetarycomputer.microsoft.com/api/stac/v1/search'\n  // },\n  'earth-search': {\n    sentinelCollectionName: ['sentinel-s2-l2a-cogs'],\n    stacSearchUrl: 'https://earth-search.aws.element84.com/v0/search'\n  }\n};\n\n/**\n * Construct query parameters to be sent to STAC API instance\n */\nfunction constructStacApiQuery(options: {\n  stac: Item | Collection;\n  startDate: string;\n  endDate: string;\n  stacSearchProvider: string;\n}): {collections: string[]; datetime: string} {\n  const {stac, startDate, endDate, stacSearchProvider} = options;\n\n  // This is a quick hack to support the same Sentinel 2 STAC object for searching both microsoft\n  // and AWS\n  const collections = STAC_SEARCH_DATA[stacSearchProvider]?.sentinelCollectionName || [stac.id];\n\n  return {\n    collections,\n    datetime: `${startDate}T00:00:00Z/${endDate}T23:59:59Z`\n  };\n}\n\n/**\n * Perform lookup to find url of desired STAC search provider\n */\nfunction getStacApiUrl(stacSearchProvider: string): string {\n  return STAC_SEARCH_DATA[stacSearchProvider].stacSearchUrl;\n}\n\nexport function getStacApiUrlParams(options: {\n  stac: CompleteSTACCollection;\n  stacSearchProvider: string;\n  startDate: string;\n  endDate: string;\n  mask?: boolean;\n  loadAssetIds: AssetIds;\n  _stacQuery?: string;\n}): URLSearchParams | null {\n  const {stac, loadAssetIds, stacSearchProvider, mask = false} = options;\n  const query = options._stacQuery || JSON.stringify(constructStacApiQuery(options));\n  const searchUrl = getStacApiUrl(stacSearchProvider);\n\n  if (!searchUrl) {\n    return null;\n  }\n\n  const bandIndexAssets = loadAssetIds.map(assetId => {\n    const mapping = DEFAULT_BAND_MAPPINGS[stac.id];\n    if (!mapping) {\n      // TODO provide a UI to setup custom band mapping\n      return assetId;\n    }\n\n    const bandIndex = mapping[assetId];\n    if (bandIndex) {\n      return bandIndex;\n    }\n\n    // This is most likely incorrect as BXX is expected, not common name\n    return assetId;\n  });\n\n  return new URLSearchParams({\n    assets: bandIndexAssets.join(','),\n    return_mask: String(mask),\n    url: searchUrl,\n    query\n  });\n}\n\nexport function bandIndexesToURLParams(\n  urlParams: URLSearchParams,\n  bandIndexes: BandIndexes,\n  useNewFormat: boolean\n): URLSearchParams {\n  if (useNewFormat) {\n    // for newer titiler versions\n    bandIndexes.forEach(bandIndex => {\n      urlParams.append('bidx', String(bandIndex + 1));\n    });\n  } else {\n    // The parameter in titiler is `bands` for landsat/sentinel and `bidx` for COG\n    // GDAL/Rasterio/rio-tiler start band indexing at one\n    // older titiler versions\n    urlParams.append('bidx', bandIndexes.map(val => val + 1).join(','));\n  }\n\n  return urlParams;\n}\n\nexport function getSingleCOGUrlParams(options: {\n  stac: CompleteSTACItem;\n  loadAssetId: string;\n  loadBandIndexes: BandIndexes;\n  mask?: boolean;\n}): URLSearchParams | null {\n  const {stac, loadAssetId, loadBandIndexes, mask = false} = options;\n  const url = stac.assets[loadAssetId].href;\n\n  if (!url) {\n    return null;\n  }\n\n  const urlParams = new URLSearchParams({\n    return_mask: String(mask),\n    url\n  });\n  return bandIndexesToURLParams(\n    urlParams,\n    loadBandIndexes,\n    Boolean(\n      stac.rasterServerUseLatestTitiler ?? getApplicationConfig().rasterServerUseLatestTitiler\n    )\n  );\n}\n\n/**\n * Construct full URL to load tile from a Titiler-based backend\n */\nexport function getTitilerUrl(options: {\n  stac: GetTileDataProps['stac'];\n  useSTACSearching: boolean;\n  x: number;\n  y: number;\n  z: number;\n}): {url: string; rasterServerUrl: string} {\n  // mask Set to false for mosaics because entire image is assumed to be valid\n  const {stac, useSTACSearching, x, y, z} = options;\n\n  if (!stac.rasterTileServerUrls?.length) {\n    throw new Error('No raster tile servers');\n  }\n\n  const pathStem = getTitilerPathMapping(stac, useSTACSearching);\n  const scale = TILE_SIZE === 512 ? '@2x' : '';\n  const domain = chooseDomain(stac.rasterTileServerUrls, x, y);\n  return {\n    url: `${domain}/${pathStem}/tiles/WebMercatorQuad/${z}/${x}/${y}${scale}.npy`,\n    rasterServerUrl: domain\n  };\n}\n\nexport function getTitilerPathMapping(\n  stac: GetTileDataProps['stac'],\n  useSTACSearching = false\n): string {\n  if (useSTACSearching) {\n    return 'stac/mosaic';\n  }\n\n  return 'cog';\n}\n\n/**\n * Choose from available domains to load images from\n *\n * @param x  x tile index\n * @param y  y tile index\n *\n * @return domain\n */\nfunction chooseDomain(domains: string[], x: number, y: number): string {\n  const index = Math.abs(x + y) % domains.length;\n  return domains[index];\n}\n\nexport function getTerrainUrl(\n  rasterTileServerUrls: string[],\n  x: number,\n  y: number,\n  z: number,\n  meshMaxError: number\n): {url: string; rasterServerUrl: string} {\n  const scale = TILE_SIZE === 512 ? '@2x' : '';\n\n  const params = new URLSearchParams({\n    url: 'terrarium',\n    mesh_max_error: meshMaxError.toFixed(2)\n  });\n  const domain = chooseDomain(rasterTileServerUrls, x, y);\n  const baseUrl = `${domain}/mesh/tiles/${z}/${x}/${y}${scale}.terrain?`;\n  return {url: baseUrl + params.toString(), rasterServerUrl: domain};\n}\n\n/**\n * get mesh max error for z value\n * @param z mercator tile z coord\n * @param multiplier multipler applied to default error\n *\n * Uses suggestion from here\n * https://www.linkedin.com/pulse/fast-cesium-terrain-rendering-new-quantized-mesh-output-alvaro-huarte/\n */\nexport function getMeshMaxError(z: number, multiplier: number): number {\n  return (77067.34 / (1 << z)) * multiplier;\n}\n\nexport const RasterLayerResources = {\n  rasterColorMap: (colormapId: string) => {\n    return `${getApplicationConfig().cdnUrl}/raster/colormaps/${colormapId}.png`;\n  }\n};\n"],"mappings":"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","ignoreList":[]}
@@ -1,7 +1,7 @@
1
1
  {
2
2
  "name": "@kepler.gl/layers",
3
3
  "author": "Shan He <shan@uber.com>",
4
- "version": "3.1.9",
4
+ "version": "3.2.0",
5
5
  "description": "kepler.gl constants used by kepler.gl components, actions and reducers",
6
6
  "license": "MIT",
7
7
  "main": "dist/index.js",
@@ -36,14 +36,14 @@
36
36
  "@deck.gl/geo-layers": "^8.9.27",
37
37
  "@deck.gl/layers": "^8.9.27",
38
38
  "@deck.gl/mesh-layers": "^8.9.27",
39
- "@kepler.gl/common-utils": "3.1.9",
40
- "@kepler.gl/constants": "3.1.9",
41
- "@kepler.gl/deckgl-arrow-layers": "3.1.9",
42
- "@kepler.gl/deckgl-layers": "3.1.9",
43
- "@kepler.gl/localization": "3.1.9",
44
- "@kepler.gl/table": "3.1.9",
45
- "@kepler.gl/types": "3.1.9",
46
- "@kepler.gl/utils": "3.1.9",
39
+ "@kepler.gl/common-utils": "3.2.0",
40
+ "@kepler.gl/constants": "3.2.0",
41
+ "@kepler.gl/deckgl-arrow-layers": "3.2.0",
42
+ "@kepler.gl/deckgl-layers": "3.2.0",
43
+ "@kepler.gl/localization": "3.2.0",
44
+ "@kepler.gl/table": "3.2.0",
45
+ "@kepler.gl/types": "3.2.0",
46
+ "@kepler.gl/utils": "3.2.0",
47
47
  "@loaders.gl/arrow": "^4.3.2",
48
48
  "@loaders.gl/core": "^4.3.2",
49
49
  "@loaders.gl/gis": "^4.3.2",
@@ -251,6 +251,16 @@ type LoadingOptions = {
251
251
  fetch?: typeof fetch | FetchLike;
252
252
  };
253
253
 
254
+ type NpyRequest = {
255
+ url: string;
256
+ rasterServerUrl: string;
257
+ options: RequestInit;
258
+ rasterServerMaxRetries?: number;
259
+ rasterServerRetryDelay?: number;
260
+ rasterServerServerErrorsToRetry?: number[];
261
+ rasterServerMaxPerServerRequests?: number;
262
+ };
263
+
254
264
  /**
255
265
  * Load NPY Array
256
266
  *
@@ -263,98 +273,106 @@ type LoadingOptions = {
263
273
  * @return image object to pass to Texture2D constructor
264
274
  */
265
275
  export async function loadNpyArray(
266
- request: {url: string; rasterServerUrl: string; options: RequestInit},
276
+ request: NpyRequest,
267
277
  split: true,
268
278
  options?: LoadingOptions
269
279
  ): Promise<Texture2DProps[] | null>;
270
280
  export async function loadNpyArray(
271
- request: {url: string; rasterServerUrl: string; options: RequestInit},
281
+ request: NpyRequest,
272
282
  split: false,
273
283
  options?: LoadingOptions
274
284
  ): Promise<Texture2DProps | null>;
275
285
  export async function loadNpyArray(
276
- request: {url: string; rasterServerUrl: string; options: RequestInit},
286
+ request: NpyRequest,
277
287
  split: boolean,
278
288
  options?: LoadingOptions
279
289
  ): Promise<Texture2DProps | Texture2DProps[] | null> {
280
- const numAttempts = 1 + getApplicationConfig().rasterServerMaxRetries;
281
-
282
- const asset = await getRequestThrottle().throttleRequest(request.rasterServerUrl, async () => {
283
- for (let attempt = 0; attempt < numAttempts; attempt++) {
284
- try {
285
- const {npy: npyOptions} = getLoaderOptions();
286
- const response: NPYLoaderResponse = await load(request.url, NPYLoader, {
287
- npy: npyOptions,
288
- fetch: options?.fetch
289
- });
290
-
291
- if (!response || !response.data || request.options.signal?.aborted) {
292
- return null;
293
- }
294
-
295
- // Float64 data needs to be coerced to Float32 for the GPU
296
- if (response.data instanceof Float64Array) {
297
- response.data = Float32Array.from(response.data);
298
- }
299
-
300
- const {data, header} = response;
301
- const {shape} = header;
302
- const {format, dataFormat, type} = getWebGL2TextureParameters(data);
303
-
304
- // TODO: check height-width or width-height
305
- // Regardless, images usually square
306
- // TODO: handle cases of 256x256x1 instead of 1x256x256
307
- const [z, height, width] = shape;
308
-
309
- // Since we now use WebGL2 data types for 8-bit textures, we set the following for all textures
310
- const mipmaps = false;
311
- const parameters = DEFAULT_HIGH_BIT_TEXTURE_PARAMETERS;
312
-
313
- if (!split) {
314
- return {
315
- data,
316
- width,
317
- height,
318
- format,
319
- dataFormat,
320
- type,
321
- parameters,
322
- mipmaps
323
- };
324
- }
325
-
326
- // Split into individual arrays
327
- const channels: Texture2DProps[] = [];
328
- const channelSize = height * width;
329
- for (let i = 0; i < z; i++) {
330
- channels.push({
331
- data: data.subarray(i * channelSize, (i + 1) * channelSize),
332
- width,
333
- height,
334
- format,
335
- dataFormat,
336
- type,
337
- parameters,
338
- mipmaps
290
+ const numAttempts =
291
+ 1 + (request.rasterServerMaxRetries ?? getApplicationConfig().rasterServerMaxRetries);
292
+
293
+ const asset = await getRequestThrottle().throttleRequest(
294
+ request.rasterServerUrl,
295
+ async () => {
296
+ for (let attempt = 0; attempt < numAttempts; attempt++) {
297
+ try {
298
+ const {npy: npyOptions} = getLoaderOptions();
299
+ const response: NPYLoaderResponse = await load(request.url, NPYLoader, {
300
+ npy: npyOptions,
301
+ fetch: options?.fetch
339
302
  });
340
- }
341
- return channels;
342
- } catch (error) {
343
- // Retry if Service Temporarily Unavailable 503 error etc.
344
- if (
345
- attempt < numAttempts &&
346
- error instanceof FetchError &&
347
- getApplicationConfig().rasterServerServerErrorsToRetry?.includes(
348
- error.response?.status as number
349
- )
350
- ) {
351
- await sleep(getApplicationConfig().rasterServerRetryDelay);
352
- continue;
303
+
304
+ if (!response || !response.data || request.options.signal?.aborted) {
305
+ return null;
306
+ }
307
+
308
+ // Float64 data needs to be coerced to Float32 for the GPU
309
+ if (response.data instanceof Float64Array) {
310
+ response.data = Float32Array.from(response.data);
311
+ }
312
+
313
+ const {data, header} = response;
314
+ const {shape} = header;
315
+ const {format, dataFormat, type} = getWebGL2TextureParameters(data);
316
+
317
+ // TODO: check height-width or width-height
318
+ // Regardless, images usually square
319
+ // TODO: handle cases of 256x256x1 instead of 1x256x256
320
+ const [z, height, width] = shape;
321
+
322
+ // Since we now use WebGL2 data types for 8-bit textures, we set the following for all textures
323
+ const mipmaps = false;
324
+ const parameters = DEFAULT_HIGH_BIT_TEXTURE_PARAMETERS;
325
+
326
+ if (!split) {
327
+ return {
328
+ data,
329
+ width,
330
+ height,
331
+ format,
332
+ dataFormat,
333
+ type,
334
+ parameters,
335
+ mipmaps
336
+ };
337
+ }
338
+
339
+ // Split into individual arrays
340
+ const channels: Texture2DProps[] = [];
341
+ const channelSize = height * width;
342
+ for (let i = 0; i < z; i++) {
343
+ channels.push({
344
+ data: data.subarray(i * channelSize, (i + 1) * channelSize),
345
+ width,
346
+ height,
347
+ format,
348
+ dataFormat,
349
+ type,
350
+ parameters,
351
+ mipmaps
352
+ });
353
+ }
354
+ return channels;
355
+ } catch (error) {
356
+ // Retry if Service Temporarily Unavailable 503 error etc.
357
+ if (
358
+ attempt < numAttempts &&
359
+ error instanceof FetchError &&
360
+ (
361
+ request.rasterServerServerErrorsToRetry ??
362
+ getApplicationConfig().rasterServerServerErrorsToRetry
363
+ )?.includes(error.response?.status as number)
364
+ ) {
365
+ await sleep(
366
+ request.rasterServerRetryDelay ?? getApplicationConfig().rasterServerRetryDelay
367
+ );
368
+ continue;
369
+ }
353
370
  }
354
371
  }
355
- }
356
- return null;
357
- });
372
+ return null;
373
+ },
374
+ request.rasterServerMaxPerServerRequests
375
+ );
358
376
 
359
377
  return asset;
360
378
  }
@@ -263,13 +263,21 @@ async function getSingleAssetSTACRequest(
263
263
  return null;
264
264
  }
265
265
 
266
- return await getAssetRequest({
266
+ const asset = await getAssetRequest({
267
267
  ...urlInfo,
268
268
  params: urlParams,
269
269
  options: {signal},
270
270
  useMask,
271
271
  responseRequiredBandIndices
272
272
  });
273
+ // Pass per-layer override for retries if present on STAC metadata
274
+ return {
275
+ ...asset,
276
+ rasterServerMaxRetries: stac.rasterServerMaxRetries,
277
+ rasterServerRetryDelay: stac.rasterServerRetryDelay,
278
+ rasterServerServerErrorsToRetry: stac.rasterServerServerErrorsToRetry,
279
+ rasterServerMaxPerServerRequests: stac.rasterServerMaxPerServerRequests
280
+ };
273
281
  }
274
282
 
275
283
  /**
@@ -335,11 +343,19 @@ async function getMultiAssetSTACRequest(
335
343
  }
336
344
 
337
345
  if (isValidRequestData(requestData)) {
338
- return await Promise.all(
346
+ const assets = await Promise.all(
339
347
  requestData.map(request =>
340
348
  getAssetRequest({...request, options: request.options ?? {signal}})
341
349
  )
342
350
  );
351
+ // Propagate per-layer retry override
352
+ return assets.map(asset => ({
353
+ ...asset,
354
+ rasterServerMaxRetries: options.stac.rasterServerMaxRetries,
355
+ rasterServerRetryDelay: options.stac.rasterServerRetryDelay,
356
+ rasterServerServerErrorsToRetry: options.stac.rasterServerServerErrorsToRetry,
357
+ rasterServerMaxPerServerRequests: options.stac.rasterServerMaxPerServerRequests
358
+ }));
343
359
  }
344
360
 
345
361
  return null;
@@ -424,18 +440,24 @@ export async function loadTerrain(props: {
424
440
  signal: AbortSignal;
425
441
  rasterTileServerUrls: string[];
426
442
  boundsForGeometry?: [number, number, number, number];
443
+ rasterServerMaxRetries: number | undefined;
444
+ rasterServerRetryDelay: number | undefined;
445
+ rasterServerServerErrorsToRetry?: number[];
446
+ rasterServerMaxPerServerRequests?: number;
427
447
  }): Promise<TerrainData | null> {
428
448
  const {
429
449
  index: {x, y, z},
430
450
  boundsForGeometry,
431
451
  signal,
432
- rasterTileServerUrls
452
+ rasterTileServerUrls,
453
+ rasterServerMaxRetries,
454
+ rasterServerMaxPerServerRequests
433
455
  } = props;
434
456
 
435
457
  const meshMaxError = getMeshMaxError(z, MESH_MULTIPLIER);
436
458
  const terrainUrlInfo = getTerrainUrl(rasterTileServerUrls, x, y, z, meshMaxError);
437
459
  const loaderOptions = getLoaderOptions();
438
- const numAttempts = 1 + getApplicationConfig().rasterServerMaxRetries;
460
+ const numAttempts = 1 + (rasterServerMaxRetries ?? getApplicationConfig().rasterServerMaxRetries);
439
461
 
440
462
  const mesh = await getRequestThrottle().throttleRequest(
441
463
  terrainUrlInfo.rasterServerUrl,
@@ -455,17 +477,21 @@ export async function loadTerrain(props: {
455
477
  if (
456
478
  attempt < numAttempts &&
457
479
  error instanceof FetchError &&
458
- getApplicationConfig().rasterServerServerErrorsToRetry?.includes(
459
- error.response?.status as number
460
- )
480
+ (
481
+ props.rasterServerServerErrorsToRetry ??
482
+ getApplicationConfig().rasterServerServerErrorsToRetry
483
+ )?.includes(error.response?.status as number)
461
484
  ) {
462
- await sleep(getApplicationConfig().rasterServerRetryDelay);
485
+ await sleep(
486
+ props.rasterServerRetryDelay ?? getApplicationConfig().rasterServerRetryDelay
487
+ );
463
488
  continue;
464
489
  }
465
490
  }
466
491
  }
467
492
  return null;
468
- }
493
+ },
494
+ rasterServerMaxPerServerRequests
469
495
  );
470
496
 
471
497
  return mesh;
@@ -83,7 +83,7 @@ const getShouldLoadTerrain = (stac, mapState, visConfig) => {
83
83
  visConfig.enableTerrain &&
84
84
  // disabled in Top view by default
85
85
  (mapState.dragRotate || visConfig.enableTerrainTopView) &&
86
- getApplicationConfig().rasterServerSupportsElevation
86
+ (stac.rasterServerSupportsElevation ?? getApplicationConfig().rasterServerSupportsElevation)
87
87
  );
88
88
  };
89
89
 
@@ -633,7 +633,11 @@ export default class RasterTileLayer extends KeplerLayer {
633
633
  boundsForGeometry: [west, north, east, south],
634
634
  index: props.index,
635
635
  signal: props.signal,
636
- rasterTileServerUrls: props.stac.rasterTileServerUrls || []
636
+ rasterTileServerUrls: props.stac.rasterTileServerUrls || [],
637
+ rasterServerMaxRetries: props.stac.rasterServerMaxRetries,
638
+ rasterServerRetryDelay: props.stac.rasterServerRetryDelay,
639
+ rasterServerServerErrorsToRetry: props.stac.rasterServerServerErrorsToRetry,
640
+ rasterServerMaxPerServerRequests: props.stac.rasterServerMaxPerServerRequests
637
641
  }));
638
642
  return {images: null, ...(terrain ? {terrain} : {})};
639
643
  }
@@ -655,7 +659,11 @@ export default class RasterTileLayer extends KeplerLayer {
655
659
  boundsForGeometry: [west, north, east, south],
656
660
  index: props.index,
657
661
  signal: props.signal,
658
- rasterTileServerUrls: props.stac.rasterTileServerUrls || []
662
+ rasterTileServerUrls: props.stac.rasterTileServerUrls || [],
663
+ rasterServerMaxRetries: props.stac.rasterServerMaxRetries,
664
+ rasterServerRetryDelay: props.stac.rasterServerRetryDelay,
665
+ rasterServerServerErrorsToRetry: props.stac.rasterServerServerErrorsToRetry,
666
+ rasterServerMaxPerServerRequests: props.stac.rasterServerMaxPerServerRequests
659
667
  })
660
668
  : null
661
669
  ]);
@@ -685,7 +693,13 @@ export default class RasterTileLayer extends KeplerLayer {
685
693
  async getTileDataPMTiles(
686
694
  props: GetTileDataDefaultProps & {
687
695
  shouldLoadTerrain: boolean;
688
- metadata: {rasterTileServerUrls: string[]};
696
+ metadata: {
697
+ rasterTileServerUrls: string[];
698
+ rasterServerMaxRetries?: number;
699
+ rasterServerRetryDelay?: number;
700
+ rasterServerServerErrorsToRetry?: number[];
701
+ rasterServerMaxPerServerRequests?: number;
702
+ };
689
703
  globalBounds: DataSourceParams['globalBounds'];
690
704
  },
691
705
  tileSource
@@ -714,7 +728,11 @@ export default class RasterTileLayer extends KeplerLayer {
714
728
  boundsForGeometry: [west, north, east, south],
715
729
  index: props.index,
716
730
  signal: props.signal,
717
- rasterTileServerUrls: metadata.rasterTileServerUrls
731
+ rasterTileServerUrls: metadata.rasterTileServerUrls,
732
+ rasterServerMaxRetries: metadata.rasterServerMaxRetries,
733
+ rasterServerRetryDelay: metadata.rasterServerRetryDelay,
734
+ rasterServerServerErrorsToRetry: metadata.rasterServerServerErrorsToRetry,
735
+ rasterServerMaxPerServerRequests: metadata.rasterServerMaxPerServerRequests
718
736
  })
719
737
  : null;
720
738
 
@@ -39,13 +39,18 @@ class RequestThrottle {
39
39
  : 'No active server queues';
40
40
  }
41
41
 
42
- async throttleRequest<T>(serverKey: string, requestFunction: () => Promise<T>): Promise<T> {
42
+ async throttleRequest<T>(
43
+ serverKey: string,
44
+ requestFunction: () => Promise<T>,
45
+ maxConcurrentRequestsOverride?: number
46
+ ): Promise<T> {
43
47
  const serverQueue = this.getServerQueue(serverKey);
48
+ const maxConcurrentRequests =
49
+ typeof maxConcurrentRequestsOverride === 'number'
50
+ ? maxConcurrentRequestsOverride
51
+ : this.maxConcurrentRequests;
44
52
 
45
- if (
46
- serverQueue.activeRequests >= this.maxConcurrentRequests &&
47
- Boolean(this.maxConcurrentRequests)
48
- ) {
53
+ if (serverQueue.activeRequests >= maxConcurrentRequests && Boolean(maxConcurrentRequests)) {
49
54
  // Wait for a slot to become available
50
55
  await new Promise<void>(resolve => {
51
56
  serverQueue.queue.push(async () => {
@@ -99,7 +99,17 @@ export type CategoricalColormapOptions = {
99
99
  maxValue?: number;
100
100
  };
101
101
 
102
- export type ExtendedKeplerSTAC = {rasterTileServerUrls?: []};
102
+ export type ExtendedKeplerSTAC = {
103
+ rasterTileServerUrls?: [];
104
+ /** Optional per-layer override for max retries when fetching raster data */
105
+ rasterServerMaxRetries?: number;
106
+ /** Optional per-layer override for retry delay between attempts (ms) */
107
+ rasterServerRetryDelay?: number;
108
+ /** Optional per-layer override for which server errors are retried */
109
+ rasterServerServerErrorsToRetry?: number[];
110
+ /** Optional per-layer override for max concurrent requests per server */
111
+ rasterServerMaxPerServerRequests?: number;
112
+ };
103
113
 
104
114
  /**
105
115
  * Custom fields we pass on to the getTileData callback
@@ -207,6 +217,14 @@ export type AssetRequestData = {
207
217
  useMask: boolean;
208
218
  /** Pass this property through the request to pick specific bands from the response */
209
219
  responseRequiredBandIndices?: number[] | null;
220
+ /** Optional per-request override for max retries when fetching NPY arrays */
221
+ rasterServerMaxRetries?: number;
222
+ /** Optional per-request override for retry delay between attempts (ms) */
223
+ rasterServerRetryDelay?: number;
224
+ /** Optional per-request override for which server errors are retried */
225
+ rasterServerServerErrorsToRetry?: number[];
226
+ /** Optional per-request override for max concurrent requests per server */
227
+ rasterServerMaxPerServerRequests?: number;
210
228
  };
211
229
 
212
230
  export type NPYLoaderDataTypes =