Search code examples
javascriptamchartsamcharts4ammap

Update image in amMap without losing zoom control- AmCharts v4


There is AmChart's amMap with images i.e. imageSeries. The objective is to change the clicked image or say validate the imageSeries without losing the zoom level. In dev mode, the image has been changed using imageURL but how to update it on the chart dynamically?

Jsfiddle

I took reference from the docs Amcharts4 imapimageseriesevents

Click on the area with dense colour i.e. Uttar Pradesh to see the images.

// Themes begin
am4core.useTheme(am4themes_animated);
// Themes end

// Create map instance
var chart = am4core.create("chartdiv", am4maps.MapChart);

// Set map definition
chart.geodata = am4geodata_indiaLow;

// Set projection
chart.projection = new am4maps.projections.Miller();

// Create map polygon series
var polygonSeries = chart.series.push(new am4maps.MapPolygonSeries());

//Set min/max fill color for each area
polygonSeries.heatRules.push({
  property: "fill",
  target: polygonSeries.mapPolygons.template,
  min: chart.colors.getIndex(1).brighten(1),
  max: chart.colors.getIndex(1).brighten(-0.3)
});

// Make map load polygon data (state shapes and names) from GeoJSON
polygonSeries.useGeodata = true;

// Set heatmap values for each state
polygonSeries.data = [
  {
    id: "IN-UP",
    value: 5130632
  }
];

// Set up heat legend
let heatLegend = chart.createChild(am4maps.HeatLegend);
heatLegend.series = polygonSeries;
heatLegend.align = "right";
heatLegend.valign = "bottom";
heatLegend.width = am4core.percent(20);
heatLegend.marginRight = am4core.percent(4);
heatLegend.minValue = 0;
heatLegend.maxValue = 40000000;

// Set up custom heat map legend labels using axis ranges
var minRange = heatLegend.valueAxis.axisRanges.create();
minRange.value = heatLegend.minValue;
minRange.label.text = "Little";
var maxRange = heatLegend.valueAxis.axisRanges.create();
maxRange.value = heatLegend.maxValue;
maxRange.label.text = "A lot!";

// Blank out internal heat legend value axis labels
heatLegend.valueAxis.renderer.labels.template.adapter.add("text", function(labelText) {
  return "";
});

// Configure series tooltip
var polygonTemplate = polygonSeries.mapPolygons.template;
polygonTemplate.applyOnClones = true;
polygonTemplate.togglable = true;
polygonTemplate.tooltipText = "{name}";
polygonTemplate.nonScalingStroke = true;
polygonTemplate.strokeOpacity = 0.5;
polygonTemplate.fill = chart.colors.getIndex(0);

var lastSelected;
polygonTemplate.events.on("hit", function(ev) {
  if (lastSelected) {
    lastSelected.isActive = false;
  }
  ev.target.series.chart.zoomToMapObject(ev.target);
  if (lastSelected !== ev.target) {
    lastSelected = ev.target;
  }
});
/* Create selected and hover states and set alternative fill color */
var ss = polygonTemplate.states.create("active");
ss.properties.fill = chart.colors.getIndex(4);

var hs = polygonTemplate.states.create("hover");
hs.properties.fill = chart.colors.getIndex(2);

var imageSeries = chart.series.push(new am4maps.MapImageSeries());
imageSeries.id = "markers";

var imageTemplate = imageSeries.mapImages.template;
imageTemplate.propertyFields.longitude = "longitude";
imageTemplate.propertyFields.latitude = "latitude";
imageTemplate.nonScaling = true;
let jsonSelected = [];
imageTemplate.events.on("hit", function(ev) {
  ev.target.dataItem.dataContext.imageURL = "";
  if ($.inArray(ev.target.dataItem.dataContext.id, jsonSelected) !== -1) {
    jsonSelected.splice($.inArray(ev.target.dataItem.dataContext.id, jsonSelected), 1);
  } else {
    jsonSelected.push(ev.target.dataItem.dataContext.id);
  }
  $('#list').empty();
  $.each(jsonSelected, (index, value) => {
    $('#list').append('<div class="selected">' + value + '<i id="' + value + '" class="removeItem fas fa-times"></i></div>');
  });
  $(".selected .removeItem").click(function() {
    jsonSelected.splice(jsonSelected.indexOf(+this.id), 1);
    $(this).parents('.selected').remove();
  });
});

var image = imageTemplate.createChild(am4core.Image);
image.propertyFields.href = "imageURL";
image.width = 20;
image.height = 20;
image.horizontalCenter = "middle";
image.verticalCenter = "middle";
imageSeries.data = [{
  "id": 101,
  "minZoomLevel": 2,
  "latitude": 28.340322,
  "longitude": 77.922881,
  "imageURL": "",
  "width": 5,
  "height": 5,
  "label": "[font-size:10]Meerut[/]"
}, {
  "id": 102,
  "minZoomLevel": 2,
  "latitude": 30.340322,
  "longitude": 78.922881,
  "imageURL": "",
  "width": 5,
  "height": 5,
  "label": "[font-size:10]Dehradun[/]"
}, {
  "id": 103,
  "minZoomLevel": 3,
  "latitude": 26.378494,
  "longitude": 80.96383,
  "imageURL": "",
  "width": 5,
  "height": 5,
  "label": "[font-size:10]Lucknow[/]"
}, {
  "id": 104,
  "minZoomLevel": 3,
  "latitude": 26.378494,
  "longitude": 82.96383,
  "imageURL": "",
  "width": 5,
  "height": 5,
  "label": "[font-size:10]Dummy[/]"
}];

var hoverState = image.states.create("hover");
hoverState.properties.scale = 1.9;
image.tooltipText = "{label}";
imageSeries.events.on("datavalidated", updateImageVisibility);
chart.events.on("zoomlevelchanged", updateImageVisibility);

function updateImageVisibility(ev) {
  var chart = ev.target.baseSprite;
  var series = chart.map.getKey("markers");
  series.mapImages.each(function(image) {
    if (image.dataItem.dataContext.minZoomLevel) {
      if (image.dataItem.dataContext.minZoomLevel >= chart.zoomLevel) {
        image.hide();
      } else {
        image.show();
      }
    }
  });
}
body {
  font-family: -apple-system, BlinkMacSystemFont, "Segoe UI", Roboto, Helvetica, Arial, sans-serif, "Apple Color Emoji", "Segoe UI Emoji", "Segoe UI Symbol";
}

#chartdiv {
  width: 100%;
  height: 100%;
}

.selected {
  border: 1px solid red;
  border-radius: 5px;
  display: inline;
  padding-right: 2px;
  margin-right: 4px;
}
<link rel="stylesheet" href="https://stackpath.bootstrapcdn.com/bootstrap/4.3.1/css/bootstrap.min.css">
<link rel="stylesheet" href="https://use.fontawesome.com/releases/v5.8.1/css/all.css">

<script src="https://code.jquery.com/jquery-3.3.1.slim.min.js"></script>
<script src="https://cdnjs.cloudflare.com/ajax/libs/popper.js/1.14.7/umd/popper.min.js"></script>
<script src="https://stackpath.bootstrapcdn.com/bootstrap/4.3.1/js/bootstrap.min.js"></script>
<script src="https://www.amcharts.com/lib/4/core.js"></script>
<script src="https://www.amcharts.com/lib/4/maps.js"></script>
<script src="https://www.amcharts.com/lib/4/geodata/indiaLow.js"></script>
<script src="https://www.amcharts.com/lib/4/themes/animated.js"></script>
<div id="list"></div>
<div id="chartdiv"></div>


Solution

  • It's still not very clear what you're trying to do. When you click an image, what image are you trying to change it to? Why would that affect the zoom level?

    I checked out your jsfiddle and instead of zoom level, it's the chart itself shifting up and down because you're populating an element that previously had no height. If you give it a min-height the chart will stop shifting up and down whenever it gains a child for the first time or loses them all.

    Here's a fork that does that:

    #list {
      min-height: 25px;
    }
    

    Addendum 2019-06-22: Change image on clicking mapImage

    To change the image on click, instead of attempting to update the original data for property field binding, please update the href property directly.

    I.e. instead of:

    ev.target.dataItem.dataContext.imageURL = "data:image/png;base64,...";
    

    Just do:

    ev.target.href = "data:image/png;base64,...";
    

    Demo below has been updated to include this, click Uttar Pradesh -> then Meerut (the icon that has a hand/pointer):

    https://jsfiddle.net/notacouch/d6u9ofh7/

    Addendum 2019-06-26: Change image on clicking "id"

    If forking the above, make sure to comment out the image re-assignment line: ev.target.href = "...";.

    If you want to change the mapImage's image when clicking a .selected element, I suggest using event delegation on the parent #list div to handle their click events, instead of recreating one per new div. Then in that event handler, since you're using jQuery you can check its textual content via .text(), convert that to a Number, and iterate through the mapImages until a matching mapImage.dataItem.dataContext.id is found. Then assign its child's image.

    Event handler sample below:

    $('#list').on('click', '.selected', function(event) {
      var id = Number($(event.target).text());
      var totalImages = imageSeries.mapImages.length;
      var mapImage;
      for(var i = 0; i < totalImages; ++i) {
        mapImage = imageSeries.mapImages.getIndex(i);
        if (mapImage.dataItem.dataContext.id === id) {
          mapImage.children.getIndex(0).href = "";
            break;
        }
      }
    });
    

    Demo:

    https://jsfiddle.net/notacouch/bxgz74t1/

    Or instead of iterating, since in this case the id is the actual id of the mapImages, we can use MapImageSeries' getImageById() method:

    $('#list').on('click', '.selected', function(event) {
      var id = Number($(event.target).text()); // or via close icon's id 
      var mapImage = imageSeries.getImageById(id);
      if (mapImage) {
        mapImage.children.getIndex(0).href = "";
      }
    });
    

    Demo:

    https://jsfiddle.net/notacouch/5ravqhoL/