Skip to content

Latest commit

 

History

History
352 lines (265 loc) · 10.5 KB

loading-data.md

File metadata and controls

352 lines (265 loc) · 10.5 KB

Loading Data

deck.gl uses loaders.gl, a framework-agnostic library to read data and resources.

deck.gl core always includes loaders for JSON and standard image formats (e.g, png, jpeg, svg). Certain layers include additional loaders supporting their own use cases. It is easy for applications to provide options to configure the behavior of the default loaders or to add loaders to support for additional formats.

Some examples of when loaders are used:

  • JSON array or object from an URL passed to the data prop of a layer
  • Texture from an image, such as image in BitmapLayer, iconAtlas in IconLayer, and texture in SimpleMeshLayer
  • Geometries from a binary tile, e.g. MVTLayer, TerrainLayer, and Tile3DLayer
  • Geometries from a standard 3D format, e.g. scenegraph in ScenegraphLayer, and mesh in SimpleMeshLayer

Customize Data Loading Behavior

All layers support a loadOptions prop that can be used to customize loading and parsing.

Example: Fetch data with credentials

In a production environment, deck.gl applications may need to load data from secure APIs that require special HTTP headers (such as Authorization) to be set.

In order to access a secure API, the loadOptions.fetch option passes through additional parameters to fetch, which deck.gl calls under the hood to load resources.

new ScatterplotLayer({
  data: 'https://secure-server.com/userActivity',
  loadOptions: {
    fetch: {
      method: 'POST',
      body: JSON.stringify(requestBody),
      headers: {
        'Authorization': `Bearer ${accessToken}`,
      }
    }
  }
});

Example: Override the default image loading options

deck.gl uses ImageLoader to read common image formats. The default loader options are:

{
  image: {type: 'auto'},
  imagebitmap: {premultiplyAlpha: 'none'}
}

The image is decoded into an ImageBitmap if the browser supports it (Firefox, Chrome, Edge) for better performance. You can override the default options for the createImageBitmap API as follows:

new IconLayer({
  iconAtlas: '/path/to/image.png',
  loadOptions: {
    imagebitmap: {
      // Flip the image vertically
      imageOrientation: 'flipY'
    }
  }
})

If the image is a SVG that does not include width and height information, createImageBitmap will throw a DOMException:

The image element contains an SVG image without intrinsic dimensions, and no resize options or crop region are specified.

This can be fixed by explicitly setting its dimensions:

new IconLayer({
  iconAtlas: '/path/to/image.svg',
  loadOptions: {
    imagebitmap: {
      resizeWidth: 256,
      resizeHeight: 256,
      resizeQuality: 'high'
    }
  }
})

Support Additional Formats

All layers support a loaders prop that can be used to add loaders.gl loaders for parsing a specific input format.

For example, the following code adds the CSVLoader to support CSV/TSV files:

import {CSVLoader} from '@loaders.gl/csv';

new HexagonLayer({
  data: '/path/to/data.tsv',
  loaders: [CSVLoader],
  loadOptions: {
    csv: {
      delimiter: '\t',
      dynamicTyping: true,
      skipEmptyLines: true
    }
  }
});

The following code adds the LASLoader to support LAS/LAZ files:

import {LASLoader} from '@loaders.gl/las';

new PointCloudLayer({
  mesh: '/path/to/pointcloud.laz',
  loaders: [LASLoader]
});

Force Reload From an URL

Usually, a layer refreshes its data when and only when the data prop changes. The following code refreshes data from the same URL every 5 minutes by changing a query parameter:

import Tabs from '@theme/Tabs'; import TabItem from '@theme/TabItem';

import {Deck} from '@deck.gl/core';
import {ScatterplotLayer} from '@deck.gl/layers';

const deckInstance = new Deck({
  initialViewState: {
    longitude: -122.4,
    latitude: 27.8,
    zoom: 8
  },
  controller: true
});

let dataVersion = 0;
function update() {
  const layers = [
    new ScatterplotLayer({
      data: `path/to/data.json?v=${++dataVersion}`,
      getPosition: d => d.position
    })
  ];

  deckInstance.setProps({layers});

  // Refresh after 5 minutes
  return setTimeout(update, 5 * 60 * 1000);
};

update();
import {Deck} from '@deck.gl/core';
import {ScatterplotLayer} from '@deck.gl/layers';

const deckInstance = new Deck({
  initialViewState: {
    longitude: -122.4,
    latitude: 27.8,
    zoom: 8
  },
  controller: true
});

type DataType = {
  position: [longitude: number, latitude: number];
};

let dataVersion: number = 0;
function update() {
  const layers = [
    new ScatterplotLayer<DataType>({
      data: `path/to/data.json?v=${++dataVersion}`,
      getPosition: (d: DataType) => d.position
    })
  ];

  deckInstance.setProps({layers});

  // Refresh after 5 minutes
  return setTimeout(update, 5 * 60 * 1000);
};

update();
import React, {useState, useEffect} from 'react';
import DeckGL from '@deck.gl/react';
import {MapViewState} from '@deck.gl/core';
import {ScatterplotLayer} from '@deck.gl/layers';

const INITIAL_VIEW_STATE: MapViewState = {
  longitude: -122.4,
  latitude: 27.8,
  zoom: 8
};

type DataType = {
  position: [longitude: number, latitude: number];
};

function App() {
  const [dataVersion, setDataVersion] = useState<number>(0);

  useEffect(() => {
    const nextUpdate = setTimeout(() => setDataVersion(dataVersion + 1), 5 * 60 * 1000);
    return () => clearTimeout(nextUpdate);
  }, [dataVersion]);

  const layers = [
    new ScatterplotLayer<DataType>({
      data: `/path/to/data.json?v=${++dataVersion}`,
      getPosition: (d: DataType) => d.position
    })
  ];

  return <DeckGL
    initialViewState={INITIAL_VIEW_STATE}
    controller
    layers={layers}
  />;
}

Loaders and Web Workers

For the best performance, some specialized loaders parse data using web workers, for example TerrainLoader in the TerrainLayer and MVTLoader in the MVTLayer. By default, the worker code is loaded from from the latest published NPM module on unpkg.com.

It might be desirable for some applications to serve the worker code itself without relying on the CDN. To do this, locate the worker bundle locally in node_modules/@loaders.gl/<module>/dist/<name>-worker.js and serve it as a static asset with your server. Point the loader to use this alternative URL using loadOptions.<name>.workerUrl:

new MVTLayer({
  loadOptions: {
    mvt: {
      // cp node_modules/@loaders.gl/mvt/dist/mvt-worker.js static/mvt-worker.js
      workerUrl: '/static/mvt-worker.js'
    }
  }
}

If the layer is used in an environment that does not support web workers, or you need to debug the loader code on the main thread, you may import the full loader like this:

import {MVTLoader} from '@loaders.gl/mvt';
new MVTLayer({
  loaders: [MVTLoader],
  loadOptions: {worker: false}
});

Refer to each specific layer's documentation to see which loaders are used.

Load Resource Without an URL

In some use cases, resources do not exist at a static URL. For example, some applications construct images dynamically based on user input. Some applications receive arbitrary binary blobs from a server via a WebSocket connection.

Before reading on, remember that you don't have to use a loader if your app already knows how to interpret the content. For example, if you have the RGBA values of all pixels of an image. you can simply construct an ImageData object:

new BitmapLayer({
  image: new ImageData(pixels, 128, 128)
})

If you have a custom-formatted binary, consider the techniques in using binary data.

The following examples only address the use cases where you need a loader/parser to interpret the incoming data.

Example: Use image from a programmatically generated SVG string

The following code dynamically generates SVG icons and convert them to data URLs.

function createSVGIcon(n: number): string {
  const label = n < 10 ? n.toString() : '10+';
  return `\
  <svg width="24" height="24" viewBox="0 0 24 24" xmlns="http://www.w3.org/2000/svg">
    <circle cx="12" cy="12" r="10" fill="#c00" stroke="#fa1" stroke-width="2"/>
    <text x="12" y="12" fill="#fff" text-anchor="middle" alignment-baseline="middle" font-size="8">${label}</text>
  </svg>`;
}

// Note that a xml string cannot be directly embedded in a data URL
// it has to be either escaped or converted to base64.
function svgToDataUrl(svg: string): string {
  return `data:image/svg+xml;charset=utf-8,${encodeURIComponent(svg)}`;
}

// You may need base64 encoding if the SVG contains certain special characters
function svgToDataUrlBase64(svg: string): string {
  return `data:image/svg+xml;base64,${btoa(svg)}`;
}

const layer = new IconLayer({
  getIcon: d => {
    url: svgToDataUrl(createSVGIcon(d.value)),
    width: 24,
    height: 24
  }
})

Example: Parse glTF from a binary blob

The following code shows how to parse a glTF model that is already loaded into an ArrayBuffer object.

There are two ways for deck.gl to load it. One is to create a blob URL:

const blob = new Blob([arraybuffer]);
const objectURL = URL.createObjectURL(blob);

const layer = new ScenegraphLayer({
  scenegraph: objectURL
});

Or more directly, import the parse utility from loaders.gl (already a dependency of deck.gl), which returns a promise:

import {parse} from '@loaders.gl/core';
import {GLTFLoader} from '@loaders.gl/gltf';

const layer = new ScenegraphLayer({
  scenegraph: parse(arraybuffer, GLTFLoader)
})