Map/LeafletVisualizer.js

"use strict";

import i18next from "i18next";
/*global require*/
var L = require("leaflet");
var AssociativeArray = require("terriajs-cesium/Source/Core/AssociativeArray")
  .default;
var Cartesian2 = require("terriajs-cesium/Source/Core/Cartesian2").default;
var Cartesian3 = require("terriajs-cesium/Source/Core/Cartesian3").default;
var Cartographic = require("terriajs-cesium/Source/Core/Cartographic").default;
var CesiumMath = require("terriajs-cesium/Source/Core/Math").default;
var Color = require("terriajs-cesium/Source/Core/Color").default;
var defined = require("terriajs-cesium/Source/Core/defined").default;
var destroyObject = require("terriajs-cesium/Source/Core/destroyObject")
  .default;
var DeveloperError = require("terriajs-cesium/Source/Core/DeveloperError")
  .default;
var Ellipsoid = require("terriajs-cesium/Source/Core/Ellipsoid").default;
var Property = require("terriajs-cesium/Source/DataSources/Property").default;
var writeTextToCanvas = require("terriajs-cesium/Source/Core/writeTextToCanvas")
  .default;
var PolylineGlowMaterialProperty = require("terriajs-cesium/Source/DataSources/PolylineGlowMaterialProperty")
  .default;

var defaultColor = Color.WHITE;
var defaultOutlineColor = Color.BLACK;
var defaultOutlineWidth = 1.0;
var defaultPixelSize = 5.0;

var defaultWidth = 5.0;

//NOT IMPLEMENTED
// Path primitive - no need identified
// Ellipse primitive - no need identified
// Ellipsoid primitive - 3d prim - no plans for this
// Model primitive - 3d prim - no plans for this

/**
 * A {@link Visualizer} which maps {@link Entity#point} to Leaflet primitives.
 * @alias LeafletGeomVisualizer
 * @constructor
 *
 * @param {LeafletScene} leafletScene The Leaflet scene that the the primitives will be rendered in.
 * @param {EntityCollection} entityCollection The entityCollection to visualize.
 */
var LeafletGeomVisualizer = function(leafletScene, entityCollection) {
  if (!defined(leafletScene)) {
    throw new DeveloperError(i18next.t("map.leafletVisualizer.devError1"));
  }
  if (!defined(entityCollection)) {
    throw new DeveloperError(i18next.t("map.leafletVisualizer.devError2"));
  }

  var featureGroup = L.featureGroup().addTo(leafletScene.map);
  entityCollection.collectionChanged.addEventListener(
    LeafletGeomVisualizer.prototype._onCollectionChanged,
    this
  );

  this._leafletScene = leafletScene;
  this._featureGroup = featureGroup;
  this._entityCollection = entityCollection;
  this._entitiesToVisualize = new AssociativeArray();
  this._entityHash = {};

  this._onCollectionChanged(entityCollection, entityCollection.values, [], []);
};

LeafletGeomVisualizer.prototype._onCollectionChanged = function(
  entityCollection,
  added,
  removed,
  changed
) {
  var i;
  var entity;
  var featureGroup = this._featureGroup;
  var entities = this._entitiesToVisualize;
  var entityHash = this._entityHash;

  for (i = added.length - 1; i > -1; i--) {
    entity = added[i];
    if (
      ((defined(entity._point) ||
        defined(entity._billboard) ||
        defined(entity._label)) &&
        defined(entity._position)) ||
      defined(entity._polyline) ||
      defined(entity._polygon)
    ) {
      entities.set(entity.id, entity);
      entityHash[entity.id] = {};
    }
  }

  for (i = changed.length - 1; i > -1; i--) {
    entity = changed[i];
    if (
      ((defined(entity._point) ||
        defined(entity._billboard) ||
        defined(entity._label)) &&
        defined(entity._position)) ||
      defined(entity._polyline) ||
      defined(entity._polygon)
    ) {
      entities.set(entity.id, entity);
      entityHash[entity.id] = entityHash[entity.id] || {};
    } else {
      cleanEntity(entity, featureGroup, entityHash);
      entities.remove(entity.id);
    }
  }

  for (i = removed.length - 1; i > -1; i--) {
    entity = removed[i];
    cleanEntity(entity, featureGroup, entityHash);
    entities.remove(entity.id);
  }
};

function cleanEntity(entity, group, entityHash) {
  var details = entityHash[entity.id];

  cleanPoint(entity, group, details);
  cleanPolygon(entity, group, details);
  cleanBillboard(entity, group, details);
  cleanLabel(entity, group, details);
  cleanPolyline(entity, group, details);

  delete entityHash[entity.id];
}

function cleanPoint(entity, group, details) {
  if (defined(details.point)) {
    group.removeLayer(details.point.layer);
    details.point = undefined;
  }
}

function cleanPolygon(entity, group, details) {
  if (defined(details.polygon)) {
    group.removeLayer(details.polygon.layer);
    details.polygon = undefined;
  }
}

function cleanBillboard(entity, group, details) {
  if (defined(details.billboard)) {
    group.removeLayer(details.billboard.layer);
    details.billboard = undefined;
  }
}

function cleanLabel(entity, group, details) {
  if (defined(details.label)) {
    group.removeLayer(details.label.layer);
    details.label = undefined;
  }
}

function cleanPolyline(entity, group, details) {
  if (defined(details.polyline)) {
    group.removeLayer(details.polyline.layer);
    details.polyline = undefined;
  }
}

/**
 * A variable to store what sort of bounds our leaflet map has been looking at
 * 0 = a normal extent
 * 1 = zoomed in close to the east/left of anti-meridian
 * 2 = zoomed in close to the west/right of the anti-meridian
 * When this value changes we'll need to recompute the location of our points
 * to help them wrap around the anti-meridian
 */
let prevBoundsType = 0;

/**
 * Updates the primitives created by this visualizer to match their
 * Entity counterpart at the given time.
 *
 * @param {JulianDate} time The time to update to.
 * @returns {Boolean} This function always returns true.
 */
LeafletGeomVisualizer.prototype.update = function(time) {
  //>>includeStart('debug', pragmas.debug);
  if (!defined(time)) {
    throw new DeveloperError(i18next.t("map.leafletVisualizer.devError3"));
  }
  //>>includeEnd('debug');
  const bounds = this._leafletScene.map.getBounds();
  let applyLocalisedAntiMeridianFix = false;
  let currentBoundsType = 0;
  if (isCloseToEasternAntiMeridian(bounds)) {
    applyLocalisedAntiMeridianFix = true;
    currentBoundsType = 1;
  } else if (isCloseToWesternAntiMeridian(bounds)) {
    applyLocalisedAntiMeridianFix = true;
    currentBoundsType = 2;
  }

  var entities = this._entitiesToVisualize.values;
  var entityHash = this._entityHash;
  for (var i = 0, len = entities.length; i < len; i++) {
    var entity = entities[i];
    var entityDetails = entityHash[entity.id];

    if (defined(entity._point)) {
      this._updatePoint(
        entity,
        time,
        entityHash,
        entityDetails,
        applyLocalisedAntiMeridianFix === true ? bounds : null,
        prevBoundsType !== currentBoundsType
      );
    }
    if (defined(entity._billboard)) {
      this._updateBillboard(
        entity,
        time,
        entityHash,
        entityDetails,
        applyLocalisedAntiMeridianFix === true ? bounds : null
      );
    }
    if (defined(entity._label)) {
      this._updateLabel(entity, time, entityHash, entityDetails);
    }
    if (defined(entity._polyline)) {
      this._updatePolyline(entity, time, entityHash, entityDetails);
    }
    if (defined(entity._polygon)) {
      this._updatePolygon(entity, time, entityHash, entityDetails);
    }
  }
  prevBoundsType = currentBoundsType;
  return true;
};

/**
 * Computes the rectangular bounds which encloses the collection of
 * entities to be visualized.
 *
 * @returns {LatLngBounds} The computed bounds.
 */
LeafletGeomVisualizer.prototype.getLatLngBounds = function() {
  let result;

  Object.keys(this._entityHash).forEach(entityId => {
    const entityDetails = this._entityHash[entityId];

    Object.keys(entityDetails).forEach(primitiveId => {
      const primitive = entityDetails[primitiveId];

      if (defined(primitive.layer)) {
        if (defined(primitive.layer.getBounds)) {
          const bounds = primitive.layer.getBounds();
          if (defined(bounds)) {
            result =
              result === undefined
                ? L.latLngBounds(bounds.getSouthWest(), bounds.getNorthEast())
                : result.extend(bounds);
          }
        }
        if (defined(primitive.layer.getLatLng)) {
          const latLng = primitive.layer.getLatLng();
          if (defined(latLng)) {
            result =
              result === undefined
                ? L.latLngBounds([latLng])
                : result.extend(latLng);
          }
        }
      }
    });
  });

  return result;
};

/**
 * Computes whether we're looking close to the east of the anti-meridian
 * Our western viewing extent must be > 140 degrees
 * @returns boolean
 */
function isCloseToEasternAntiMeridian(bounds) {
  const w = bounds.getWest();
  const e = bounds.getEast();
  if (w > 140 && (e < -140 || e > 180)) {
    return true;
  }
  return false;
}

/**
 * Computes whether we're looking close to the west of the anti-meridian
 * Our eastern viewing extent must be < -140 degrees
 * @returns boolean
 */
function isCloseToWesternAntiMeridian(bounds) {
  const w = bounds.getWest();
  const e = bounds.getEast();
  if ((w > 180 || w < -140) && e < -140) {
    return true;
  }
  return false;
}

var cartographicScratch = new Cartographic();

function positionToLatLng(position, bounds) {
  var cartographic = Ellipsoid.WGS84.cartesianToCartographic(
    position,
    cartographicScratch
  );

  let lon = CesiumMath.toDegrees(cartographic.longitude);
  if (bounds !== null) {
    if (isCloseToEasternAntiMeridian(bounds)) {
      if (lon < -140) {
        lon = lon + 360;
      }
    } else if (isCloseToWesternAntiMeridian(bounds)) {
      if (lon > 140) {
        lon = lon - 360;
      }
    }
  }

  return L.latLng(CesiumMath.toDegrees(cartographic.latitude), lon);
}

LeafletGeomVisualizer.prototype._updatePoint = function(
  entity,
  time,
  entityHash,
  entityDetails,
  bounds,
  boundsJustChanged
) {
  var featureGroup = this._featureGroup;
  var pointGraphics = entity._point;

  var show =
    entity.isAvailable(time) &&
    Property.getValueOrDefault(pointGraphics._show, time, true);
  if (!show) {
    cleanPoint(entity, featureGroup, entityDetails);
    return;
  }

  var details = entityDetails.point;
  if (!defined(details)) {
    details = entityDetails.point = {
      layer: undefined,
      lastPosition: new Cartesian3(),
      lastPixelSize: 1,
      lastColor: new Color(),
      lastOutlineColor: new Color(),
      lastOutlineWidth: 1
    };
  }

  var position = Property.getValueOrUndefined(entity._position, time);
  if (!defined(position)) {
    cleanPoint(entity, featureGroup, entityDetails);
    return;
  }

  var pixelSize = Property.getValueOrDefault(
    pointGraphics._pixelSize,
    time,
    defaultPixelSize
  );
  var color = Property.getValueOrDefault(
    pointGraphics._color,
    time,
    defaultColor
  );
  var outlineColor = Property.getValueOrDefault(
    pointGraphics._outlineColor,
    time,
    defaultOutlineColor
  );
  var outlineWidth = Property.getValueOrDefault(
    pointGraphics._outlineWidth,
    time,
    defaultOutlineWidth
  );

  var layer = details.layer;

  if (!defined(layer)) {
    var pointOptions = {
      radius: pixelSize / 2.0,
      fillColor: color.toCssColorString(),
      fillOpacity: color.alpha,
      color: outlineColor.toCssColorString(),
      weight: outlineWidth,
      opacity: outlineColor.alpha
    };

    layer = details.layer = L.circleMarker(
      positionToLatLng(position, bounds),
      pointOptions
    );
    layer.on("click", featureClicked.bind(undefined, this, entity));
    layer.on("mousedown", featureMousedown.bind(undefined, this, entity));
    featureGroup.addLayer(layer);

    Cartesian3.clone(position, details.lastPosition);
    details.lastPixelSize = pixelSize;
    Color.clone(color, details.lastColor);
    Color.clone(outlineColor, details.lastOutlineColor);
    details.lastOutlineWidth = outlineWidth;

    return layer;
  }

  if (!Cartesian3.equals(position, details.lastPosition) || boundsJustChanged) {
    layer.setLatLng(positionToLatLng(position, bounds));
    Cartesian3.clone(position, details.lastPosition);
  }

  if (pixelSize !== details.lastPixelSize) {
    layer.setRadius(pixelSize / 2.0);
    details.lastPixelSize = pixelSize;
  }

  var options = layer.options;
  var applyStyle = false;

  if (!Color.equals(color, details.lastColor)) {
    options.fillColor = color.toCssColorString();
    options.fillOpacity = color.alpha;
    Color.clone(color, details.lastColor);
    applyStyle = true;
  }

  if (!Color.equals(outlineColor, details.lastOutlineColor)) {
    options.color = outlineColor.toCssColorString();
    options.opacity = outlineColor.alpha;
    Color.clone(outlineColor, details.lastOutlineColor);
    applyStyle = true;
  }

  if (outlineWidth !== details.lastOutlineWidth) {
    options.weight = outlineWidth;
    details.lastOutlineWidth = outlineWidth;
    applyStyle = true;
  }

  if (applyStyle) {
    layer.setStyle(options);
  }
  return layer;
};

LeafletGeomVisualizer.prototype._updatePolygon = function(
  entity,
  time,
  entityHash,
  entityDetails
) {
  var featureGroup = this._featureGroup;
  var polygonGraphics = entity._polygon;

  var show =
    entity.isAvailable(time) &&
    Property.getValueOrDefault(polygonGraphics._show, time, true);
  if (!show) {
    cleanPolygon(entity, featureGroup, entityDetails);
    return;
  }

  var details = entityDetails.polygon;
  if (!defined(details)) {
    details = entityDetails.polygon = {
      layer: undefined,
      lastHierarchy: undefined,
      lastFill: undefined,
      lastFillColor: new Color(),
      lastOutline: undefined,
      lastOutlineColor: new Color()
    };
  }

  var hierarchy = Property.getValueOrUndefined(
    polygonGraphics._hierarchy,
    time
  );
  if (!defined(hierarchy)) {
    cleanPolygon(entity, featureGroup, entityDetails);
    return;
  }

  var fill = Property.getValueOrDefault(polygonGraphics._fill, time, true);
  var outline = Property.getValueOrDefault(
    polygonGraphics._outline,
    time,
    true
  );
  var outlineColor = Property.getValueOrDefault(
    polygonGraphics._outlineColor,
    time,
    defaultOutlineColor
  );

  var material = Property.getValueOrUndefined(polygonGraphics._material, time);
  var fillColor;
  if (defined(material) && defined(material.color)) {
    fillColor = material.color;
  } else {
    fillColor = defaultColor;
  }

  var layer = details.layer;
  if (!defined(layer)) {
    var polygonOptions = {
      fill: fill,
      fillColor: fillColor.toCssColorString(),
      fillOpacity: fillColor.alpha,
      weight: outline ? 1.0 : 0.0,
      color: outlineColor.toCssColorString(),
      opacity: outlineColor.alpha
    };

    layer = details.layer = L.polygon(
      hierarchyToLatLngs(hierarchy),
      polygonOptions
    );

    layer.on("click", featureClicked.bind(undefined, this, entity));
    layer.on("mousedown", featureMousedown.bind(undefined, this, entity));
    featureGroup.addLayer(layer);

    details.lastHierarchy = hierarchy;
    details.lastFill = fill;
    details.lastOutline = outline;
    Color.clone(fillColor, details.lastFillColor);
    Color.clone(outlineColor, details.lastOutlineColor);

    return;
  }

  if (hierarchy !== details.lastHierachy) {
    layer.setLatLngs(hierarchyToLatLngs(hierarchy));
    details.lastHierachy = hierarchy;
  }

  var options = layer.options;
  var applyStyle = false;

  if (fill !== details.lastFill) {
    options.fill = fill;
    details.lastFill = fill;
    applyStyle = true;
  }

  if (outline !== details.lastOutline) {
    options.weight = outline ? 1.0 : 0.0;
    details.lastOutline = outline;
    applyStyle = true;
  }

  if (!Color.equals(fillColor, details.lastFillColor)) {
    options.fillColor = fillColor.toCssColorString();
    options.fillOpacity = fillColor.alpha;
    Color.clone(fillColor, details.lastFillColor);
    applyStyle = true;
  }

  if (!Color.equals(outlineColor, details.lastOutlineColor)) {
    options.color = outlineColor.toCssColorString();
    options.opacity = outlineColor.alpha;
    Color.clone(outlineColor, details.lastOutlineColor);
    applyStyle = true;
  }

  if (applyStyle) {
    layer.setStyle(options);
  }
};

function hierarchyToLatLngs(hierarchy) {
  // This function currently does not handle polygons with holes.

  var positions = Array.isArray(hierarchy) ? hierarchy : hierarchy.positions;

  return convertEntityPositionsToLatLons(positions);
}

//Recolor an image using 2d canvas
function recolorBillboard(img, color) {
  var canvas = document.createElement("canvas");
  canvas.width = img.width;
  canvas.height = img.height;

  // Copy the image contents to the canvas
  var context = canvas.getContext("2d");
  context.drawImage(img, 0, 0);
  var image = context.getImageData(0, 0, canvas.width, canvas.height);
  var normClr = [color.red, color.green, color.blue, color.alpha];

  var length = image.data.length; //pixel count * 4
  for (var i = 0; i < length; i += 4) {
    for (var j = 0; j < 4; j++) {
      image.data[j + i] *= normClr[j];
    }
  }

  context.putImageData(image, 0, 0);
  return canvas.toDataURL();
  //    return context.getImageData(0, 0, canvas.width, canvas.height);
}

//Single pixel black dot
var tmpImage =
  "";

//NYI: currently skipping all the camera distance related properties
LeafletGeomVisualizer.prototype._updateBillboard = function(
  entity,
  time,
  entityHash,
  entityDetails,
  bounds
) {
  var markerGraphics = entity._billboard;
  var featureGroup = this._featureGroup;
  var position, marker;

  var details = entityDetails.billboard;
  if (!defined(details)) {
    details = entityDetails.billboard = {
      layer: undefined
    };
  }

  var geomLayer = details.layer;

  var show =
    entity.isAvailable(time) &&
    Property.getValueOrDefault(markerGraphics._show, time, true);
  if (show) {
    position = Property.getValueOrUndefined(entity._position, time);
    show = defined(position);
  }
  if (!show) {
    cleanBillboard(entity, featureGroup, entityDetails);
    return;
  }

  var latlng = positionToLatLng(position, bounds);
  var image = Property.getValueOrDefault(
    markerGraphics._image,
    time,
    undefined
  );
  var height = Property.getValueOrDefault(
    markerGraphics._height,
    time,
    undefined
  );
  var width = Property.getValueOrDefault(
    markerGraphics._width,
    time,
    undefined
  );
  var color = Property.getValueOrDefault(
    markerGraphics._color,
    time,
    defaultColor
  );
  var scale = Property.getValueOrDefault(markerGraphics._scale, time, 1.0);
  var verticalOrigin = Property.getValueOrDefault(
    markerGraphics._verticalOrigin,
    time,
    0
  );
  var horizontalOrigin = Property.getValueOrDefault(
    markerGraphics._horizontalOrigin,
    time,
    0
  );
  var pixelOffset = Property.getValueOrDefault(
    markerGraphics._pixelOffset,
    time,
    Cartesian2.ZERO
  );

  var imageUrl;
  if (defined(image)) {
    if (typeof image === "string") {
      imageUrl = image;
    } else if (defined(image.toDataURL)) {
      imageUrl = image.toDataURL();
    } else if (defined(image.url)) {
      imageUrl = image.url;
    } else {
      imageUrl = image.src;
    }
  }

  var iconOptions = {
    color: color.toCssColorString(),
    origUrl: imageUrl,
    scale: scale,
    horizontalOrigin: horizontalOrigin, //value: left, center, right
    verticalOrigin: verticalOrigin //value: bottom, center, top
  };

  if (defined(height) || defined(width)) {
    iconOptions.iconSize = [width, height];
  }

  var redrawIcon = false;
  if (!defined(geomLayer)) {
    var markerOptions = { icon: L.icon({ iconUrl: tmpImage }) };
    marker = L.marker(latlng, markerOptions);
    marker.on("click", featureClicked.bind(undefined, this, entity));
    marker.on("mousedown", featureMousedown.bind(undefined, this, entity));
    featureGroup.addLayer(marker);
    details.layer = marker;
    redrawIcon = true;
  } else {
    marker = geomLayer;
    if (!marker._latlng.equals(latlng)) {
      marker.setLatLng(latlng);
    }
    for (var prop in iconOptions) {
      if (iconOptions[prop] !== marker.options.icon.options[prop]) {
        redrawIcon = true;
        break;
      }
    }
  }

  if (redrawIcon) {
    var drawBillboard = function(image, dataurl) {
      iconOptions.iconUrl = dataurl || image;
      if (!defined(iconOptions.iconSize)) {
        iconOptions.iconSize = [image.width * scale, image.height * scale];
      }
      var w = iconOptions.iconSize[0],
        h = iconOptions.iconSize[1];
      var xOff = (w / 2) * (1 - horizontalOrigin) - pixelOffset.x;
      var yOff = (h / 2) * (1 + verticalOrigin) - pixelOffset.y;
      iconOptions.iconAnchor = [xOff, yOff];

      if (!color.equals(defaultColor)) {
        iconOptions.iconUrl = recolorBillboard(image, color);
      }
      marker.setIcon(L.icon(iconOptions));
    };
    var img = new Image();
    img.onload = function() {
      drawBillboard(img, imageUrl);
    };
    img.src = imageUrl;
  }
};

LeafletGeomVisualizer.prototype._updateLabel = function(
  entity,
  time,
  entityHash,
  entityDetails
) {
  var labelGraphics = entity._label;
  var featureGroup = this._featureGroup;
  var position, marker;

  var details = entityDetails.label;
  if (!defined(details)) {
    details = entityDetails.label = {
      layer: undefined
    };
  }

  var geomLayer = details.layer;

  var show =
    entity.isAvailable(time) &&
    Property.getValueOrDefault(labelGraphics._show, time, true);
  if (show) {
    position = Property.getValueOrUndefined(entity._position, time);
    show = defined(position);
  }
  if (!show) {
    cleanLabel(entity, featureGroup, entityDetails);
    return;
  }

  var cart = Ellipsoid.WGS84.cartesianToCartographic(position);
  var latlng = L.latLng(
    CesiumMath.toDegrees(cart.latitude),
    CesiumMath.toDegrees(cart.longitude)
  );
  var text = Property.getValueOrDefault(labelGraphics._text, time, undefined);
  var font = Property.getValueOrDefault(labelGraphics._font, time, undefined);
  var scale = Property.getValueOrDefault(labelGraphics._scale, time, 1.0);
  var fillColor = Property.getValueOrDefault(
    labelGraphics._fillColor,
    time,
    defaultColor
  );
  var verticalOrigin = Property.getValueOrDefault(
    labelGraphics._verticalOrigin,
    time,
    0
  );
  var horizontalOrigin = Property.getValueOrDefault(
    labelGraphics._horizontalOrigin,
    time,
    0
  );
  var pixelOffset = Property.getValueOrDefault(
    labelGraphics._pixelOffset,
    time,
    Cartesian2.ZERO
  );

  var iconOptions = {
    text: text,
    font: font,
    color: fillColor.toCssColorString(),
    scale: scale,
    horizontalOrigin: horizontalOrigin, //value: left, center, right
    verticalOrigin: verticalOrigin //value: bottom, center, top
  };

  var redrawLabel = false;
  if (!defined(geomLayer)) {
    var markerOptions = { icon: L.icon({ iconUrl: tmpImage }) };
    marker = L.marker(latlng, markerOptions);
    marker.on("click", featureClicked.bind(undefined, this, entity));
    marker.on("mousedown", featureMousedown.bind(undefined, this, entity));
    featureGroup.addLayer(marker);
    details.layer = marker;
    redrawLabel = true;
  } else {
    marker = geomLayer;
    if (!marker._latlng.equals(latlng)) {
      marker.setLatLng(latlng);
    }
    for (var prop in iconOptions) {
      if (iconOptions[prop] !== marker.options.icon.options[prop]) {
        redrawLabel = true;
        break;
      }
    }
  }

  if (redrawLabel) {
    var drawBillboard = function(image, dataurl) {
      iconOptions.iconUrl = dataurl || image;
      if (!defined(iconOptions.iconSize)) {
        iconOptions.iconSize = [image.width * scale, image.height * scale];
      }
      var w = iconOptions.iconSize[0],
        h = iconOptions.iconSize[1];
      var xOff = (w / 2) * (1 - horizontalOrigin) - pixelOffset.x;
      var yOff = (h / 2) * (1 + verticalOrigin) - pixelOffset.y;
      iconOptions.iconAnchor = [xOff, yOff];
      marker.setIcon(L.icon(iconOptions));
    };

    var canvas = writeTextToCanvas(text, { fillColor: fillColor, font: font });
    var imageUrl = canvas.toDataURL();

    var img = new Image();
    img.onload = function() {
      drawBillboard(img, imageUrl);
    };
    img.src = imageUrl;
  }
};

function convertEntityPositionsToLatLons(positions) {
  var carts = Ellipsoid.WGS84.cartesianArrayToCartographicArray(positions);
  var latlngs = [];
  let lastLongitude = null;
  for (var p = 0; p < carts.length; p++) {
    let lon = CesiumMath.toDegrees(carts[p].longitude);

    if (lastLongitude - lon > 180) {
      lon = lon + 360;
    } else if (lastLongitude - lon < -180) {
      lon = lon - 360;
    }

    latlngs.push(L.latLng(CesiumMath.toDegrees(carts[p].latitude), lon));
    lastLongitude = lon;
  }
  return latlngs;
}

LeafletGeomVisualizer.prototype._updatePolyline = function(
  entity,
  time,
  entityHash,
  entityDetails
) {
  var polylineGraphics = entity._polyline;
  var featureGroup = this._featureGroup;
  var positions, polyline;

  var details = entityDetails.polyline;
  if (!defined(details)) {
    details = entityDetails.polyline = {
      layer: undefined
    };
  }

  var geomLayer = details.layer;

  var show =
    entity.isAvailable(time) &&
    Property.getValueOrDefault(polylineGraphics._show, time, true);
  if (show) {
    positions = Property.getValueOrUndefined(polylineGraphics._positions, time);
    show = defined(positions);
  }
  if (!show) {
    cleanPolyline(entity, featureGroup, entityDetails);
    return;
  }

  const latlngs = convertEntityPositionsToLatLons(positions);

  var color;
  var width;
  if (polylineGraphics._material instanceof PolylineGlowMaterialProperty) {
    color = defaultColor;
    width = defaultWidth;
  } else {
    color = Property.getValueOrDefault(
      polylineGraphics._material.color,
      time,
      defaultColor
    );
    width = Property.getValueOrDefault(
      polylineGraphics._width,
      time,
      defaultWidth
    );
  }

  var polylineOptions = {
    color: color.toCssColorString(),
    weight: width,
    opacity: color.alpha
  };

  if (!defined(geomLayer)) {
    if (latlngs.length > 0) {
      polyline = L.polyline(latlngs, polylineOptions);
      polyline.on("click", featureClicked.bind(undefined, this, entity));
      polyline.on("mousedown", featureMousedown.bind(undefined, this, entity));
      featureGroup.addLayer(polyline);
      details.layer = polyline;
    }
  } else {
    polyline = geomLayer;
    var curLatLngs = polyline.getLatLngs();
    var bPosChange = latlngs.length !== curLatLngs.length;
    for (var i = 0; i < curLatLngs.length && !bPosChange; i++) {
      if (!curLatLngs[i].equals(latlngs[i])) {
        bPosChange = true;
      }
    }
    if (bPosChange) {
      polyline.setLatLngs(latlngs);
    }
    for (var prop in polylineOptions) {
      if (polylineOptions[prop] !== polyline.options[prop]) {
        polyline.setStyle(polylineOptions);
        break;
      }
    }
  }
};

/**
 * Returns true if this object was destroyed; otherwise, false.
 *
 * @returns {Boolean} True if this object was destroyed; otherwise, false.
 */
LeafletGeomVisualizer.prototype.isDestroyed = function() {
  return false;
};

/**
 * Removes and destroys all primitives created by this instance.
 */
LeafletGeomVisualizer.prototype.destroy = function() {
  var entities = this._entitiesToVisualize.values;
  var entityHash = this._entityHash;

  for (var i = entities.length - 1; i > -1; i--) {
    cleanEntity(entities[i], this._featureGroup, entityHash);
  }

  this._entityCollection.collectionChanged.removeEventListener(
    LeafletGeomVisualizer.prototype._onCollectionChanged,
    this
  );
  this._leafletScene.map.removeLayer(this._featureGroup);
  return destroyObject(this);
};

////////////////////////////////////////////////////////

var LeafletVisualizer = function() {};

LeafletVisualizer.prototype.visualizersCallback = function(
  leafletScene,
  entityCluster,
  dataSource
) {
  var entities = dataSource.entities;
  return [new LeafletGeomVisualizer(leafletScene, entities)];
};

function featureClicked(visualizer, entity, event) {
  visualizer._leafletScene.featureClicked.raiseEvent(entity, event);
}

function featureMousedown(visualizer, entity, event) {
  visualizer._leafletScene.featureMousedown.raiseEvent(entity, event);
}

module.exports = LeafletVisualizer;