• Edit
  • Download
  • <!doctype html>
    <html>
    
    <head>
      <meta charset="utf-8">
      <title>ZingSoft Demo</title>
      <script nonce="undefined" src="https://cdn.zingchart.com/zingchart.min.js"></script>
      <style>
        .zc-parties {
          padding: 1rem;
          display: flex;
          justify-content: center;
          background: #fff;
          box-sizing: border-box;
        }
    
        .zc-body {
          background: #fff;
        }
    
        .chart--container {
          height: 100%;
          width: 100%;
          min-height: 530px;
        }
    
        .zc-ref {
          display: none;
        }
      </style>
    </head>
    
    <body class="zc-body">
    
      <div class="zc-parties">
    
        <div id="myChart">
          <a href="https://www.zingchart.com/" rel="noopener" class="zc-ref">Powered by ZingChart</a>
        </div>
    
      </div>
    
      <script>
        ZC.LICENSE = ["569d52cefae586f634c54f86dc99e6a9", "b55b025e438fa8a98e32482b5f768ff5"];
        // INIT
        // -----------------------------
        // Define Module Location
        zingchart.MODULESDIR = "https://cdn.zingchart.com/modules/";
        // Load Maps
        zingchart.loadModules('maps, maps-fraL2, color-scale', init);
    
        function init() {
    
          // DEFINE CHART LOCATIONS (IDS)
          // -----------------------------
          // Main chart render location(s)
          let chartId = 'myChart';
    
    
          // PARTY CHARTS DATA & CONFIG
          // -----------------------------
    
          // 3 main parties
          let parties = {
            REM: {
              name: 'REM: La Republique En Marche!',
              color: '#FB8B24',
              leader: 'Christophe Castaner',
              icon: ''
            },
            LR: {
              name: 'LR: The Republicans',
              color: '#00A8E8',
              leader: 'Laurent Wauquiez',
              icon: ''
            },
            PS: {
              name: 'PS: Socialist Party',
              color: '#FD2C88',
              leader: 'Olivier Faure',
              icon: ''
            }
          };
    
          let items = zingchart.maps.getItems('fraL2'),
            i;
    
          // For each department, create fake data with percentages for each party
          let scores = {
            REM: {},
            LR: {},
            PS: {}
          }
          // Fill w/ data
          for (let i = 0; i < items.length; i++) {
            let base = 35000;
            let above90 = rand(0, 100) > 90;
            let lenCheck = i % (Math.round(items.length / 4)) === 0;
            if (above90 || lenCheck) base = 25000;
            let pp1 = parseFloat((rand(base, 45000) / 1000).toFixed(3));
            let pp2 = parseFloat((rand(base, 45000) / 1000).toFixed(3));
            let pp3 = parseFloat((100 - pp1 - pp2).toFixed(3));
            scores['REM'][items[i]] = pp1;
            scores['LR'][items[i]] = pp2;
            scores['PS'][items[i]] = pp3;
          }
    
          // Main chart config
          let gmain = {
            type: 'null',
            backgroundColor: 'none',
            borderColor: '#ddd',
            borderWidth: 1,
            width: '66.66%',
            height: '55%',
            x: '0%',
            y: '0%',
            title: {
              fontSize: '13px',
              text: 'Projected election results'
            },
            shapes: [{
              type: 'zingchart.maps',
              options: {
                id: 'mapmain',
                name: 'fraL2',
                x: '2.5%',
                y: '5%',
                width: '95%',
                height: '95%',
                scale: true,
                zooming: false,
                panning: false,
                scrolling: false,
                choropleth: {
                  aspect: 'intervals',
                  steps: [0, 1, 2, 3],
                  colors: [parties['REM'].color, parties['LR'].color, parties['PS'].color]
                },
                style: {
                  borderAlpha: 0.1,
                  borderColor: '#fff',
                  controls: {
                    visible: false
                  },
                  label: {
                    visible: false
                  },
                  hoverState: {
                    backgroundColor: 'none',
                    shadowAlpha: 0.2,
                    shadowDistance: 2,
                    shadow: true,
                    shadowColor: '#333'
                  },
                  tooltip: {
                    text: '<span style="font-size:29px;font-weight:bold;">%text</span>%data-preview<br><br><br><br>',
                    align: 'left',
                    backgroundColor: '#f9f9f9',
                    borderRadius: '3px',
                    borderWidth: 0,
                    callout: true,
                    calloutWidth: 16,
                    calloutHeight: 8,
                    color: '#000',
                    fontSize: '13px',
                    lineHeight: 15,
                    padding: '10px 20px',
                    shadow: true,
                    shadowAlpha: 0.7,
                    shadowColor: '#333',
                    shadowDistance: 2,
                  },
                  items: {}
                }
              }
            }]
          };
    
          for (let i = 0; i < items.length; i++) {
            let fVal = 0,
              sPreview = '';
    
            let aPreview = [];
            aPreview.push(['REM', scores['REM'][items[i]]]);
            aPreview.push(['LR', scores['LR'][items[i]]]);
            aPreview.push(['PS', scores['PS'][items[i]]]);
            aPreview.sort(function(a, b) {
              return b[1] - a[1];
            });
            for (let p = 0; p < aPreview.length; p++) {
              let bars = createBar(aPreview[p][1], parties[aPreview[p][0]].color);
              sPreview += '<br>' + (p + 1) + ' ' + bars + ' ' + aPreview[p][0] + ': ' + aPreview[p][1] + '%';
            }
    
            if (scores['REM'][items[i]] > scores['LR'][items[i]] && scores['REM'][items[i]] > scores['PS'][items[i]]) {
              fVal = 0.5;
            }
            if (scores['LR'][items[i]] > scores['REM'][items[i]] && scores['LR'][items[i]] > scores['PS'][items[i]]) {
              fVal = 1.5;
            }
            if (scores['PS'][items[i]] > scores['REM'][items[i]] && scores['PS'][items[i]] > scores['LR'][items[i]]) {
              fVal = 2.5;
            }
            gmain.shapes[0].options.style.items[items[i]] = {
              dataValue: fVal,
              dataPreview: sPreview
            };
          }
    
    
          // PARTY 1
          // -----------------------------
          let gp1 = {
            type: 'null',
            backgroundColor: 'none',
            borderColor: '#ddd',
            borderWidth: 1,
            width: '33.33%',
            height: '33.33%',
            x: '66.66%',
            y: '0%',
            title: {
              fontSize: '11px',
              text: parties['REM'].name
            },
            colorScale: {
              cursor: {
                size: 3
              },
              item: {
                fontSize: 10,
                offsetY: -5
              },
              layout: 'h',
              map: true,
              margin: 'auto auto 20 auto',
              width: '100px',
              height: '10px'
            },
            shapes: [{
              type: 'zingchart.maps',
              options: {
                id: 'mapp1',
                name: 'fraL2',
                colorScale: true,
                panning: false,
                scale: true,
                scrolling: false,
                x: '5%',
                y: '10%',
                width: '90%',
                height: '80%',
                zooming: false,
                choropleth: {
                  aspect: 'gradient',
                  progression: 'lin',
                  color: parties['REM'].color,
                  maxPercent: 50,
                  effect: 'lighten',
                  mirrored: true
                },
                style: {
                  borderAlpha: 0.1,
                  borderColor: parties['REM'].color,
                  controls: {
                    visible: false
                  },
                  label: {
                    visible: false
                  },
                  hoverState: {
                    backgroundColor: 'none',
                    shadowAlpha: 0.1,
                    shadowDistance: 0,
                    shadow: true,
                    shadowColor: '#369'
                  },
                  tooltip: {
                    text: '%text: %data-value%',
                    backgroundColor: parties['REM'].color,
                    borderRadius: '3px',
                    borderWidth: 0,
                    callout: true,
                    calloutWidth: '16px',
                    calloutHeight: '8px',
                    color: '#fff',
                    fontSize: 13,
                    fontWeight: 'bold',
                    padding: '10px 20px',
                    shadow: true,
                    shadowColor: '#333',
                    shadowAlpha: 0.7,
                    shadowDistance: 2
                  },
                  items: {}
                }
              }
            }]
          };
    
          for (let i = 0; i < items.length; i++) {
            gp1.shapes[0].options.style.items[items[i]] = {
              dataValue: scores['REM'][items[i]]
            };
          }
    
    
          // PARTY 2
          // -----------------------------
          let gp2 = {
            type: 'null',
            backgroundColor: 'none',
            borderColor: '#ddd',
            borderWidth: 1,
            x: '66.66%',
            y: '33.33%',
            width: '33.33%',
            height: '33.33%',
            title: {
              fontSize: '11px',
              text: parties['LR'].name
            },
            colorScale: {
              layout: 'h',
              width: '100px',
              height: '10px',
              map: true,
              margin: 'auto auto 20px auto',
              cursor: {
                size: 3
              },
              item: {
                fontSize: '10px',
                offsetY: '-5px'
              }
            },
            shapes: [{
              type: 'zingchart.maps',
              options: {
                id: 'mapp2',
                name: 'fraL2',
                colorScale: true,
                panning: false,
                scale: true,
                scrolling: false,
                width: '90%',
                height: '80%',
                x: '5%',
                y: '10%',
                zooming: false,
                choropleth: {
                  aspect: 'gradient',
                  color: parties['LR'].color,
                  effect: 'lighten',
                  maxPercent: 50,
                  mirrored: true,
                  progression: 'lin'
                },
                style: {
                  borderAlpha: 0.1,
                  borderColor: parties['LR'].color,
                  controls: {
                    visible: false
                  },
                  label: {
                    visible: false
                  },
                  hoverState: {
                    backgroundColor: 'none',
                    shadow: true,
                    shadowAlpha: 0.1,
                    shadowColor: '#369',
                    shadowDistance: 0
                  },
                  tooltip: {
                    text: '%text: %data-value%',
                    backgroundColor: parties['LR'].color,
                    borderRadius: '3px',
                    borderWidth: 0,
                    callout: true,
                    calloutWidth: '16px',
                    calloutHeight: '8px',
                    color: '#fff',
                    fontSize: '13px',
                    fontWeight: 'bold',
                    padding: '10px 20px',
                    shadow: true,
                    shadowAlpha: 0.7,
                    shadowColor: '#333',
                    shadowDistance: 2,
                  },
                  items: {}
                }
              }
            }]
          };
    
          for (let i = 0; i < items.length; i++) {
            gp2.shapes[0].options.style.items[items[i]] = {
              dataValue: scores['LR'][items[i]]
            };
          }
    
    
          // PARTY 3
          // -----------------------------
          let gp3 = {
            type: 'null',
            backgroundColor: 'none',
            borderColor: '#ddd',
            borderWidth: 1,
            width: '33.33%',
            height: '33.33%',
            x: '66.66%',
            y: '66.66%',
            title: {
              fontSize: '11px',
              text: parties['PS'].name
            },
            colorScale: {
              layout: 'h',
              map: true,
              margin: 'auto auto 20px auto',
              width: '100px',
              height: '10px',
              cursor: {
                size: 3
              },
              item: {
                fontSize: '10px',
                offsetY: '-5px'
              }
            },
            shapes: [{
              type: 'zingchart.maps',
              options: {
                id: 'mapp3',
                name: 'fraL2',
                colorScale: true,
                panning: false,
                scale: true,
                scrolling: false,
                width: '90%',
                height: '80%',
                x: '5%',
                y: '10%',
                zooming: false,
                choropleth: {
                  aspect: 'gradient',
                  color: parties['PS'].color,
                  effect: 'lighten',
                  maxPercent: 50,
                  mirrored: true,
                  progression: 'lin'
                },
                style: {
                  borderAlpha: 0.1,
                  borderColor: parties['PS'].color,
                  controls: {
                    visible: false
                  },
                  label: {
                    visible: false
                  },
                  hoverState: {
                    backgroundColor: 'none',
                    shadow: true,
                    shadowAlpha: 0.1,
                    shadowColor: '#369',
                    shadowDistance: 0
                  },
                  tooltip: {
                    text: '%text: %data-value%',
                    backgroundColor: parties['PS'].color,
                    borderRadius: '3px',
                    borderWidth: 0,
                    callout: true,
                    calloutWidth: '16px',
                    calloutHeight: '8px',
                    color: '#fff',
                    fontSize: '13px',
                    fontWeight: 'bold',
                    padding: '10px 20px',
                    shadow: true,
                    shadowAlpha: 0.7,
                    shadowColor: '#333',
                    shadowDistance: 2
                  },
                  items: {}
                }
              }
            }]
          };
    
          for (let i = 0; i < items.length; i++) {
            gp3.shapes[0].options.style.items[items[i]] = {
              dataValue: scores['PS'][items[i]]
            };
          }
    
    
          // TIME CHART
          // -----------------------------
          let gtime = {
            type: 'line',
            backgroundColor: 'none',
            borderColor: '#ddd',
            borderWidth: '1px',
            width: '66.66%',
            height: '45%',
            x: '0%',
            y: '55%',
            title: {
              text: 'Public trust for leaders of main political parties (trend)',
              fontSize: '13px'
            },
            plot: {
              maxNodes: 0,
              maxTrackers: 0
            },
            plotarea: {
              margin: '140px 30px 30px 40px'
            },
            scaleX: {
              maxItems: 6,
              step: 'month',
              item: {
                fontSize: 10
              },
              transform: {
                type: 'date',
                text: '%M %Y'
              }
            },
            scaleY: {
              maxItems: 6,
              format: '%v%',
              step: 20,
              item: {
                fontSize: '10px'
              }
            },
            crosshairX: {
              plotLabel: {
                text: '<span style="color:%color;font-weight:bold;">%node-value%</span> %plot-text',
                backgroundColor: '#f9f9f9',
                borderRadius: '5px',
                borderWidth: 0,
                callout: false,
                distance: 10,
                fontSize: '12px',
                multiple: false,
                padding: '8px',
                shadow: true,
                shadowAlpha: 0.5,
                shadowColor: '#333',
                shadowDistance: 1
              },
              scaleLabel: {
                fontSize: '10px'
              },
              mask: {
                backgroundColor: '#eee'
              }
            },
            shapes: [{
                type: 'circle',
                backgroundImage: parties['REM'].icon,
                borderWidth: '6px',
                borderColor: parties['REM'].color,
                size: '32px',
                x: '20%',
                y: '64px',
                label: {
                  text: parties['REM'].leader + '<br>(REM)',
                  fontSize: '10px',
                  offsetY: '48px'
                }
              },
              {
                type: 'circle',
                backgroundImage: parties['LR'].icon,
                borderWidth: '6px',
                borderColor: parties['LR'].color,
                size: 32,
                x: '50%',
                y: '64px',
                label: {
                  text: parties['LR'].leader + '<br>(LR)',
                  fontSize: '10px',
                  offsetY: '48px'
                }
              },
              {
                type: 'circle',
                size: 32,
                backgroundImage: parties['PS'].icon,
                borderColor: parties['PS'].color,
                borderWidth: '6px',
                x: '80%',
                y: '64px',
                label: {
                  text: parties['PS'].leader + '<br>(PS)',
                  fontSize: '10px',
                  offsetY: '48px'
                }
              }
            ],
            series: []
          };
          let sdata = [
            [],
            [],
            []
          ];
    
          let ts = Date.UTC(2018, 10, 15);
          let pc1 = 20 + 40 * Math.random();
          let pc2 = 20 + 40 * Math.random();
          let pc3 = 10 + 20 * Math.random();
    
          for (let d = 0; d < 48; d++) {
            pc1 += -6 + 12 * Math.random();
            if (pc1 > 80 || pc1 < 20) {
              pc1 += ((pc1 > 80) ? -1 : 1) * 6 * Math.random();
            }
            pc1 = parseFloat(pc1.toFixed(1));
    
            pc2 += -5 + 10 * Math.random();
            if (pc1 > 80 || pc1 < 20) {
              pc2 += ((pc2 > 80) ? -1 : 1) * 5 * Math.random();
            }
            pc2 = parseFloat(pc2.toFixed(1));
    
            pc3 += -3 + 6 * Math.random();
            if (pc1 > 80 || pc1 < 20) {
              pc3 += ((pc3 > 80) ? -1 : 1) * 3 * Math.random();
            }
            pc3 = parseFloat(pc3.toFixed(1));
    
            sdata[0].push([ts, pc1]);
            sdata[1].push([ts, pc2]);
            sdata[2].push([ts, pc3]);
    
            ts -= 1000 * 60 * 60 * 24 * 30.5;
          }
    
          gtime.series.push({
            lineColor: parties['REM'].color,
            text: parties['REM'].leader,
            values: sdata[0].reverse()
          });
          gtime.series.push({
            lineColor: parties['LR'].color,
            text: parties['LR'].leader,
            values: sdata[1].reverse()
          });
          gtime.series.push({
            lineColor: parties['PS'].color,
            text: parties['PS'].leader,
            values: sdata[2].reverse()
          });
    
    
          // CHART CONFIG
          // -----------------------------
          let cdata = {
            backgroundColor: '#fff',
            borderColor: '#ddd',
            borderWidth: '2px',
            graphset: [
              gmain,
              gp1,
              gp2,
              gp3,
              gtime
            ]
          };
    
    
          // RENDER CHARTS
          // -----------------------------
          zingchart.render({
            id: chartId,
            width: 600,
            height: 600,
            output: 'svg',
            data: cdata,
            modules: 'color-scale'
          });
        }
    
    
        // HELPER FNS
        // -----------------------------
    
        function createBar(score, color) {
          let sBar = "<span style='font-weight:bold;color:" + color + "'>";
          for (let b = 0; b < Math.round(score / 4); b++) {
            sBar += '\u2588';
          }
          sBar += '</span>';
          return sBar;
        }
    
        function rand(min, max) {
          return Math.round(min + (max - min) * Math.random());
        };
      </script>
    </body>
    
    </html>
    <!doctype html>
    <html>
    
    <head>
      <meta charset="utf-8">
      <title>ZingSoft Demo</title>
      <script src="https://cdn.zingchart.com/zingchart.min.js"></script>
    </head>
    
    <body class="zc-body">
    
      <div class="zc-parties">
    
        <div id="myChart">
          <a href="https://www.zingchart.com/" rel="noopener" class="zc-ref">Powered by ZingChart</a>
        </div>
    
      </div>
    
    </body>
    
    </html>
    .zc-parties {
      padding: 1rem;
      display: flex;
      justify-content: center;
      background: #fff;
      box-sizing: border-box;
    }
    
    .zc-body {
      background: #fff;
    }
    
    .chart--container {
      height: 100%;
      width: 100%;
      min-height: 530px;
    }
    
    .zc-ref {
      display: none;
    }
    // INIT
    // -----------------------------
    // Define Module Location
    zingchart.MODULESDIR = "https://cdn.zingchart.com/modules/";
    // Load Maps
    zingchart.loadModules('maps, maps-fraL2, color-scale', init);
    
    function init() {
    
      // DEFINE CHART LOCATIONS (IDS)
      // -----------------------------
      // Main chart render location(s)
      let chartId = 'myChart';
    
    
      // PARTY CHARTS DATA & CONFIG
      // -----------------------------
    
      // 3 main parties
      let parties = {
        REM: {
          name: 'REM: La Republique En Marche!',
          color: '#FB8B24',
          leader: 'Christophe Castaner',
          icon: ''
        },
        LR: {
          name: 'LR: The Republicans',
          color: '#00A8E8',
          leader: 'Laurent Wauquiez',
          icon: ''
        },
        PS: {
          name: 'PS: Socialist Party',
          color: '#FD2C88',
          leader: 'Olivier Faure',
          icon: ''
        }
      };
    
      let items = zingchart.maps.getItems('fraL2'),
        i;
    
      // For each department, create fake data with percentages for each party
      let scores = {
        REM: {},
        LR: {},
        PS: {}
      }
      // Fill w/ data
      for (let i = 0; i < items.length; i++) {
        let base = 35000;
        let above90 = rand(0, 100) > 90;
        let lenCheck = i % (Math.round(items.length / 4)) === 0;
        if (above90 || lenCheck) base = 25000;
        let pp1 = parseFloat((rand(base, 45000) / 1000).toFixed(3));
        let pp2 = parseFloat((rand(base, 45000) / 1000).toFixed(3));
        let pp3 = parseFloat((100 - pp1 - pp2).toFixed(3));
        scores['REM'][items[i]] = pp1;
        scores['LR'][items[i]] = pp2;
        scores['PS'][items[i]] = pp3;
      }
    
      // Main chart config
      let gmain = {
        type: 'null',
        backgroundColor: 'none',
        borderColor: '#ddd',
        borderWidth: 1,
        width: '66.66%',
        height: '55%',
        x: '0%',
        y: '0%',
        title: {
          fontSize: '13px',
          text: 'Projected election results'
        },
        shapes: [{
          type: 'zingchart.maps',
          options: {
            id: 'mapmain',
            name: 'fraL2',
            x: '2.5%',
            y: '5%',
            width: '95%',
            height: '95%',
            scale: true,
            zooming: false,
            panning: false,
            scrolling: false,
            choropleth: {
              aspect: 'intervals',
              steps: [0, 1, 2, 3],
              colors: [parties['REM'].color, parties['LR'].color, parties['PS'].color]
            },
            style: {
              borderAlpha: 0.1,
              borderColor: '#fff',
              controls: {
                visible: false
              },
              label: {
                visible: false
              },
              hoverState: {
                backgroundColor: 'none',
                shadowAlpha: 0.2,
                shadowDistance: 2,
                shadow: true,
                shadowColor: '#333'
              },
              tooltip: {
                text: '<span style="font-size:29px;font-weight:bold;">%text</span>%data-preview<br><br><br><br>',
                align: 'left',
                backgroundColor: '#f9f9f9',
                borderRadius: '3px',
                borderWidth: 0,
                callout: true,
                calloutWidth: 16,
                calloutHeight: 8,
                color: '#000',
                fontSize: '13px',
                lineHeight: 15,
                padding: '10px 20px',
                shadow: true,
                shadowAlpha: 0.7,
                shadowColor: '#333',
                shadowDistance: 2,
              },
              items: {}
            }
          }
        }]
      };
    
      for (let i = 0; i < items.length; i++) {
        let fVal = 0,
          sPreview = '';
    
        let aPreview = [];
        aPreview.push(['REM', scores['REM'][items[i]]]);
        aPreview.push(['LR', scores['LR'][items[i]]]);
        aPreview.push(['PS', scores['PS'][items[i]]]);
        aPreview.sort(function(a, b) {
          return b[1] - a[1];
        });
        for (let p = 0; p < aPreview.length; p++) {
          let bars = createBar(aPreview[p][1], parties[aPreview[p][0]].color);
          sPreview += '<br>' + (p + 1) + ' ' + bars + ' ' + aPreview[p][0] + ': ' + aPreview[p][1] + '%';
        }
    
        if (scores['REM'][items[i]] > scores['LR'][items[i]] && scores['REM'][items[i]] > scores['PS'][items[i]]) {
          fVal = 0.5;
        }
        if (scores['LR'][items[i]] > scores['REM'][items[i]] && scores['LR'][items[i]] > scores['PS'][items[i]]) {
          fVal = 1.5;
        }
        if (scores['PS'][items[i]] > scores['REM'][items[i]] && scores['PS'][items[i]] > scores['LR'][items[i]]) {
          fVal = 2.5;
        }
        gmain.shapes[0].options.style.items[items[i]] = {
          dataValue: fVal,
          dataPreview: sPreview
        };
      }
    
    
      // PARTY 1
      // -----------------------------
      let gp1 = {
        type: 'null',
        backgroundColor: 'none',
        borderColor: '#ddd',
        borderWidth: 1,
        width: '33.33%',
        height: '33.33%',
        x: '66.66%',
        y: '0%',
        title: {
          fontSize: '11px',
          text: parties['REM'].name
        },
        colorScale: {
          cursor: {
            size: 3
          },
          item: {
            fontSize: 10,
            offsetY: -5
          },
          layout: 'h',
          map: true,
          margin: 'auto auto 20 auto',
          width: '100px',
          height: '10px'
        },
        shapes: [{
          type: 'zingchart.maps',
          options: {
            id: 'mapp1',
            name: 'fraL2',
            colorScale: true,
            panning: false,
            scale: true,
            scrolling: false,
            x: '5%',
            y: '10%',
            width: '90%',
            height: '80%',
            zooming: false,
            choropleth: {
              aspect: 'gradient',
              progression: 'lin',
              color: parties['REM'].color,
              maxPercent: 50,
              effect: 'lighten',
              mirrored: true
            },
            style: {
              borderAlpha: 0.1,
              borderColor: parties['REM'].color,
              controls: {
                visible: false
              },
              label: {
                visible: false
              },
              hoverState: {
                backgroundColor: 'none',
                shadowAlpha: 0.1,
                shadowDistance: 0,
                shadow: true,
                shadowColor: '#369'
              },
              tooltip: {
                text: '%text: %data-value%',
                backgroundColor: parties['REM'].color,
                borderRadius: '3px',
                borderWidth: 0,
                callout: true,
                calloutWidth: '16px',
                calloutHeight: '8px',
                color: '#fff',
                fontSize: 13,
                fontWeight: 'bold',
                padding: '10px 20px',
                shadow: true,
                shadowColor: '#333',
                shadowAlpha: 0.7,
                shadowDistance: 2
              },
              items: {}
            }
          }
        }]
      };
    
      for (let i = 0; i < items.length; i++) {
        gp1.shapes[0].options.style.items[items[i]] = {
          dataValue: scores['REM'][items[i]]
        };
      }
    
    
      // PARTY 2
      // -----------------------------
      let gp2 = {
        type: 'null',
        backgroundColor: 'none',
        borderColor: '#ddd',
        borderWidth: 1,
        x: '66.66%',
        y: '33.33%',
        width: '33.33%',
        height: '33.33%',
        title: {
          fontSize: '11px',
          text: parties['LR'].name
        },
        colorScale: {
          layout: 'h',
          width: '100px',
          height: '10px',
          map: true,
          margin: 'auto auto 20px auto',
          cursor: {
            size: 3
          },
          item: {
            fontSize: '10px',
            offsetY: '-5px'
          }
        },
        shapes: [{
          type: 'zingchart.maps',
          options: {
            id: 'mapp2',
            name: 'fraL2',
            colorScale: true,
            panning: false,
            scale: true,
            scrolling: false,
            width: '90%',
            height: '80%',
            x: '5%',
            y: '10%',
            zooming: false,
            choropleth: {
              aspect: 'gradient',
              color: parties['LR'].color,
              effect: 'lighten',
              maxPercent: 50,
              mirrored: true,
              progression: 'lin'
            },
            style: {
              borderAlpha: 0.1,
              borderColor: parties['LR'].color,
              controls: {
                visible: false
              },
              label: {
                visible: false
              },
              hoverState: {
                backgroundColor: 'none',
                shadow: true,
                shadowAlpha: 0.1,
                shadowColor: '#369',
                shadowDistance: 0
              },
              tooltip: {
                text: '%text: %data-value%',
                backgroundColor: parties['LR'].color,
                borderRadius: '3px',
                borderWidth: 0,
                callout: true,
                calloutWidth: '16px',
                calloutHeight: '8px',
                color: '#fff',
                fontSize: '13px',
                fontWeight: 'bold',
                padding: '10px 20px',
                shadow: true,
                shadowAlpha: 0.7,
                shadowColor: '#333',
                shadowDistance: 2,
              },
              items: {}
            }
          }
        }]
      };
    
      for (let i = 0; i < items.length; i++) {
        gp2.shapes[0].options.style.items[items[i]] = {
          dataValue: scores['LR'][items[i]]
        };
      }
    
    
      // PARTY 3
      // -----------------------------
      let gp3 = {
        type: 'null',
        backgroundColor: 'none',
        borderColor: '#ddd',
        borderWidth: 1,
        width: '33.33%',
        height: '33.33%',
        x: '66.66%',
        y: '66.66%',
        title: {
          fontSize: '11px',
          text: parties['PS'].name
        },
        colorScale: {
          layout: 'h',
          map: true,
          margin: 'auto auto 20px auto',
          width: '100px',
          height: '10px',
          cursor: {
            size: 3
          },
          item: {
            fontSize: '10px',
            offsetY: '-5px'
          }
        },
        shapes: [{
          type: 'zingchart.maps',
          options: {
            id: 'mapp3',
            name: 'fraL2',
            colorScale: true,
            panning: false,
            scale: true,
            scrolling: false,
            width: '90%',
            height: '80%',
            x: '5%',
            y: '10%',
            zooming: false,
            choropleth: {
              aspect: 'gradient',
              color: parties['PS'].color,
              effect: 'lighten',
              maxPercent: 50,
              mirrored: true,
              progression: 'lin'
            },
            style: {
              borderAlpha: 0.1,
              borderColor: parties['PS'].color,
              controls: {
                visible: false
              },
              label: {
                visible: false
              },
              hoverState: {
                backgroundColor: 'none',
                shadow: true,
                shadowAlpha: 0.1,
                shadowColor: '#369',
                shadowDistance: 0
              },
              tooltip: {
                text: '%text: %data-value%',
                backgroundColor: parties['PS'].color,
                borderRadius: '3px',
                borderWidth: 0,
                callout: true,
                calloutWidth: '16px',
                calloutHeight: '8px',
                color: '#fff',
                fontSize: '13px',
                fontWeight: 'bold',
                padding: '10px 20px',
                shadow: true,
                shadowAlpha: 0.7,
                shadowColor: '#333',
                shadowDistance: 2
              },
              items: {}
            }
          }
        }]
      };
    
      for (let i = 0; i < items.length; i++) {
        gp3.shapes[0].options.style.items[items[i]] = {
          dataValue: scores['PS'][items[i]]
        };
      }
    
    
      // TIME CHART
      // -----------------------------
      let gtime = {
        type: 'line',
        backgroundColor: 'none',
        borderColor: '#ddd',
        borderWidth: '1px',
        width: '66.66%',
        height: '45%',
        x: '0%',
        y: '55%',
        title: {
          text: 'Public trust for leaders of main political parties (trend)',
          fontSize: '13px'
        },
        plot: {
          maxNodes: 0,
          maxTrackers: 0
        },
        plotarea: {
          margin: '140px 30px 30px 40px'
        },
        scaleX: {
          maxItems: 6,
          step: 'month',
          item: {
            fontSize: 10
          },
          transform: {
            type: 'date',
            text: '%M %Y'
          }
        },
        scaleY: {
          maxItems: 6,
          format: '%v%',
          step: 20,
          item: {
            fontSize: '10px'
          }
        },
        crosshairX: {
          plotLabel: {
            text: '<span style="color:%color;font-weight:bold;">%node-value%</span> %plot-text',
            backgroundColor: '#f9f9f9',
            borderRadius: '5px',
            borderWidth: 0,
            callout: false,
            distance: 10,
            fontSize: '12px',
            multiple: false,
            padding: '8px',
            shadow: true,
            shadowAlpha: 0.5,
            shadowColor: '#333',
            shadowDistance: 1
          },
          scaleLabel: {
            fontSize: '10px'
          },
          mask: {
            backgroundColor: '#eee'
          }
        },
        shapes: [{
            type: 'circle',
            backgroundImage: parties['REM'].icon,
            borderWidth: '6px',
            borderColor: parties['REM'].color,
            size: '32px',
            x: '20%',
            y: '64px',
            label: {
              text: parties['REM'].leader + '<br>(REM)',
              fontSize: '10px',
              offsetY: '48px'
            }
          },
          {
            type: 'circle',
            backgroundImage: parties['LR'].icon,
            borderWidth: '6px',
            borderColor: parties['LR'].color,
            size: 32,
            x: '50%',
            y: '64px',
            label: {
              text: parties['LR'].leader + '<br>(LR)',
              fontSize: '10px',
              offsetY: '48px'
            }
          },
          {
            type: 'circle',
            size: 32,
            backgroundImage: parties['PS'].icon,
            borderColor: parties['PS'].color,
            borderWidth: '6px',
            x: '80%',
            y: '64px',
            label: {
              text: parties['PS'].leader + '<br>(PS)',
              fontSize: '10px',
              offsetY: '48px'
            }
          }
        ],
        series: []
      };
      let sdata = [
        [],
        [],
        []
      ];
    
      let ts = Date.UTC(2018, 10, 15);
      let pc1 = 20 + 40 * Math.random();
      let pc2 = 20 + 40 * Math.random();
      let pc3 = 10 + 20 * Math.random();
    
      for (let d = 0; d < 48; d++) {
        pc1 += -6 + 12 * Math.random();
        if (pc1 > 80 || pc1 < 20) {
          pc1 += ((pc1 > 80) ? -1 : 1) * 6 * Math.random();
        }
        pc1 = parseFloat(pc1.toFixed(1));
    
        pc2 += -5 + 10 * Math.random();
        if (pc1 > 80 || pc1 < 20) {
          pc2 += ((pc2 > 80) ? -1 : 1) * 5 * Math.random();
        }
        pc2 = parseFloat(pc2.toFixed(1));
    
        pc3 += -3 + 6 * Math.random();
        if (pc1 > 80 || pc1 < 20) {
          pc3 += ((pc3 > 80) ? -1 : 1) * 3 * Math.random();
        }
        pc3 = parseFloat(pc3.toFixed(1));
    
        sdata[0].push([ts, pc1]);
        sdata[1].push([ts, pc2]);
        sdata[2].push([ts, pc3]);
    
        ts -= 1000 * 60 * 60 * 24 * 30.5;
      }
    
      gtime.series.push({
        lineColor: parties['REM'].color,
        text: parties['REM'].leader,
        values: sdata[0].reverse()
      });
      gtime.series.push({
        lineColor: parties['LR'].color,
        text: parties['LR'].leader,
        values: sdata[1].reverse()
      });
      gtime.series.push({
        lineColor: parties['PS'].color,
        text: parties['PS'].leader,
        values: sdata[2].reverse()
      });
    
    
      // CHART CONFIG
      // -----------------------------
      let cdata = {
        backgroundColor: '#fff',
        borderColor: '#ddd',
        borderWidth: '2px',
        graphset: [
          gmain,
          gp1,
          gp2,
          gp3,
          gtime
        ]
      };
    
    
      // RENDER CHARTS
      // -----------------------------
      zingchart.render({
        id: chartId,
        width: 600,
        height: 600,
        output: 'svg',
        data: cdata,
        modules: 'color-scale'
      });
    }
    
    
    // HELPER FNS
    // -----------------------------
    
    function createBar(score, color) {
      let sBar = "<span style='font-weight:bold;color:" + color + "'>";
      for (let b = 0; b < Math.round(score / 4); b++) {
        sBar += '\u2588';
      }
      sBar += '</span>';
      return sBar;
    }
    
    function rand(min, max) {
      return Math.round(min + (max - min) * Math.random());
    };