const title = 'XXX' const imgRise = '' const imgFall = '' const color = { bar: ['#A1CBFE', '#4397FD'], x1: '#333', x2: '#333', y: '#333', label: '#333' } const data = [ { district: '福田区', '2022': 50, '2021': 40 }, { district: '罗湖区', '2022': 40, '2021': 50 }, { district: '盐田区', '2022': 40, '2021': 60 }, { district: '南山区', '2022': 50, '2021': 60 }, { district: '宝安区', '2022': 30, '2021': 10 }, { district: '龙岗区', '2022': 20, '2021': 30 }, { district: '龙华区', '2022': 30, '2021': 40 }, { district: '坪山区', '2022': 60, '2021': 50 }, { district: '光明区', '2022': 50, '2021': 40 }, { district: '大鹏新区', '2022': 40, '2021': 40 } ] const xAxisYear = [] // x轴1 年 const xAxisDistrict = [] // x轴2 行政区 const yearFont = [] // 前一年 const yearBack = [] // 后一年 const yearStatus = [] // 状态 const year = Object.keys(data[0]).filter(e => !isNaN(+e[0])).sort() // 获取年去掉行政区字段并排序 data.forEach((e, i) => { xAxisYear.push(year.join(' ')) xAxisDistrict.push(e.district) yearFont.push(e[year[0]]) yearBack.push(e[year[1]]) const s = parseFloat(+(((+e[year[1]] - e[year[0]]) / e[year[0]]) * 100).toFixed(2)) + '%' // 上升下降百分比 const m = Math.max(+e[year[0]], +e[year[1]]) // 行政区最大值 使百分比图标在当前行政区顶部 yearStatus.push({ name: e.district, value: m, percent: s, label: { normal: { show: true, offset: [0, -50], color: 'rgba(25, 100, 150, 1)', textStyle: { fontSize: '20' }, formatter: function (params) { return [ `{value|${s}}`, `{icon|}` ].join('\n') }, rich: { value: { fontSize: 16, // lineHeight: 30, color: !s.startsWith('-') ? '#1afa29' : '#d81e06', marginTop: -50 }, icon: { align: 'center', backgroundColor: { image: !s.startsWith('-') ? imgRise : imgFall } } } } } }) }) option = { title: { text: title, left: '1%', top: '3%' }, tooltip: { trigger: 'axis', formatter: (params) => { let res = `<h3 style="font-size:14px;line-height:14px;font-weight:400;color:#333;">${params[0].name}</h3>` for (const item of params) { if (item.seriesName !== 'yearStatus') { res += '<div>' + '<span style="display:inline-block;margin-right:5px;border-radius:10px;width:10px;height:10px;background-color:' + item.color.colorStops[0].color + ';"></span>' + item.seriesName + ' : ' + item.value + '</div>' } } return res }, axisPointer: { type: 'shadow', lineStyle: { color: '#57617B' } } }, toolbox: { show: true, right: '20%', top: '2%', feature: { dataZoom: { yAxisIndex: 'none' }, // dataView: { // readOnly: false // }, magicType: { type: ['line', 'bar'] }, restore: {}, saveAsImage: {} } }, grid: { top: '20%', left: '5%', right: '5%', bottom: '5%' }, legend: { data: year, right: '3%', top: '3%', textStyle: { color: '#5195da' } }, xAxis: [ { data: xAxisYear, axisLabel: { interval: 0 } }, { type: 'category', offset: 20, axisPointer: { type: 'none' }, axisTick: { show: false }, axisLine: { show: false }, position: 'bottom', data: xAxisDistrict } ], yAxis: { nameTextStyle: { color: '#000' }, type: 'value', axisLine: { show: true, lineStyle: { color: '#3585d5' } }, splitLine: { show: false, lineStyle: { type: 'dotted', color: '#3585d5' } }, axisTick: { show: false }, axisLabel: { fontSize: 16, color: '#000' }, boundaryGap: ['20%', '20%'] }, series: [ { name: year[0], xAxisIndex: 1, type: 'bar', barGap:0 , smooth: true, symbol: 'none', showSymbol: false, symbolSize: 8, itemStyle: { normal: { label: { show: true, position: 'top', textStyle: { color: 'black', fontSize: 12 } }, color: { x: 0, y: 0, x2: 0, y2: 1, type: 'linear', global: false, colorStops: [ { offset: 0, color: '#00FFE3' }, { offset: 1, color: '#4693EC' } ] } } }, data: yearFont }, { name: 'yearStatus', data: yearStatus, type: 'scatter', symbol: 'circle', symbolSize: 0.01, xAxisIndex: 1, itemStyle: { normal: { color: 'rgba(0,0,0,0)' } } }, { name: year[1], xAxisIndex: 1, type: 'bar', barGap:0 , smooth: true, symbol: 'none', showSymbol: false, symbolSize: 8, itemStyle: { normal: { label: { show: true, position: 'top', textStyle: { color: 'black', fontSize: 12 } }, color: { x: 0, y: 0, x2: 0, y2: 1, type: 'linear', global: false, colorStops: [ { offset: 0, color: '#28D8E8' }, { offset: 1, color: '#0489F0' } ] } } }, data: yearBack } ] };