• Edit
  • Download
    1. <!doctype html>
    2. <html>
    3.  
    4. <head>
    5. <meta charset="utf-8">
    6. <title>ZingSoft Demo</title>
    7. <script nonce="undefined" src="https://cdn.zingchart.com/zingchart.min.js"></script>
    8. <style>
    9. .zc-parties {
    10. padding: 1rem;
    11. display: flex;
    12. justify-content: center;
    13. background: #fff;
    14. box-sizing: border-box;
    15. }
    16.  
    17. .zc-body {
    18. background: #fff;
    19. }
    20.  
    21. .chart--container {
    22. height: 100%;
    23. width: 100%;
    24. min-height: 530px;
    25. }
    26.  
    27. .zc-ref {
    28. display: none;
    29. }
    30. </style>
    31. </head>
    32.  
    33. <body class="zc-body">
    34.  
    35. <div class="zc-parties">
    36.  
    37. <div id="myChart">
    38. <a href="https://www.zingchart.com/" rel="noopener" class="zc-ref">Powered by ZingChart</a>
    39. </div>
    40.  
    41. </div>
    42.  
    43. <script>
    44. ZC.LICENSE = ["569d52cefae586f634c54f86dc99e6a9", "b55b025e438fa8a98e32482b5f768ff5"];
    45. // INIT
    46. // -----------------------------
    47. // Define Module Location
    48. zingchart.MODULESDIR = "https://cdn.zingchart.com/modules/";
    49. // Load Maps
    50. zingchart.loadModules('maps, maps-fraL2, color-scale', init);
    51.  
    52. function init() {
    53.  
    54. // DEFINE CHART LOCATIONS (IDS)
    55. // -----------------------------
    56. // Main chart render location(s)
    57. let chartId = 'myChart';
    58.  
    59.  
    60. // PARTY CHARTS DATA & CONFIG
    61. // -----------------------------
    62.  
    63. // 3 main parties
    64. let parties = {
    65. REM: {
    66. name: 'REM: La Republique En Marche!',
    67. color: '#FB8B24',
    68. leader: 'Christophe Castaner',
    69. icon: ''
    70. },
    71. LR: {
    72. name: 'LR: The Republicans',
    73. color: '#00A8E8',
    74. leader: 'Laurent Wauquiez',
    75. icon: ''
    76. },
    77. PS: {
    78. name: 'PS: Socialist Party',
    79. color: '#FD2C88',
    80. leader: 'Olivier Faure',
    81. icon: ''
    82. }
    83. };
    84.  
    85. let items = zingchart.maps.getItems('fraL2'),
    86. i;
    87.  
    88. // For each department, create fake data with percentages for each party
    89. let scores = {
    90. REM: {},
    91. LR: {},
    92. PS: {}
    93. }
    94. // Fill w/ data
    95. for (let i = 0; i < items.length; i++) {
    96. let base = 35000;
    97. let above90 = rand(0, 100) > 90;
    98. let lenCheck = i % (Math.round(items.length / 4)) === 0;
    99. if (above90 || lenCheck) base = 25000;
    100. let pp1 = parseFloat((rand(base, 45000) / 1000).toFixed(3));
    101. let pp2 = parseFloat((rand(base, 45000) / 1000).toFixed(3));
    102. let pp3 = parseFloat((100 - pp1 - pp2).toFixed(3));
    103. scores['REM'][items[i]] = pp1;
    104. scores['LR'][items[i]] = pp2;
    105. scores['PS'][items[i]] = pp3;
    106. }
    107.  
    108. // Main chart config
    109. let gmain = {
    110. type: 'null',
    111. backgroundColor: 'none',
    112. borderColor: '#ddd',
    113. borderWidth: 1,
    114. width: '66.66%',
    115. height: '55%',
    116. x: '0%',
    117. y: '0%',
    118. title: {
    119. fontSize: '13px',
    120. text: 'Projected election results'
    121. },
    122. shapes: [{
    123. type: 'zingchart.maps',
    124. options: {
    125. id: 'mapmain',
    126. name: 'fraL2',
    127. x: '2.5%',
    128. y: '5%',
    129. width: '95%',
    130. height: '95%',
    131. scale: true,
    132. zooming: false,
    133. panning: false,
    134. scrolling: false,
    135. choropleth: {
    136. aspect: 'intervals',
    137. steps: [0, 1, 2, 3],
    138. colors: [parties['REM'].color, parties['LR'].color, parties['PS'].color]
    139. },
    140. style: {
    141. borderAlpha: 0.1,
    142. borderColor: '#fff',
    143. controls: {
    144. visible: false
    145. },
    146. label: {
    147. visible: false
    148. },
    149. hoverState: {
    150. backgroundColor: 'none',
    151. shadowAlpha: 0.2,
    152. shadowDistance: 2,
    153. shadow: true,
    154. shadowColor: '#333'
    155. },
    156. tooltip: {
    157. text: '<span style="font-size:29px;font-weight:bold;">%text</span>%data-preview<br><br><br><br>',
    158. align: 'left',
    159. backgroundColor: '#f9f9f9',
    160. borderRadius: '3px',
    161. borderWidth: 0,
    162. callout: true,
    163. calloutWidth: 16,
    164. calloutHeight: 8,
    165. color: '#000',
    166. fontSize: '13px',
    167. lineHeight: 15,
    168. padding: '10px 20px',
    169. shadow: true,
    170. shadowAlpha: 0.7,
    171. shadowColor: '#333',
    172. shadowDistance: 2,
    173. },
    174. items: {}
    175. }
    176. }
    177. }]
    178. };
    179.  
    180. for (let i = 0; i < items.length; i++) {
    181. let fVal = 0,
    182. sPreview = '';
    183.  
    184. let aPreview = [];
    185. aPreview.push(['REM', scores['REM'][items[i]]]);
    186. aPreview.push(['LR', scores['LR'][items[i]]]);
    187. aPreview.push(['PS', scores['PS'][items[i]]]);
    188. aPreview.sort(function(a, b) {
    189. return b[1] - a[1];
    190. });
    191. for (let p = 0; p < aPreview.length; p++) {
    192. let bars = createBar(aPreview[p][1], parties[aPreview[p][0]].color);
    193. sPreview += '<br>' + (p + 1) + ' ' + bars + ' ' + aPreview[p][0] + ': ' + aPreview[p][1] + '%';
    194. }
    195.  
    196. if (scores['REM'][items[i]] > scores['LR'][items[i]] && scores['REM'][items[i]] > scores['PS'][items[i]]) {
    197. fVal = 0.5;
    198. }
    199. if (scores['LR'][items[i]] > scores['REM'][items[i]] && scores['LR'][items[i]] > scores['PS'][items[i]]) {
    200. fVal = 1.5;
    201. }
    202. if (scores['PS'][items[i]] > scores['REM'][items[i]] && scores['PS'][items[i]] > scores['LR'][items[i]]) {
    203. fVal = 2.5;
    204. }
    205. gmain.shapes[0].options.style.items[items[i]] = {
    206. dataValue: fVal,
    207. dataPreview: sPreview
    208. };
    209. }
    210.  
    211.  
    212. // PARTY 1
    213. // -----------------------------
    214. let gp1 = {
    215. type: 'null',
    216. backgroundColor: 'none',
    217. borderColor: '#ddd',
    218. borderWidth: 1,
    219. width: '33.33%',
    220. height: '33.33%',
    221. x: '66.66%',
    222. y: '0%',
    223. title: {
    224. fontSize: '11px',
    225. text: parties['REM'].name
    226. },
    227. colorScale: {
    228. cursor: {
    229. size: 3
    230. },
    231. item: {
    232. fontSize: 10,
    233. offsetY: -5
    234. },
    235. layout: 'h',
    236. map: true,
    237. margin: 'auto auto 20 auto',
    238. width: '100px',
    239. height: '10px'
    240. },
    241. shapes: [{
    242. type: 'zingchart.maps',
    243. options: {
    244. id: 'mapp1',
    245. name: 'fraL2',
    246. colorScale: true,
    247. panning: false,
    248. scale: true,
    249. scrolling: false,
    250. x: '5%',
    251. y: '10%',
    252. width: '90%',
    253. height: '80%',
    254. zooming: false,
    255. choropleth: {
    256. aspect: 'gradient',
    257. progression: 'lin',
    258. color: parties['REM'].color,
    259. maxPercent: 50,
    260. effect: 'lighten',
    261. mirrored: true
    262. },
    263. style: {
    264. borderAlpha: 0.1,
    265. borderColor: parties['REM'].color,
    266. controls: {
    267. visible: false
    268. },
    269. label: {
    270. visible: false
    271. },
    272. hoverState: {
    273. backgroundColor: 'none',
    274. shadowAlpha: 0.1,
    275. shadowDistance: 0,
    276. shadow: true,
    277. shadowColor: '#369'
    278. },
    279. tooltip: {
    280. text: '%text: %data-value%',
    281. backgroundColor: parties['REM'].color,
    282. borderRadius: '3px',
    283. borderWidth: 0,
    284. callout: true,
    285. calloutWidth: '16px',
    286. calloutHeight: '8px',
    287. color: '#fff',
    288. fontSize: 13,
    289. fontWeight: 'bold',
    290. padding: '10px 20px',
    291. shadow: true,
    292. shadowColor: '#333',
    293. shadowAlpha: 0.7,
    294. shadowDistance: 2
    295. },
    296. items: {}
    297. }
    298. }
    299. }]
    300. };
    301.  
    302. for (let i = 0; i < items.length; i++) {
    303. gp1.shapes[0].options.style.items[items[i]] = {
    304. dataValue: scores['REM'][items[i]]
    305. };
    306. }
    307.  
    308.  
    309. // PARTY 2
    310. // -----------------------------
    311. let gp2 = {
    312. type: 'null',
    313. backgroundColor: 'none',
    314. borderColor: '#ddd',
    315. borderWidth: 1,
    316. x: '66.66%',
    317. y: '33.33%',
    318. width: '33.33%',
    319. height: '33.33%',
    320. title: {
    321. fontSize: '11px',
    322. text: parties['LR'].name
    323. },
    324. colorScale: {
    325. layout: 'h',
    326. width: '100px',
    327. height: '10px',
    328. map: true,
    329. margin: 'auto auto 20px auto',
    330. cursor: {
    331. size: 3
    332. },
    333. item: {
    334. fontSize: '10px',
    335. offsetY: '-5px'
    336. }
    337. },
    338. shapes: [{
    339. type: 'zingchart.maps',
    340. options: {
    341. id: 'mapp2',
    342. name: 'fraL2',
    343. colorScale: true,
    344. panning: false,
    345. scale: true,
    346. scrolling: false,
    347. width: '90%',
    348. height: '80%',
    349. x: '5%',
    350. y: '10%',
    351. zooming: false,
    352. choropleth: {
    353. aspect: 'gradient',
    354. color: parties['LR'].color,
    355. effect: 'lighten',
    356. maxPercent: 50,
    357. mirrored: true,
    358. progression: 'lin'
    359. },
    360. style: {
    361. borderAlpha: 0.1,
    362. borderColor: parties['LR'].color,
    363. controls: {
    364. visible: false
    365. },
    366. label: {
    367. visible: false
    368. },
    369. hoverState: {
    370. backgroundColor: 'none',
    371. shadow: true,
    372. shadowAlpha: 0.1,
    373. shadowColor: '#369',
    374. shadowDistance: 0
    375. },
    376. tooltip: {
    377. text: '%text: %data-value%',
    378. backgroundColor: parties['LR'].color,
    379. borderRadius: '3px',
    380. borderWidth: 0,
    381. callout: true,
    382. calloutWidth: '16px',
    383. calloutHeight: '8px',
    384. color: '#fff',
    385. fontSize: '13px',
    386. fontWeight: 'bold',
    387. padding: '10px 20px',
    388. shadow: true,
    389. shadowAlpha: 0.7,
    390. shadowColor: '#333',
    391. shadowDistance: 2,
    392. },
    393. items: {}
    394. }
    395. }
    396. }]
    397. };
    398.  
    399. for (let i = 0; i < items.length; i++) {
    400. gp2.shapes[0].options.style.items[items[i]] = {
    401. dataValue: scores['LR'][items[i]]
    402. };
    403. }
    404.  
    405.  
    406. // PARTY 3
    407. // -----------------------------
    408. let gp3 = {
    409. type: 'null',
    410. backgroundColor: 'none',
    411. borderColor: '#ddd',
    412. borderWidth: 1,
    413. width: '33.33%',
    414. height: '33.33%',
    415. x: '66.66%',
    416. y: '66.66%',
    417. title: {
    418. fontSize: '11px',
    419. text: parties['PS'].name
    420. },
    421. colorScale: {
    422. layout: 'h',
    423. map: true,
    424. margin: 'auto auto 20px auto',
    425. width: '100px',
    426. height: '10px',
    427. cursor: {
    428. size: 3
    429. },
    430. item: {
    431. fontSize: '10px',
    432. offsetY: '-5px'
    433. }
    434. },
    435. shapes: [{
    436. type: 'zingchart.maps',
    437. options: {
    438. id: 'mapp3',
    439. name: 'fraL2',
    440. colorScale: true,
    441. panning: false,
    442. scale: true,
    443. scrolling: false,
    444. width: '90%',
    445. height: '80%',
    446. x: '5%',
    447. y: '10%',
    448. zooming: false,
    449. choropleth: {
    450. aspect: 'gradient',
    451. color: parties['PS'].color,
    452. effect: 'lighten',
    453. maxPercent: 50,
    454. mirrored: true,
    455. progression: 'lin'
    456. },
    457. style: {
    458. borderAlpha: 0.1,
    459. borderColor: parties['PS'].color,
    460. controls: {
    461. visible: false
    462. },
    463. label: {
    464. visible: false
    465. },
    466. hoverState: {
    467. backgroundColor: 'none',
    468. shadow: true,
    469. shadowAlpha: 0.1,
    470. shadowColor: '#369',
    471. shadowDistance: 0
    472. },
    473. tooltip: {
    474. text: '%text: %data-value%',
    475. backgroundColor: parties['PS'].color,
    476. borderRadius: '3px',
    477. borderWidth: 0,
    478. callout: true,
    479. calloutWidth: '16px',
    480. calloutHeight: '8px',
    481. color: '#fff',
    482. fontSize: '13px',
    483. fontWeight: 'bold',
    484. padding: '10px 20px',
    485. shadow: true,
    486. shadowAlpha: 0.7,
    487. shadowColor: '#333',
    488. shadowDistance: 2
    489. },
    490. items: {}
    491. }
    492. }
    493. }]
    494. };
    495.  
    496. for (let i = 0; i < items.length; i++) {
    497. gp3.shapes[0].options.style.items[items[i]] = {
    498. dataValue: scores['PS'][items[i]]
    499. };
    500. }
    501.  
    502.  
    503. // TIME CHART
    504. // -----------------------------
    505. let gtime = {
    506. type: 'line',
    507. backgroundColor: 'none',
    508. borderColor: '#ddd',
    509. borderWidth: '1px',
    510. width: '66.66%',
    511. height: '45%',
    512. x: '0%',
    513. y: '55%',
    514. title: {
    515. text: 'Public trust for leaders of main political parties (trend)',
    516. fontSize: '13px'
    517. },
    518. plot: {
    519. maxNodes: 0,
    520. maxTrackers: 0
    521. },
    522. plotarea: {
    523. margin: '140px 30px 30px 40px'
    524. },
    525. scaleX: {
    526. maxItems: 6,
    527. step: 'month',
    528. item: {
    529. fontSize: 10
    530. },
    531. transform: {
    532. type: 'date',
    533. text: '%M %Y'
    534. }
    535. },
    536. scaleY: {
    537. maxItems: 6,
    538. format: '%v%',
    539. step: 20,
    540. item: {
    541. fontSize: '10px'
    542. }
    543. },
    544. crosshairX: {
    545. plotLabel: {
    546. text: '<span style="color:%color;font-weight:bold;">%node-value%</span> %plot-text',
    547. backgroundColor: '#f9f9f9',
    548. borderRadius: '5px',
    549. borderWidth: 0,
    550. callout: false,
    551. distance: 10,
    552. fontSize: '12px',
    553. multiple: false,
    554. padding: '8px',
    555. shadow: true,
    556. shadowAlpha: 0.5,
    557. shadowColor: '#333',
    558. shadowDistance: 1
    559. },
    560. scaleLabel: {
    561. fontSize: '10px'
    562. },
    563. mask: {
    564. backgroundColor: '#eee'
    565. }
    566. },
    567. shapes: [{
    568. type: 'circle',
    569. backgroundImage: parties['REM'].icon,
    570. borderWidth: '6px',
    571. borderColor: parties['REM'].color,
    572. size: '32px',
    573. x: '20%',
    574. y: '64px',
    575. label: {
    576. text: parties['REM'].leader + '<br>(REM)',
    577. fontSize: '10px',
    578. offsetY: '48px'
    579. }
    580. },
    581. {
    582. type: 'circle',
    583. backgroundImage: parties['LR'].icon,
    584. borderWidth: '6px',
    585. borderColor: parties['LR'].color,
    586. size: 32,
    587. x: '50%',
    588. y: '64px',
    589. label: {
    590. text: parties['LR'].leader + '<br>(LR)',
    591. fontSize: '10px',
    592. offsetY: '48px'
    593. }
    594. },
    595. {
    596. type: 'circle',
    597. size: 32,
    598. backgroundImage: parties['PS'].icon,
    599. borderColor: parties['PS'].color,
    600. borderWidth: '6px',
    601. x: '80%',
    602. y: '64px',
    603. label: {
    604. text: parties['PS'].leader + '<br>(PS)',
    605. fontSize: '10px',
    606. offsetY: '48px'
    607. }
    608. }
    609. ],
    610. series: []
    611. };
    612. let sdata = [
    613. [],
    614. [],
    615. []
    616. ];
    617.  
    618. let ts = Date.UTC(2018, 10, 15);
    619. let pc1 = 20 + 40 * Math.random();
    620. let pc2 = 20 + 40 * Math.random();
    621. let pc3 = 10 + 20 * Math.random();
    622.  
    623. for (let d = 0; d < 48; d++) {
    624. pc1 += -6 + 12 * Math.random();
    625. if (pc1 > 80 || pc1 < 20) {
    626. pc1 += ((pc1 > 80) ? -1 : 1) * 6 * Math.random();
    627. }
    628. pc1 = parseFloat(pc1.toFixed(1));
    629.  
    630. pc2 += -5 + 10 * Math.random();
    631. if (pc1 > 80 || pc1 < 20) {
    632. pc2 += ((pc2 > 80) ? -1 : 1) * 5 * Math.random();
    633. }
    634. pc2 = parseFloat(pc2.toFixed(1));
    635.  
    636. pc3 += -3 + 6 * Math.random();
    637. if (pc1 > 80 || pc1 < 20) {
    638. pc3 += ((pc3 > 80) ? -1 : 1) * 3 * Math.random();
    639. }
    640. pc3 = parseFloat(pc3.toFixed(1));
    641.  
    642. sdata[0].push([ts, pc1]);
    643. sdata[1].push([ts, pc2]);
    644. sdata[2].push([ts, pc3]);
    645.  
    646. ts -= 1000 * 60 * 60 * 24 * 30.5;
    647. }
    648.  
    649. gtime.series.push({
    650. lineColor: parties['REM'].color,
    651. text: parties['REM'].leader,
    652. values: sdata[0].reverse()
    653. });
    654. gtime.series.push({
    655. lineColor: parties['LR'].color,
    656. text: parties['LR'].leader,
    657. values: sdata[1].reverse()
    658. });
    659. gtime.series.push({
    660. lineColor: parties['PS'].color,
    661. text: parties['PS'].leader,
    662. values: sdata[2].reverse()
    663. });
    664.  
    665.  
    666. // CHART CONFIG
    667. // -----------------------------
    668. let cdata = {
    669. backgroundColor: '#fff',
    670. borderColor: '#ddd',
    671. borderWidth: '2px',
    672. graphset: [
    673. gmain,
    674. gp1,
    675. gp2,
    676. gp3,
    677. gtime
    678. ]
    679. };
    680.  
    681.  
    682. // RENDER CHARTS
    683. // -----------------------------
    684. zingchart.render({
    685. id: chartId,
    686. width: 600,
    687. height: 600,
    688. output: 'svg',
    689. data: cdata,
    690. modules: 'color-scale'
    691. });
    692. }
    693.  
    694.  
    695. // HELPER FNS
    696. // -----------------------------
    697.  
    698. function createBar(score, color) {
    699. let sBar = "<span style='font-weight:bold;color:" + color + "'>";
    700. for (let b = 0; b < Math.round(score / 4); b++) {
    701. sBar += '\u2588';
    702. }
    703. sBar += '</span>';
    704. return sBar;
    705. }
    706.  
    707. function rand(min, max) {
    708. return Math.round(min + (max - min) * Math.random());
    709. };
    710. </script>
    711. </body>
    712.  
    713. </html>
    1. <!doctype html>
    2. <html>
    3.  
    4. <head>
    5. <meta charset="utf-8">
    6. <title>ZingSoft Demo</title>
    7. <script src="https://cdn.zingchart.com/zingchart.min.js"></script>
    8. </head>
    9.  
    10. <body class="zc-body">
    11.  
    12. <div class="zc-parties">
    13.  
    14. <div id="myChart">
    15. <a href="https://www.zingchart.com/" rel="noopener" class="zc-ref">Powered by ZingChart</a>
    16. </div>
    17.  
    18. </div>
    19.  
    20. </body>
    21.  
    22. </html>
    1. .zc-parties {
    2. padding: 1rem;
    3. display: flex;
    4. justify-content: center;
    5. background: #fff;
    6. box-sizing: border-box;
    7. }
    8.  
    9. .zc-body {
    10. background: #fff;
    11. }
    12.  
    13. .chart--container {
    14. height: 100%;
    15. width: 100%;
    16. min-height: 530px;
    17. }
    18.  
    19. .zc-ref {
    20. display: none;
    21. }
    1. // INIT
    2. // -----------------------------
    3. // Define Module Location
    4. zingchart.MODULESDIR = "https://cdn.zingchart.com/modules/";
    5. // Load Maps
    6. zingchart.loadModules('maps, maps-fraL2, color-scale', init);
    7.  
    8. function init() {
    9.  
    10. // DEFINE CHART LOCATIONS (IDS)
    11. // -----------------------------
    12. // Main chart render location(s)
    13. let chartId = 'myChart';
    14.  
    15.  
    16. // PARTY CHARTS DATA & CONFIG
    17. // -----------------------------
    18.  
    19. // 3 main parties
    20. let parties = {
    21. REM: {
    22. name: 'REM: La Republique En Marche!',
    23. color: '#FB8B24',
    24. leader: 'Christophe Castaner',
    25. icon: ''
    26. },
    27. LR: {
    28. name: 'LR: The Republicans',
    29. color: '#00A8E8',
    30. leader: 'Laurent Wauquiez',
    31. icon: ''
    32. },
    33. PS: {
    34. name: 'PS: Socialist Party',
    35. color: '#FD2C88',
    36. leader: 'Olivier Faure',
    37. icon: ''
    38. }
    39. };
    40.  
    41. let items = zingchart.maps.getItems('fraL2'),
    42. i;
    43.  
    44. // For each department, create fake data with percentages for each party
    45. let scores = {
    46. REM: {},
    47. LR: {},
    48. PS: {}
    49. }
    50. // Fill w/ data
    51. for (let i = 0; i < items.length; i++) {
    52. let base = 35000;
    53. let above90 = rand(0, 100) > 90;
    54. let lenCheck = i % (Math.round(items.length / 4)) === 0;
    55. if (above90 || lenCheck) base = 25000;
    56. let pp1 = parseFloat((rand(base, 45000) / 1000).toFixed(3));
    57. let pp2 = parseFloat((rand(base, 45000) / 1000).toFixed(3));
    58. let pp3 = parseFloat((100 - pp1 - pp2).toFixed(3));
    59. scores['REM'][items[i]] = pp1;
    60. scores['LR'][items[i]] = pp2;
    61. scores['PS'][items[i]] = pp3;
    62. }
    63.  
    64. // Main chart config
    65. let gmain = {
    66. type: 'null',
    67. backgroundColor: 'none',
    68. borderColor: '#ddd',
    69. borderWidth: 1,
    70. width: '66.66%',
    71. height: '55%',
    72. x: '0%',
    73. y: '0%',
    74. title: {
    75. fontSize: '13px',
    76. text: 'Projected election results'
    77. },
    78. shapes: [{
    79. type: 'zingchart.maps',
    80. options: {
    81. id: 'mapmain',
    82. name: 'fraL2',
    83. x: '2.5%',
    84. y: '5%',
    85. width: '95%',
    86. height: '95%',
    87. scale: true,
    88. zooming: false,
    89. panning: false,
    90. scrolling: false,
    91. choropleth: {
    92. aspect: 'intervals',
    93. steps: [0, 1, 2, 3],
    94. colors: [parties['REM'].color, parties['LR'].color, parties['PS'].color]
    95. },
    96. style: {
    97. borderAlpha: 0.1,
    98. borderColor: '#fff',
    99. controls: {
    100. visible: false
    101. },
    102. label: {
    103. visible: false
    104. },
    105. hoverState: {
    106. backgroundColor: 'none',
    107. shadowAlpha: 0.2,
    108. shadowDistance: 2,
    109. shadow: true,
    110. shadowColor: '#333'
    111. },
    112. tooltip: {
    113. text: '<span style="font-size:29px;font-weight:bold;">%text</span>%data-preview<br><br><br><br>',
    114. align: 'left',
    115. backgroundColor: '#f9f9f9',
    116. borderRadius: '3px',
    117. borderWidth: 0,
    118. callout: true,
    119. calloutWidth: 16,
    120. calloutHeight: 8,
    121. color: '#000',
    122. fontSize: '13px',
    123. lineHeight: 15,
    124. padding: '10px 20px',
    125. shadow: true,
    126. shadowAlpha: 0.7,
    127. shadowColor: '#333',
    128. shadowDistance: 2,
    129. },
    130. items: {}
    131. }
    132. }
    133. }]
    134. };
    135.  
    136. for (let i = 0; i < items.length; i++) {
    137. let fVal = 0,
    138. sPreview = '';
    139.  
    140. let aPreview = [];
    141. aPreview.push(['REM', scores['REM'][items[i]]]);
    142. aPreview.push(['LR', scores['LR'][items[i]]]);
    143. aPreview.push(['PS', scores['PS'][items[i]]]);
    144. aPreview.sort(function(a, b) {
    145. return b[1] - a[1];
    146. });
    147. for (let p = 0; p < aPreview.length; p++) {
    148. let bars = createBar(aPreview[p][1], parties[aPreview[p][0]].color);
    149. sPreview += '<br>' + (p + 1) + ' ' + bars + ' ' + aPreview[p][0] + ': ' + aPreview[p][1] + '%';
    150. }
    151.  
    152. if (scores['REM'][items[i]] > scores['LR'][items[i]] && scores['REM'][items[i]] > scores['PS'][items[i]]) {
    153. fVal = 0.5;
    154. }
    155. if (scores['LR'][items[i]] > scores['REM'][items[i]] && scores['LR'][items[i]] > scores['PS'][items[i]]) {
    156. fVal = 1.5;
    157. }
    158. if (scores['PS'][items[i]] > scores['REM'][items[i]] && scores['PS'][items[i]] > scores['LR'][items[i]]) {
    159. fVal = 2.5;
    160. }
    161. gmain.shapes[0].options.style.items[items[i]] = {
    162. dataValue: fVal,
    163. dataPreview: sPreview
    164. };
    165. }
    166.  
    167.  
    168. // PARTY 1
    169. // -----------------------------
    170. let gp1 = {
    171. type: 'null',
    172. backgroundColor: 'none',
    173. borderColor: '#ddd',
    174. borderWidth: 1,
    175. width: '33.33%',
    176. height: '33.33%',
    177. x: '66.66%',
    178. y: '0%',
    179. title: {
    180. fontSize: '11px',
    181. text: parties['REM'].name
    182. },
    183. colorScale: {
    184. cursor: {
    185. size: 3
    186. },
    187. item: {
    188. fontSize: 10,
    189. offsetY: -5
    190. },
    191. layout: 'h',
    192. map: true,
    193. margin: 'auto auto 20 auto',
    194. width: '100px',
    195. height: '10px'
    196. },
    197. shapes: [{
    198. type: 'zingchart.maps',
    199. options: {
    200. id: 'mapp1',
    201. name: 'fraL2',
    202. colorScale: true,
    203. panning: false,
    204. scale: true,
    205. scrolling: false,
    206. x: '5%',
    207. y: '10%',
    208. width: '90%',
    209. height: '80%',
    210. zooming: false,
    211. choropleth: {
    212. aspect: 'gradient',
    213. progression: 'lin',
    214. color: parties['REM'].color,
    215. maxPercent: 50,
    216. effect: 'lighten',
    217. mirrored: true
    218. },
    219. style: {
    220. borderAlpha: 0.1,
    221. borderColor: parties['REM'].color,
    222. controls: {
    223. visible: false
    224. },
    225. label: {
    226. visible: false
    227. },
    228. hoverState: {
    229. backgroundColor: 'none',
    230. shadowAlpha: 0.1,
    231. shadowDistance: 0,
    232. shadow: true,
    233. shadowColor: '#369'
    234. },
    235. tooltip: {
    236. text: '%text: %data-value%',
    237. backgroundColor: parties['REM'].color,
    238. borderRadius: '3px',
    239. borderWidth: 0,
    240. callout: true,
    241. calloutWidth: '16px',
    242. calloutHeight: '8px',
    243. color: '#fff',
    244. fontSize: 13,
    245. fontWeight: 'bold',
    246. padding: '10px 20px',
    247. shadow: true,
    248. shadowColor: '#333',
    249. shadowAlpha: 0.7,
    250. shadowDistance: 2
    251. },
    252. items: {}
    253. }
    254. }
    255. }]
    256. };
    257.  
    258. for (let i = 0; i < items.length; i++) {
    259. gp1.shapes[0].options.style.items[items[i]] = {
    260. dataValue: scores['REM'][items[i]]
    261. };
    262. }
    263.  
    264.  
    265. // PARTY 2
    266. // -----------------------------
    267. let gp2 = {
    268. type: 'null',
    269. backgroundColor: 'none',
    270. borderColor: '#ddd',
    271. borderWidth: 1,
    272. x: '66.66%',
    273. y: '33.33%',
    274. width: '33.33%',
    275. height: '33.33%',
    276. title: {
    277. fontSize: '11px',
    278. text: parties['LR'].name
    279. },
    280. colorScale: {
    281. layout: 'h',
    282. width: '100px',
    283. height: '10px',
    284. map: true,
    285. margin: 'auto auto 20px auto',
    286. cursor: {
    287. size: 3
    288. },
    289. item: {
    290. fontSize: '10px',
    291. offsetY: '-5px'
    292. }
    293. },
    294. shapes: [{
    295. type: 'zingchart.maps',
    296. options: {
    297. id: 'mapp2',
    298. name: 'fraL2',
    299. colorScale: true,
    300. panning: false,
    301. scale: true,
    302. scrolling: false,
    303. width: '90%',
    304. height: '80%',
    305. x: '5%',
    306. y: '10%',
    307. zooming: false,
    308. choropleth: {
    309. aspect: 'gradient',
    310. color: parties['LR'].color,
    311. effect: 'lighten',
    312. maxPercent: 50,
    313. mirrored: true,
    314. progression: 'lin'
    315. },
    316. style: {
    317. borderAlpha: 0.1,
    318. borderColor: parties['LR'].color,
    319. controls: {
    320. visible: false
    321. },
    322. label: {
    323. visible: false
    324. },
    325. hoverState: {
    326. backgroundColor: 'none',
    327. shadow: true,
    328. shadowAlpha: 0.1,
    329. shadowColor: '#369',
    330. shadowDistance: 0
    331. },
    332. tooltip: {
    333. text: '%text: %data-value%',
    334. backgroundColor: parties['LR'].color,
    335. borderRadius: '3px',
    336. borderWidth: 0,
    337. callout: true,
    338. calloutWidth: '16px',
    339. calloutHeight: '8px',
    340. color: '#fff',
    341. fontSize: '13px',
    342. fontWeight: 'bold',
    343. padding: '10px 20px',
    344. shadow: true,
    345. shadowAlpha: 0.7,
    346. shadowColor: '#333',
    347. shadowDistance: 2,
    348. },
    349. items: {}
    350. }
    351. }
    352. }]
    353. };
    354.  
    355. for (let i = 0; i < items.length; i++) {
    356. gp2.shapes[0].options.style.items[items[i]] = {
    357. dataValue: scores['LR'][items[i]]
    358. };
    359. }
    360.  
    361.  
    362. // PARTY 3
    363. // -----------------------------
    364. let gp3 = {
    365. type: 'null',
    366. backgroundColor: 'none',
    367. borderColor: '#ddd',
    368. borderWidth: 1,
    369. width: '33.33%',
    370. height: '33.33%',
    371. x: '66.66%',
    372. y: '66.66%',
    373. title: {
    374. fontSize: '11px',
    375. text: parties['PS'].name
    376. },
    377. colorScale: {
    378. layout: 'h',
    379. map: true,
    380. margin: 'auto auto 20px auto',
    381. width: '100px',
    382. height: '10px',
    383. cursor: {
    384. size: 3
    385. },
    386. item: {
    387. fontSize: '10px',
    388. offsetY: '-5px'
    389. }
    390. },
    391. shapes: [{
    392. type: 'zingchart.maps',
    393. options: {
    394. id: 'mapp3',
    395. name: 'fraL2',
    396. colorScale: true,
    397. panning: false,
    398. scale: true,
    399. scrolling: false,
    400. width: '90%',
    401. height: '80%',
    402. x: '5%',
    403. y: '10%',
    404. zooming: false,
    405. choropleth: {
    406. aspect: 'gradient',
    407. color: parties['PS'].color,
    408. effect: 'lighten',
    409. maxPercent: 50,
    410. mirrored: true,
    411. progression: 'lin'
    412. },
    413. style: {
    414. borderAlpha: 0.1,
    415. borderColor: parties['PS'].color,
    416. controls: {
    417. visible: false
    418. },
    419. label: {
    420. visible: false
    421. },
    422. hoverState: {
    423. backgroundColor: 'none',
    424. shadow: true,
    425. shadowAlpha: 0.1,
    426. shadowColor: '#369',
    427. shadowDistance: 0
    428. },
    429. tooltip: {
    430. text: '%text: %data-value%',
    431. backgroundColor: parties['PS'].color,
    432. borderRadius: '3px',
    433. borderWidth: 0,
    434. callout: true,
    435. calloutWidth: '16px',
    436. calloutHeight: '8px',
    437. color: '#fff',
    438. fontSize: '13px',
    439. fontWeight: 'bold',
    440. padding: '10px 20px',
    441. shadow: true,
    442. shadowAlpha: 0.7,
    443. shadowColor: '#333',
    444. shadowDistance: 2
    445. },
    446. items: {}
    447. }
    448. }
    449. }]
    450. };
    451.  
    452. for (let i = 0; i < items.length; i++) {
    453. gp3.shapes[0].options.style.items[items[i]] = {
    454. dataValue: scores['PS'][items[i]]
    455. };
    456. }
    457.  
    458.  
    459. // TIME CHART
    460. // -----------------------------
    461. let gtime = {
    462. type: 'line',
    463. backgroundColor: 'none',
    464. borderColor: '#ddd',
    465. borderWidth: '1px',
    466. width: '66.66%',
    467. height: '45%',
    468. x: '0%',
    469. y: '55%',
    470. title: {
    471. text: 'Public trust for leaders of main political parties (trend)',
    472. fontSize: '13px'
    473. },
    474. plot: {
    475. maxNodes: 0,
    476. maxTrackers: 0
    477. },
    478. plotarea: {
    479. margin: '140px 30px 30px 40px'
    480. },
    481. scaleX: {
    482. maxItems: 6,
    483. step: 'month',
    484. item: {
    485. fontSize: 10
    486. },
    487. transform: {
    488. type: 'date',
    489. text: '%M %Y'
    490. }
    491. },
    492. scaleY: {
    493. maxItems: 6,
    494. format: '%v%',
    495. step: 20,
    496. item: {
    497. fontSize: '10px'
    498. }
    499. },
    500. crosshairX: {
    501. plotLabel: {
    502. text: '<span style="color:%color;font-weight:bold;">%node-value%</span> %plot-text',
    503. backgroundColor: '#f9f9f9',
    504. borderRadius: '5px',
    505. borderWidth: 0,
    506. callout: false,
    507. distance: 10,
    508. fontSize: '12px',
    509. multiple: false,
    510. padding: '8px',
    511. shadow: true,
    512. shadowAlpha: 0.5,
    513. shadowColor: '#333',
    514. shadowDistance: 1
    515. },
    516. scaleLabel: {
    517. fontSize: '10px'
    518. },
    519. mask: {
    520. backgroundColor: '#eee'
    521. }
    522. },
    523. shapes: [{
    524. type: 'circle',
    525. backgroundImage: parties['REM'].icon,
    526. borderWidth: '6px',
    527. borderColor: parties['REM'].color,
    528. size: '32px',
    529. x: '20%',
    530. y: '64px',
    531. label: {
    532. text: parties['REM'].leader + '<br>(REM)',
    533. fontSize: '10px',
    534. offsetY: '48px'
    535. }
    536. },
    537. {
    538. type: 'circle',
    539. backgroundImage: parties['LR'].icon,
    540. borderWidth: '6px',
    541. borderColor: parties['LR'].color,
    542. size: 32,
    543. x: '50%',
    544. y: '64px',
    545. label: {
    546. text: parties['LR'].leader + '<br>(LR)',
    547. fontSize: '10px',
    548. offsetY: '48px'
    549. }
    550. },
    551. {
    552. type: 'circle',
    553. size: 32,
    554. backgroundImage: parties['PS'].icon,
    555. borderColor: parties['PS'].color,
    556. borderWidth: '6px',
    557. x: '80%',
    558. y: '64px',
    559. label: {
    560. text: parties['PS'].leader + '<br>(PS)',
    561. fontSize: '10px',
    562. offsetY: '48px'
    563. }
    564. }
    565. ],
    566. series: []
    567. };
    568. let sdata = [
    569. [],
    570. [],
    571. []
    572. ];
    573.  
    574. let ts = Date.UTC(2018, 10, 15);
    575. let pc1 = 20 + 40 * Math.random();
    576. let pc2 = 20 + 40 * Math.random();
    577. let pc3 = 10 + 20 * Math.random();
    578.  
    579. for (let d = 0; d < 48; d++) {
    580. pc1 += -6 + 12 * Math.random();
    581. if (pc1 > 80 || pc1 < 20) {
    582. pc1 += ((pc1 > 80) ? -1 : 1) * 6 * Math.random();
    583. }
    584. pc1 = parseFloat(pc1.toFixed(1));
    585.  
    586. pc2 += -5 + 10 * Math.random();
    587. if (pc1 > 80 || pc1 < 20) {
    588. pc2 += ((pc2 > 80) ? -1 : 1) * 5 * Math.random();
    589. }
    590. pc2 = parseFloat(pc2.toFixed(1));
    591.  
    592. pc3 += -3 + 6 * Math.random();
    593. if (pc1 > 80 || pc1 < 20) {
    594. pc3 += ((pc3 > 80) ? -1 : 1) * 3 * Math.random();
    595. }
    596. pc3 = parseFloat(pc3.toFixed(1));
    597.  
    598. sdata[0].push([ts, pc1]);
    599. sdata[1].push([ts, pc2]);
    600. sdata[2].push([ts, pc3]);
    601.  
    602. ts -= 1000 * 60 * 60 * 24 * 30.5;
    603. }
    604.  
    605. gtime.series.push({
    606. lineColor: parties['REM'].color,
    607. text: parties['REM'].leader,
    608. values: sdata[0].reverse()
    609. });
    610. gtime.series.push({
    611. lineColor: parties['LR'].color,
    612. text: parties['LR'].leader,
    613. values: sdata[1].reverse()
    614. });
    615. gtime.series.push({
    616. lineColor: parties['PS'].color,
    617. text: parties['PS'].leader,
    618. values: sdata[2].reverse()
    619. });
    620.  
    621.  
    622. // CHART CONFIG
    623. // -----------------------------
    624. let cdata = {
    625. backgroundColor: '#fff',
    626. borderColor: '#ddd',
    627. borderWidth: '2px',
    628. graphset: [
    629. gmain,
    630. gp1,
    631. gp2,
    632. gp3,
    633. gtime
    634. ]
    635. };
    636.  
    637.  
    638. // RENDER CHARTS
    639. // -----------------------------
    640. zingchart.render({
    641. id: chartId,
    642. width: 600,
    643. height: 600,
    644. output: 'svg',
    645. data: cdata,
    646. modules: 'color-scale'
    647. });
    648. }
    649.  
    650.  
    651. // HELPER FNS
    652. // -----------------------------
    653.  
    654. function createBar(score, color) {
    655. let sBar = "<span style='font-weight:bold;color:" + color + "'>";
    656. for (let b = 0; b < Math.round(score / 4); b++) {
    657. sBar += '\u2588';
    658. }
    659. sBar += '</span>';
    660. return sBar;
    661. }
    662.  
    663. function rand(min, max) {
    664. return Math.round(min + (max - min) * Math.random());
    665. };