Legend

Sometimes chart elements are self-descriptive, like Pie chart slices with labels, or a Line chart with one line series. Other times, user might need some visual clues to make sense of the information. That's where legend comes in.

This article walks through all the kinds of legends available in amCharts 4, and ways to configure and use them.

Classic legend

A classic legend is a collection of items representing some chart data elements such as series or slices.

It usually contains a marker (on an XY chart marker usually tries to reflect the shape of related series), a title, and (optionally) a value.

A typical roll-of-the-mill legend

Creating

In most situations, to create a legend you only need to assign an instance of Legend to chart's legend property:

chart.legend = new am4charts.Legend();
chart.legend = new am4charts.Legend();
{
  // ...
  "legend": {
    // No need to specify "type". The chart will assume "Legend"
  }
}

NOTE Legend is a universal control, suitable for most chart types, so it is included in all chart modules. So, it's accessible as charts.Legendmap.Legend, etc.

Positioning

By default, the chart will try to place the legend at the bottom.

You can change legend's position using it's position property. Available options are: "left", "right", "top", "bottom" (default), and "absolute".

NOTE Legend placement is important, as with all positions, except "absolute", the size of the legend will affect the space left for the actual chart.

Absolute positioning (position = "absolute") allows placing legend anywhere, in relation to chart's container, event outside its bounds.

Absolutely positioned legend needs its x and y coordinates to be set. Those can exceed chart width/height, or be negative to place the legend outside chart's container.

Such (absolutely positioned) legends do not affect the size of the chart.

Legend markers

Default markers

By default, the legend, when drawing markers, will try to mirror the look of the relative elements. For example if the legend item is a LineSeries styled as a red thick line with round bullets, the legend will draw a chunk of a line with a bullet, exactly as in series itself.

If you'd rather like to use default square markers, simply set useDefaultMarker to true.

Customizing markers

As we saw in the previous chapter, marker's look depend on the series it is representing. If we don't want those to mirror series look, we can opt out by setting useDefaultMarker = true.

But what if we want to display something completely different - neither resembling series look, nor default square?

Luckily, amCharts 4 Legend allows us to do anything.

The Legend's item itself is a Container which means you can place absolutely anything in it.

The default look places a RoundedRectangle in it. If all we want is to tweak some settings of the rectangle, we can do so by modifying the already existing element in the marker's template:

chart.legend = new am4charts.Legend();
chart.legend.useDefaultMarker = true;
let marker = chart.legend.markers.template.children.getIndex(0);
marker.cornerRadius(12, 12, 12, 12);
marker.strokeWidth = 2;
marker.strokeOpacity = 1;
marker.stroke = am4core.color("#ccc");
chart.legend = new am4charts.Legend();
chart.legend.useDefaultMarker = true;
var marker = chart.legend.markers.template.children.getIndex(0);
marker.cornerRadius(12, 12, 12, 12);
marker.strokeWidth = 2;
marker.strokeOpacity = 1;
marker.stroke = am4core.color("#ccc");
{
  // ...
  "legend": {
    "useDefaultMarker": true,
    "markers": {
      "children": [{
        "cornerRadiusTopLeft": 12,
        "cornerRadiusTopRight": 12,
        "cornerRadiusBottomRight": 12,
        "cornerRadiusBottomLeft": 12,
        "strokeWidth": 2,
        "strokeOpacity": 1,
        "stroke": "#ccc"
      }]
    }
  }
}

And voilà, we turned default squares into circles with grey outline:

See the Pen amCharts V4: Legend (markers) by amCharts (@amcharts) on CodePen.24419

NOTE Please note that outline (stroke) is disabled by default, hence us needing to set strokeOpacity = 1.

Using marker templates

In the previous example we accessed elements within marker template children list.

We can set properties directly to template as well.

let markerTemplate = chart.legend.markers.template;
markerTemplate.width = 40;
markerTemplate.height = 40;
var markerTemplate = chart.legend.markers.template;
markerTemplate.width = 40;
markerTemplate.height = 40;
{
  // ...
  "legend": {
    "useDefaultMarker": true,
    "markers": {
      "width": 40,
      "height": 40
    }
  }
}

See the Pen amCharts V4: Legend (markers 2) by amCharts (@amcharts) on CodePen.24419

Creating new markers

Suppose, we want to go even further - somewhere rectangle's properties can't accommodate.

We're in luck, because, as we already mentioned, Legend marker is a container, so we can add anything there. And we're not limited to a single thing. For all we care, we can add the whole chart and his dog there.

Let's explore how this works in reality.

We're going to use a simple SVG image of a dollar sign as a marker. Something that looks like this: (except maybe smaller)

Below is the SVG and its Base64 representation (for use in an URL):

<?xml version="1.0"?>
<svg height="21px" version="1.1" viewBox="0 0 21 21" width="21px" xmlns="http://www.w3.org/2000/svg" xmlns:xlink="http://www.w3.org/1999/xlink">
  <title />
  <desc />
  <defs />
  <g fill="none" fill-rule="evenodd" id="Business-Icon" stroke="none" stroke-width="1" transform="translate(-277.000000, -227.000000)">
    <g id="Coin" transform="translate(275.000000, 225.000000)">
      <rect height="25" id="Rectangle-49" width="25" x="0" y="0" />
      <g id="Group-34" transform="translate(2.000000, 2.000000)">
        <path d="M10.5,21 C4.70101013,21 0,16.2989899 0,10.5 C0,4.70101013 4.70101013,0 10.5,0 C16.2989899,0 21,4.70101013 21,10.5 C21,16.2989899 16.2989899,21 10.5,21 Z M10.5,18 C14.6421356,18 18,14.6421356 18,10.5 C18,6.35786438 14.6421356,3 10.5,3 C6.35786438,3 3,6.35786438 3,10.5 C3,14.6421356 6.35786438,18 10.5,18 Z" fill="#3B5AFB" fill-rule="nonzero" id="Combined-Shape" />
        <path d="M10.5,17 C6.91014913,17 4,14.0898509 4,10.5 C4,6.91014913 6.91014913,4 10.5,4 C14.0898509,4 17,6.91014913 17,10.5 C17,14.0898509 14.0898509,17 10.5,17 Z M11,8 L11,7.5 C11,7.22385763 10.7761424,7 10.5,7 C10.2238576,7 10,7.22385763 10,7.5 L10,8 L9.5139452,8 C8.68551807,8 8.0139452,8.67157288 8.0139452,9.5 L8.0139452,10.0007567 C8.0139452,10.8291838 8.68551807,11.5007567 9.5139452,11.5007567 L11.5000049,11.5007567 C11.7761473,11.5007567 12.0000049,11.7246143 12.0000049,12.0007567 L12.0000049,12.5 C12.0000049,12.7761424 11.7761473,13 11.5000049,13 L9.01225525,13 C8.9916704,12.749839 8.78597816,12.549668 8.52914442,12.541857 C8.25312966,12.5334627 8.02257057,12.750412 8.01417627,13.0264268 L8.00023597,13.4848008 C7.99166064,13.7667682 8.21790708,14 8.5000049,14 L10,14 L10,14.4912415 C10,14.7673838 10.2238576,14.9912415 10.5,14.9912415 C10.7761424,14.9912415 11,14.7673838 11,14.4912415 L11,14 L11.5000049,14 C12.328432,14 13.0000049,13.3284271 13.0000049,12.5 L13.0000049,12.0007567 C13.0000049,11.1723296 12.328432,10.5007567 11.5000049,10.5007567 L9.5139452,10.5007567 C9.23780282,10.5007567 9.0139452,10.2768991 9.0139452,10.0007567 L9.0139452,9.5 C9.0139452,9.22385763 9.23780282,9 9.5139452,9 L11.9679559,9 C12.0222203,9.21744733 12.2188481,9.37854004 12.453101,9.37854004 C12.7292434,9.37854004 12.953101,9.15468241 12.953101,8.87854004 L12.953101,8.5 C12.953101,8.22385763 12.7292434,8 12.453101,8 L11,8 Z" fill="#44D5E9" id="Combined-Shape" />
      </g>
    </g>
  </g>
</svg>


NOTE Psst, iconfinder.com is a great source for free and commercial vector icons. We got the above from them.

Now that we have a custom marker, we can:

  1. Clear out current marker template by calling its disposeChildren() method;
  2. Create a new Image in marker template;
  3. Assign our Base64-encoded SVG into image's href property.
/* Remove square from marker template */
var marker = chart.legend.markers.template;
marker.disposeChildren();

/* Add custom image instead */
let dollar = marker.createChild(am4core.Image);
dollar.width = 40;
dollar.height = 40;
dollar.verticalCenter = "top";
dollar.horizontalCenter = "left";
dollar.href = "data:image/svg+xml;base64,......";
/* Remove square from marker template */
var marker = chart.legend.markers.template;
marker.disposeChildren();

/* Add custom image instead */
let dollar = marker.createChild(am4core.Image);
dollar.width = 40;
dollar.height = 40;
dollar.verticalCenter = "top";
dollar.horizontalCenter = "left";
dollar.href = "data:image/svg+xml;base64,......";
/**
 * Unfortunately this is not yet possible using JSON config
 */

Let's see how it turned out:

See the Pen amCharts V4: Legend (markers 2) by amCharts (@amcharts) on CodePen.24419

Disabling markers

Don't need markers at all? Simply set disabled = true on marker template:

chart.legend.markers.template.disabled = true;
chart.legend.markers.template.disabled = true;
{
  // ...
  "legend": {
    "useDefaultMarker": true,
    "markers": {
      "disabled": true
    }
  }
}

See the Pen amCharts V4: Legend (markers 3) by amCharts (@amcharts) on CodePen.24419

The above example uses label formatting tricks to make it more useful, now that markers are gone. We'll see how to make that happen in the next section.

Legend labels

Series' relation to Legend is defined by its legendSettings property.

legendSettings is an object of type LegendSettings (make sure you click on a link to check its available properties), that holds some settings that are needed by Legend in order to know what information to display as a label and value in its items representing particular series.

We know, we don't like the above sentence either, so let's put it some other way.

A Legend creates a Legend item for each series in the chart. We already saw how it "inherits" marker look from the series.

That leaves it with a textual label, and (optionally) value label.

That's when it turns to series' legendSettings. Basically, it says "OK series X, show me what do you want me to put into your Legend label."

Here's a basic chart with a few Line series and a default legend, we're going to be using in this section as a lab rat:

See the Pen amCharts V4: Legend (labels 1) by amCharts (@amcharts) on CodePen.24419

Name labels

For a series (name) label, Legend will look for labelText in legendSettings.

The default is "{name}", which is a code for "replace this text with a name attribute of the target series.

MORE INFO For more information about text formatting, make sure you check out our "Formatting Strings" article.

If I wanted to override default, I could assign to anything else, by modifying series' legendSettings.labelText.

series1.legendSettings.labelText = "Series: [bold {stroke}]{name}[/]";
series1.legendSettings.labelText = "Series: [bold {stroke}]{name}[/]";
{
  // ...
  "series": {
    "type": "LineSeries",
    // ...
    "legendSettings": {
      "labelText": "Series: [bold {stroke}]{name}[/]"
    }
  }
}

Notice, how we can use the {...} placeholders even within formatting [...] blocks. In our case we're applying series' color to part (!) of the legend label. How [insert your favorite word here] cool is that?

See the Pen amCharts V4: Legend (labels 2) by amCharts (@amcharts) on CodePen.24419

Value labels

Besides static (name) label, Legend can display an additional "value" label.

A value label is basically a "secondary" label which can be used to display information we want to be separate from the main label. It's mostly used to display item's value, hence us using term "value label".

The reason to have value labels is that we can align and arrange them into columns, which makes them more prominent and easy to read, then if they would be mashed together with the name labels.

Just take a look at these two images for comparison, to see what we mean by that:

  

It's much much clearer when we have values arranged in a separate column, isn't it?

What goes in there is determined by legendSettings.valueText, which is empty by default on an XY Chart, but is set to "{value}" on a Pie chart (yes, Pie chart is also a serial chart in amCharts 4) so that each slice's value is displayed next to its name in Legend.

As with the name label, it can have static text, formatting blocks, and placeholders, or any combination of.

Say, we want to display a series' close (last) value next to its name. We can use {valueY.close} placeholder for it.

series1.legendSettings.valueText = "{valueY.close}";
series1.legendSettings.valueText = "{valueY.close}";
{
  // ...
  "series": {
    "type": "LineSeries",
    // ...
    "legendSettings": {
      // ...
      "valueText": "{valueY.close}"
    }
  }
}

See the Pen amCharts V4: Legend (labels 3) by amCharts (@amcharts) on CodePen.24419

TIP Could we have included "{valueY.close}" into labelText? Yes, absolutely. We'd be be sacrificing our ability to arrange values in a separate column, though.

Interacting with cursor

Now that we know our way around Legend's static labels, let's talk about dynamic ones.

Legend can be set up to change its labels dynamically based on user's interaction with the chart.

For that we will need to enabled a chart Cursor.

SIDE READING Enabling a cursor is as easy as creating a cursor of appropriate type (XYCursor for XY chart, or RadarCursor for Radar chart) and assigning it to chart's cursor property. We're not going to get into it here, though. Make sure you check out our "Chart Cursor" article for more juice on this super-useful tool.

We already have a Cursor in our previous examples. All we need to do is to configure our legendSettings to apply different information on Legend items once the cursor goes through the chart.

Similarly to how we used labelText and valueText, we're going to use itemLabelText and itemValueText.

The latter two function exactly like the former ones, except they are applied only when chart Cursor is hovering over plot area. If set, these will override static labels.

Let's try overriding our value label (which currently shows close value for each series) with a value under Cursor:

series1.legendSettings.itemValueText = "[bold]{valueY}[/bold]";
series1.legendSettings.itemValueText = "[bold]{valueY}[/bold]";
{
  // ...
  "series": {
    "type": "LineSeries",
    // ...
    "legendSettings": {
      // ...
      "itemValueText": "[bold]{valueY}[/bold]"
    }
  }
}

Now, when we glide over our chart with a Cursor, the values next to Legend items will change to reflect actual values under Cursor.

It's not just value that can change, we can apply the whole different label, including formatting. (notice how we made "rollover" values bold in the above example)

See the Pen amCharts V4: Legend (labels 4) by amCharts (@amcharts) on CodePen.24419

Aligning labels

@TODO Content for this section is being prepared.

Controlling spacing

Normally, there's some spacing applied around each legend item. To be exact: 10 pixels on top and bottom.

If you'd like to increase or reduce that spacing it's as simple as setting padding for Legend's item container template:

chart.legend.itemContainers.template.paddingTop = 5;
chart.legend.itemContainers.template.paddingBottom = 5;
chart.legend.itemContainers.template.paddingTop = 5;
chart.legend.itemContainers.template.paddingBottom = 5;
{
  // ...
  "legend": {
    "itemContainers": {
      "paddingTop": 5,
      "paddingBottom": 5
    }
  }
}

Handling events

A Legend by itself does not have any specific events. However, you can still attach generic events to its items.

We're going to be using Legend's itemContainers list template for that. It's a list template, so we can put any events on its template object and they will be added to any actual Legend items created.

For example, if I'd like to add "hit" event on a legend marker, I'd add that event on Legend's itemContainers.template:

chart.legend.itemContainers.template.events.on("hit", function(ev) {
  console.log("Clicked on", ev.target);
});
chart.legend.itemContainers.template.events.on("hit", function(ev) {
 console.log("Clicked on", ev.target);
});
{
  // ...
  "legend": {
    "itemContainers": {
      "events": {
        "hit": function(ev) {
          console.log("Clicked on", ev.target);
        })
      }
    }
  }
}

MORE INFO Refer to our "Event Listeners" article for more juice on event handling.

An event file, received by the handler function will contain reference to the actual Container for clicked item.

By itself it's not very useful to us. However, using its properties, we can easily deduce the Series it belongs to.

The target Container for the event, contains dataItem, which in turn contains dataContext property, which will hold reference to related Series.

So, reworking our example from before, this looks much more useful:

chart.legend.itemContainers.template.events.on("hit", function(ev) {
  alert("Clicked on " + ev.target.dataItem.dataContext.name);
});
chart.legend.itemContainers.template.events.on("hit", function(ev) {
 alert("Clicked on " + ev.target.dataItem.dataContext.name);
});
{
  // ...
  "legend": {
    "itemContainers": {
      "events": {
        "hit": function(ev) {
          alert("Clicked on " + ev.target.dataItem.dataContext.name);
        })
      }
    }
  }
}

Try clicking on legend items here:

See the Pen amCharts V4: Legend (events) by amCharts (@amcharts) on CodePen.24419

Adding custom items

You can create a standalone instance of a Legend, that is not tied into any of the chart's data or objects, with completely custom items.

For that, we will need to instantiate a new Legend object, add our item info directly to it's data property, as well as push it into one of the chart's container's, such as chartContainer.

This is best explained with an example:

let legend = new am4charts.Legend();
legend.parent = chart.chartContainer;
legend.background.fill = am4core.color("#000");
legend.background.fillOpacity = 0.05;
legend.width = 120;
legend.align = "right";
legend.data = [{
  "name": "2016",
  "fill":"#72A6B2"
}, {
  "name": "2017",
  "fill": "#667E93"
}, {
  "name": "2018",
  "fill": "#488BB2"
}];
var legend = new am4charts.Legend();
legend.parent = chart.chartContainer;
legend.background.fill = am4core.color("#000");
legend.background.fillOpacity = 0.05;
legend.width = 120;
legend.align = "right";
legend.data = [{
  "name": "2016",
  "fill":"#72A6B2"
}, {
  "name": "2017",
  "fill": "#667E93"
}, {
  "name": "2018",
  "fill": "#488BB2"
}];
{
  // ...
  "chartContainer": {
    "children": [{
      "type": "Legend",
      "width": 120,
      "align": "right",
      "background": {
        "fill": "#000",
        "fillOpacity": 0.05
      },
      "data": [{
        "name": "2016",
        "fill":"#72A6B2"
      }, {
        "name": "2017",
        "fill": "#667E93"
      }, {
        "name": "2018",
        "fill": "#488BB2"
      }]
    }]
  }
}

IMPORTANT Natural temptation would be to assign our custom Legend to chart's legend property. If we did that, chart would take over legend's data. So, we're adding it to one of the chart's containers.

MORE INFO For more information on chart built-in containers, visit "Pre-defined chart containers".

TIP If you'd like some of the items to be disabled by default, add visible: false to their data.

Let's see how it works on a live example: (on a map actually)

See the Pen amCharts V4: Map (legend) by amCharts (@amcharts) on CodePen.24419

MORE INFO If you'd like more info on how adding and arranging children of containers work, make sure you read our "Working with Containers" article.

Heat legend

A Heat legend is used to depict a range of values and their corresponding color (heat).

It's so cool, we have a separate article for it: "Heat Legend".

Advanced usage

Below are some links to tutorials, related to advanced Legend usage:

Related demos