Image Markers

Image Markers

Image Markers

As we have seen in the Clustered Markers tutorial, we can place markers on a map using the following steps:

For each marker,

  • Create an Altus Image object
  • Use the Image object to instantiate an Altus Texture
  • Create a MarkerData object to assign position, altitude, descriptive text, a weight (TODO: explain this) and the texture
  • Specify where in the image to anchor it to the position
//Creates a square image with the given dimensions and colors
function squareImage(width, height, red, green, blue, alpha){
    var arr = new AltusUnified.VectorByte();
    for (i = 0; i < width * height; i += 1) {
        arr.push_back(red);
        arr.push_back(green);
        arr.push_back(blue);
        arr.push_back(alpha);
    }
    var image = new AltusUnified.Image(width, height, arr);
    arr.delete();
    return image;
}
//Converts an image to a texture
function textureFromImage(image){
    var tex = new AltusUnified.Texture(image, false);
    return tex;
}
//Returns a square texture with the given dimensions and colors
function squareTexture(width, height, red, green, blue, alpha){
    var image = squareImage(width, height, red, green, blue, alpha);
    var tex = textureFromImage(image);
    image.delete();
    return tex;
}
//Returns a populated MarkerData object
function createMarkerData(uid, lat, lon, alt, weight, minLevel, metaData, texture, anchorX, anchorY){
    var markerData = new AltusUnified.MarkerData(uid, lat, lon, alt, weight, metaData, texture);
    markerData.minimumLevel = minLevel;
    var anchorPoint = new AltusUnified.vec2d(anchorX, anchorY);
    markerData.anchorPoint_set(anchorPoint);
    anchorPoint.delete();
    return markerData;
}

From here we can add the markers to a vector, instantiate a map layer, add the markers to the layer and then add the layer to our scene.

Using Images Instead of Text and Polygons

To use an image such as a png or jpeg, we must first decide how we want to get the image data. We will look at two ways to do this:

  • Load the image over the network using an AJAX request
  • Provide the image data directly using an encoded data url

Loading an image over the Network using an AJAX request

With an XMLHttpRequest, we can send an AJAX request for our image, and specify a "response type" of "arraybuffer" to get the image data in a convenient form:

 function requestData(url, onLoad, onError, onTimeout, timeout) {
    var xhr = new XMLHttpRequest();
    xhr.open('GET', url, true);
    xhr.timeout = timeout || 60000; // default to 60 seconds
    xhr.responseType = 'arraybuffer';

    //Handle download timeouts
    xhr.ontimeout = onTimeout || function(e) {
      console.log(e)
    };

    //Handle successful downloads
    xhr.onload = function() {
      if (xhr.status == 200 && xhr.response) {
        onLoad(xhr.response);
      } else {
        onError(e);
      }
    };

    //Handle other errors
    xhr.onerror = onError || function(e) {
      console.error(e);
    };

    //Send the download request
    xhr.send(null);
  }

We can push this array data to a vector, populate an Altus Image object and then An Altus Texture as follows:

 function createTextureFromXhr(response) {
    var byteArray = new Uint8Array(response),
        vector = new AltusUnified.VectorByte(),
        altusImage = new AltusUnified.Image(),
        texture;

    pushToVector(vector, byteArray);

    altusImage.loadFromMemory(vector);
    altusImage.multiplyAlpha();

    texture = new AltusUnified.Texture(altusImage, false);

    vector.delete();
    altusImage.delete();

    return texture;
  }

We create a function to add the markers to a map layer given the texture, and call it within a callback to our AJAX function:

 function addMarkers(texture) {
    var anchor = new AltusUnified.vec2d(texture.getImageWidth() / 2, texture.getImageHeight()), //anchor to bottom center
        markers = [
          new AltusUnified.MarkerData(0, 38.889469, -77.035258, 0, 1, "Washington Monument", texture),
          new AltusUnified.MarkerData(1, 38.889803, -77.009114, 0, 2, "United States Capital", texture),
          new AltusUnified.MarkerData(2, 38.8977, -77.0365, 0, 3, "White House", texture),
          new AltusUnified.MarkerData(3, 38.8893, -77.050111, 0, 4, "Lincoln Memorial", texture),
          new AltusUnified.MarkerData(4, 38.8957, -77.0559, 0, 5, "Kennedy Center", texture),
          new AltusUnified.MarkerData(5, 38.891111, -77.047778, 0, 6, "Vietnam Memorial", texture),
          new AltusUnified.MarkerData(6, 38.888, -77.02, 0, 7, "Air & Space Museum", texture),
          new AltusUnified.MarkerData(7, 38.89731, -77.00626, 0, 8, "Union Station", texture)
        ];

    for (var i = 0; i < markers.length; i++) {
      markers[i].anchorPoint_set(anchor);
    }

    var vectorMarkerData = new AltusUnified.VectorMarkerData();
    addElementsToVectorAndDelete(vectorMarkerData, markers);

    var markerMap = new AltusUnified.ClusteredMarkerMap("clusteredMarkerMap", vectorMarkerData, 32, 20, AltusUnified.TargetImageFormat.FOUR_BPP, false);

    AltusUnified.scene.addMap(markerMap);
    markerMap.setOrder(312);

    vectorMarkerData.delete();
    markerMap.delete();
    texture.delete();
  }
 function addMarkersAsync() {
    requestData("https://cdn.ba3.us/images/blue-dot.png", function(response) {
        var texture = createTextureFromXhr(response);
        addMarkers(texture);
      }
    );
  }

Provide image data directly using an encoded data url

Using a data url, we can create a JavaScript Image object and set its source to a string containing a base64 encoded version of our PNG.

With this method, we need to render the image to a hidden canvas object.

 function createCanvas(width, height) {
    var canvas = document.createElement('canvas');
    canvas.width = width;
    canvas.height = height;
    return canvas;
  }
 function getImageDataFromCanvas(image) {
    var canvas = createCanvas(image.width, image.height),
        context = canvas.getContext("2d"),
        imageData,
        imageBytes;

    // Copy the image contents to the canvas        
    context.clearRect(0, 0, image.width, image.height);
    context.drawImage(image, 0, 0);

    return context.getImageData(0, 0, image.width, image.height);
  }
 function createTextureFromImageData(imageData, width, height, mipEnabled) {
    var vector = new AltusUnified.VectorByte(),
        image,
        texture;

    pushToVector(vector, imageData);

    image = new AltusUnified.Image(width, height, vector);
    image.multiplyAlpha();

    texture = new AltusUnified.Texture(image, mipEnabled);

    vector.delete();
    image.delete();

    return texture;
  }
 function addMarkersFromEncodedData() {
    var image = new Image(),
        imageData,
        imageBytes,
        texture;

    image.src = "";

    imageData = getImageDataFromCanvas(image);
    imageBytes = new Uint8Array(imageData.data);

    texture = createTextureFromImageData(imageBytes, image.width, image.height, true)
    addMarkers(texture);
  }

Putting it all together

var AltusUnified = new Altus(document.getElementById("AltusDiv"));

function createImageMarkersExample() {

  /*C2*/
  function addBaseMap(mapName, url) {

    // setup tile source
    var internetTileProvider = new AltusUnified.InternetTileProvider(mapName, url);

    // create map description
    var mapDesc = AltusUnified.VirtualMap.defaultRasterMapDesc();
    var newMap = new AltusUnified.VirtualMap(mapName, mapDesc, internetTileProvider);

    // add map to scene
    AltusUnified.scene.addMap(newMap);

    newMap.delete();
    mapDesc.delete();
    internetTileProvider.delete();
  };
  /*C2*/

  function pushToVector(vector, array) {
    var size = array.length;
    for (var i = 0; i < size; i++) {
      vector.push_back(array[i]);
    }
  }

  function deleteElements(array) {
    var size = array.length;
    for (var i = 0; i < size; i++) {
      array[i].delete();
    }
    array.length = 0;
  }

  function addElementsToVectorAndDelete(vector, array) {
    pushToVector(vector, array);
    deleteElements(array);
  }

  /*C3*/
  function requestData(url, onLoad, onError, onTimeout, timeout) {
    var xhr = new XMLHttpRequest();
    xhr.open('GET', url, true);
    xhr.timeout = timeout || 60000; // default to 60 seconds
    xhr.responseType = 'arraybuffer';

    //Handle download timeouts
    xhr.ontimeout = onTimeout || function(e) {
      console.log(e)
    };

    //Handle successful downloads
    xhr.onload = function() {
      if (xhr.status == 200 && xhr.response) {
        onLoad(xhr.response);
      } else {
        onError(e);
      }
    };

    //Handle other errors
    xhr.onerror = onError || function(e) {
      console.error(e);
    };

    //Send the download request
    xhr.send(null);
  }
  /*C3*/

  /*C4*/
  function createTextureFromImageData(imageData, width, height, mipEnabled) {
    var vector = new AltusUnified.VectorByte(),
        image,
        texture;

    pushToVector(vector, imageData);

    image = new AltusUnified.Image(width, height, vector);
    image.multiplyAlpha();

    texture = new AltusUnified.Texture(image, mipEnabled);

    vector.delete();
    image.delete();

    return texture;
  }
  /*C4*/

  /*C5*/
  function createTextureFromXhr(response) {
    var byteArray = new Uint8Array(response),
        vector = new AltusUnified.VectorByte(),
        altusImage = new AltusUnified.Image(),
        texture;

    pushToVector(vector, byteArray);

    altusImage.loadFromMemory(vector);
    altusImage.multiplyAlpha();

    texture = new AltusUnified.Texture(altusImage, false);

    vector.delete();
    altusImage.delete();

    return texture;
  }
  /*C5*/

  /*C6*/
  function createCanvas(width, height) {
    var canvas = document.createElement('canvas');
    canvas.width = width;
    canvas.height = height;
    return canvas;
  }
  /*C6*/

  /*C7*/
  function getImageDataFromCanvas(image) {
    var canvas = createCanvas(image.width, image.height),
        context = canvas.getContext("2d"),
        imageData,
        imageBytes;

    // Copy the image contents to the canvas        
    context.clearRect(0, 0, image.width, image.height);
    context.drawImage(image, 0, 0);

    return context.getImageData(0, 0, image.width, image.height);
  }
  /*C7*/

  /*C8*/
  function addMarkers(texture) {
    var anchor = new AltusUnified.vec2d(texture.getImageWidth() / 2, texture.getImageHeight()), //anchor to bottom center
        markers = [
          new AltusUnified.MarkerData(0, 38.889469, -77.035258, 0, 1, "Washington Monument", texture),
          new AltusUnified.MarkerData(1, 38.889803, -77.009114, 0, 2, "United States Capital", texture),
          new AltusUnified.MarkerData(2, 38.8977, -77.0365, 0, 3, "White House", texture),
          new AltusUnified.MarkerData(3, 38.8893, -77.050111, 0, 4, "Lincoln Memorial", texture),
          new AltusUnified.MarkerData(4, 38.8957, -77.0559, 0, 5, "Kennedy Center", texture),
          new AltusUnified.MarkerData(5, 38.891111, -77.047778, 0, 6, "Vietnam Memorial", texture),
          new AltusUnified.MarkerData(6, 38.888, -77.02, 0, 7, "Air & Space Museum", texture),
          new AltusUnified.MarkerData(7, 38.89731, -77.00626, 0, 8, "Union Station", texture)
        ];

    for (var i = 0; i < markers.length; i++) {
      markers[i].anchorPoint_set(anchor);
    }

    var vectorMarkerData = new AltusUnified.VectorMarkerData();
    addElementsToVectorAndDelete(vectorMarkerData, markers);

    var markerMap = new AltusUnified.ClusteredMarkerMap("clusteredMarkerMap", vectorMarkerData, 32, 20, AltusUnified.TargetImageFormat.FOUR_BPP, false);

    AltusUnified.scene.addMap(markerMap);
    markerMap.setOrder(312);

    vectorMarkerData.delete();
    markerMap.delete();
    texture.delete();
  }
  /*C8*/

  function setPosition(lat, lon, altitude) {
    // create position object
    var pos = new AltusUnified.GeographicPosition(lat, lon, altitude);

    // create orientation object - camera pointed like standard 2D map view
    var orientation = new AltusUnified.Orientation(0, 90, 0);

    // create default scale object
    var scale = new AltusUnified.vec3d(1, 1, 1);

    // set transfrom to scene
    var trans = new AltusUnified.Transform(pos, orientation, scale);
    AltusUnified.scene.camera().transform.set(trans);

    pos.delete();
    orientation.delete();
    scale.delete();
  }

  /*C9*/
  function addMarkersFromEncodedData() {
    var image = new Image(),
        imageData,
        imageBytes,
        texture;

    image.src = "";

    imageData = getImageDataFromCanvas(image);
    imageBytes = new Uint8Array(imageData.data);

    texture = createTextureFromImageData(imageBytes, image.width, image.height, true)
    addMarkers(texture);
  }
  /*C9*/

  /*C10*/
  function addMarkersFromImgTag() {
    var image = getMarkerImage(), // window scoped function
        imageData,
        imageBytes,
        texture;

    imageData = getImageDataFromCanvas(image);
    imageBytes = new Uint8Array(imageData.data);

    texture = createTextureFromImageData(imageBytes, image.width, image.height, true)
    addMarkers(texture);

  }
  /*C10*/

  /*C11*/
  function addMarkersAsync() {
    requestData("https://cdn.ba3.us/images/blue-dot.png", function(response) {
        var texture = createTextureFromXhr(response);
        addMarkers(texture);
      }
    );
  }
  /*C11*/

  return {
    addBaseMap: addBaseMap,
    addMarkersFromEncodedData: addMarkersFromEncodedData,
    addMarkersFromImgTag: addMarkersFromImgTag,
    addMarkersAsync: addMarkersAsync,
    setPosition: setPosition
  }
}

// Called by the mapping engine after it has initialized
function onAltusEngineReady() {
  var ImageMarkersExample = createImageMarkersExample(),
      mapName = "MapBox Aerial",
      tileProviderUrl = "https://a.tiles.mapbox.com/v4/dxjacob.ho6k3ag9/{z}/{x}/{y}.jpg?access_token=pk.eyJ1IjoiZHhqYWNvYiIsImEiOiJwYXotMmtVIn0.rvNzd7EZTKqynbx-9BQdtA";

  ImageMarkersExample.addBaseMap(mapName, tileProviderUrl);

  // ImageMarkersExample.addMarkersFromEncodedData();
  // ImageMarkersExample.addMarkersFromImgTag();
  ImageMarkersExample.addMarkersAsync();

  //Position of camera 
  ImageMarkersExample.setPosition(38.889469, -77.035258, 10000);

  AltusUnified.scene.atmospherics().setSunLocationType(AltusUnified.LocationType.DIRECTION_VIEW_OFFSET); //no dark side
  AltusUnified.scene.atmospherics().setLightingType(AltusUnified.LightingType.REALISTIC); //blue sky

  console.log("Version Hash - " + AltusUnified.Scene.versionHash());
  console.log("Version Tag - " + AltusUnified.Scene.versionTag());
};

See Demo


AltusMappingEngine Web v2.0.ut.2153.g60764257e master

COPYRIGHT (C) 2017, BA3, LLC ALL RIGHTS RESERVED