CustomChart.vue 31 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915
  1. <template>
  2. <div ref="chartRef" :style="{ height, width }"></div>
  3. </template>
  4. <script lang="ts" setup>
  5. import { ref, Ref, watch } from 'vue';
  6. import { useECharts } from '/@/hooks/web/useECharts';
  7. import { get } from 'lodash-es';
  8. import { ModuleDataChart } from '/@/views/vent/deviceManager/configurationTable/types';
  9. import { EChartsOption, graphic } from 'echarts';
  10. import { getData, getFormattedText } from '../../hooks/helper';
  11. import { getAssetURL } from '/@/utils/ui';
  12. const props = withDefaults(
  13. defineProps<{
  14. chartData: Record<string, any>[] | Record<string, any>;
  15. chartConfig: ModuleDataChart;
  16. height?: string;
  17. width?: string;
  18. }>(),
  19. {
  20. chartData: () => [],
  21. height: '100%',
  22. width: '100%',
  23. }
  24. );
  25. const chartRef = ref<HTMLDivElement | null>(null);
  26. const { setOptions, getInstance } = useECharts(chartRef as Ref<HTMLDivElement>);
  27. // 核心方法,生成适用与 echart 的选项,这个方法需要适配 chart 类型的每种子类型
  28. const genChartOption = () => {
  29. const inst = getInstance();
  30. const domWidth = inst ? inst.getWidth() : 500;
  31. // 依据每一个图表配置生成图表选项
  32. const { yAxis = [], xAxis = [], legend, order, type, sortBy, series, dataZoom = [] } = props.chartConfig;
  33. const textStyle = {
  34. color: '#fff',
  35. };
  36. let sorttedData: any[] = [];
  37. if (Array.isArray(props.chartData)) {
  38. sorttedData = props.chartData;
  39. } else {
  40. sorttedData = [props.chartData];
  41. }
  42. // 如果这个配置指明了需要排序则执行排序
  43. if (sortBy && order) {
  44. sorttedData.sort((pre, cur) => {
  45. if (order === 'asc') {
  46. return get(pre, sortBy, 0) - get(cur, sortBy, 0);
  47. } else {
  48. return get(cur, sortBy, 0) - get(pre, sortBy, 0);
  49. }
  50. });
  51. }
  52. // 该项作为下面所有图表依赖的基准系列数据
  53. const baseSeries: { name: string; data: [string, string][]; color: string }[] = sorttedData.reduce((res: any[], baseData) => {
  54. series.forEach((serie) => {
  55. // 将读取出的数据转为数组
  56. const temp = getData(baseData, serie.readFrom) || [];
  57. res.push({
  58. name: getFormattedText(baseData, serie.label),
  59. data: (Array.isArray(temp) ? temp : [temp]).map((data) => {
  60. return [getData(data, serie.xprop), getData(data, serie.yprop)]; /** x y */
  61. // return { name: getData(data, serie.xprop), value: getData(data, serie.yprop) }; /** x y */
  62. }),
  63. color: serie.color,
  64. });
  65. });
  66. return res;
  67. }, []);
  68. const baseDataZoom = dataZoom.map((e, i) => {
  69. return {
  70. type: 'slider',
  71. show: e.show,
  72. // show: get(baseSeries, '[0].data.length', 1) > e.maxAxisLength,
  73. xAxisIndex: i,
  74. end: e.end,
  75. };
  76. });
  77. // if (type === 'pie') {
  78. // return {
  79. // textStyle,
  80. // legend: {
  81. // textStyle,
  82. // show: legend.show,
  83. // },
  84. // tooltip: {
  85. // trigger: 'item',
  86. // },
  87. // color: ['#d9a1ff', '#00d1ff', '#82fe78'],
  88. // series: baseSeries.map((serie) => {
  89. // return {
  90. // type: 'pie',
  91. // radius: ['50%', '75%'],
  92. // center: ['50%', '55%'],
  93. // labelLine: { show: false },
  94. // label: { show: false },
  95. // itemStyle: {
  96. // shadowBlur: 20,
  97. // shadowColor: '#259bcf',
  98. // },
  99. // data: serie.data.map((e) => ({
  100. // name: e[0],
  101. // value: e[1],
  102. // })),
  103. // };
  104. // }),
  105. // };
  106. // }
  107. if (type === 'pie') {
  108. return {
  109. textStyle,
  110. legend: {
  111. textStyle,
  112. show: legend.show,
  113. },
  114. tooltip: {
  115. trigger: 'item',
  116. },
  117. color: ['#73C0DE', '#5470C6', '#91CC75', '#FAC858', '#ED6666', '#17d1b2', '#2ae271', '#11bce7', '#c127f0', '#ee125b'],
  118. series: baseSeries.reduce((curr: EChartsOption[], serie) => {
  119. const colors = ['#73C0DE', '#5470C6', '#91CC75', '#FAC858', '#ED6666', '#17d1b2', '#2ae271', '#11bce7', '#c127f0', '#ee125b'];
  120. if (baseSeries.length === 1) {
  121. curr.push({
  122. ...serie,
  123. type: 'pie',
  124. radius: ['30%', '64%'],
  125. center: ['50%', '55%'],
  126. z: 1,
  127. padAngle: 5,
  128. itemStyle: {
  129. shadowColor: 'rgba(0, 0, 0, 0.5)',
  130. borderWidth: 10,
  131. },
  132. label: {
  133. show: true,
  134. position: 'inner',
  135. formatter: '{d}%',
  136. color: '#fff',
  137. fontSize: 14,
  138. rich: {
  139. d: {
  140. fontFamily: '微软雅黑',
  141. fontSize: 16,
  142. color: '#fff',
  143. lineHeight: 30,
  144. },
  145. },
  146. },
  147. labelLine: { show: false },
  148. data: serie.data.map((e) => ({
  149. name: e[0],
  150. value: e[1],
  151. })),
  152. });
  153. curr.push({
  154. ...serie,
  155. type: 'pie',
  156. radius: ['64%', '74%'],
  157. center: ['50%', '55%'],
  158. labelLine: { show: false },
  159. label: { show: false },
  160. padAngle: 5,
  161. avoidLabelOverlap: false,
  162. itemStyle: {
  163. borderWidth: 10,
  164. normal: {
  165. color: function (obj) {
  166. return `${colors[obj['dataIndex']]}88`;
  167. },
  168. },
  169. },
  170. data: serie.data.map((e) => ({
  171. name: e[0],
  172. value: e[1],
  173. })),
  174. });
  175. }
  176. curr.push({
  177. ...serie,
  178. type: 'pie',
  179. radius: ['25%', '30%'],
  180. center: ['50%', '55%'],
  181. labelLine: { show: false },
  182. label: { show: false },
  183. z: 5,
  184. padAngle: 5,
  185. itemStyle: {
  186. borderWidth: 10,
  187. normal: {
  188. color: function (obj) {
  189. return `${colors[obj['dataIndex']]}88`;
  190. },
  191. },
  192. },
  193. data: serie.data.map((e) => ({
  194. name: e[0],
  195. value: e[1],
  196. })),
  197. });
  198. return curr;
  199. }, []),
  200. };
  201. }
  202. if (type === 'pie_halo') {
  203. // 光环的图片
  204. const img =
  205. '';
  206. const color = ['#70F081', '#EEE780', '#F07070', '#ffe000', '#ffa800', '#ff5b00', '#ff3000'];
  207. return {
  208. textStyle,
  209. legend: {
  210. textStyle,
  211. show: legend.show,
  212. },
  213. tooltip: {
  214. trigger: 'item',
  215. },
  216. graphic: {
  217. elements: [
  218. {
  219. type: 'image',
  220. z: 3,
  221. style: {
  222. image: img,
  223. width: 120,
  224. height: 120,
  225. },
  226. left: 'center',
  227. top: 'center',
  228. position: [20, 20],
  229. },
  230. ],
  231. },
  232. series: baseSeries.reduce((curr: EChartsOption[], serie) => {
  233. curr.push({
  234. ...serie,
  235. type: 'pie',
  236. clockWise: false,
  237. radius: [45, 45],
  238. label: {
  239. show: true,
  240. position: 'outside',
  241. color: '#ddd',
  242. formatter: get(legend, 'formatter', '{b}\n{c}pa'),
  243. },
  244. labelLine: {
  245. length: 30,
  246. length2: 40,
  247. show: true,
  248. color: '#00ffff',
  249. },
  250. data: serie.data.map((e, i) => ({
  251. name: e[0],
  252. value: e[1],
  253. itemStyle: {
  254. normal: {
  255. borderWidth: 10,
  256. shadowBlur: 30,
  257. borderColor: color[i],
  258. shadowColor: color[i],
  259. },
  260. },
  261. })),
  262. });
  263. return curr;
  264. }, []),
  265. };
  266. }
  267. if (type === 'pie_drag') {
  268. return {
  269. legend: baseSeries.map((e) => {
  270. return {
  271. orient: 'vertical',
  272. left: '54%',
  273. top: '8%',
  274. itemGap: 34,
  275. itemWidth: 10,
  276. itemHeight: 10,
  277. align: 'left',
  278. textStyle: {
  279. fontSize: 14,
  280. color: '#b3b8cc',
  281. },
  282. data: e.data.map((e) => {
  283. return `${e[0]}: ${e[1]}`;
  284. }),
  285. };
  286. }),
  287. graphic: {
  288. elements: [
  289. {
  290. type: 'image',
  291. style: {
  292. image: getAssetURL('home-container/pie.png'),
  293. width: 20,
  294. height: 100,
  295. },
  296. left: 'center',
  297. top: 'center',
  298. },
  299. ],
  300. },
  301. tooltip: {
  302. formatter: '{b}',
  303. },
  304. series: baseSeries.reduce((curr: EChartsOption[], serie) => {
  305. curr.push({
  306. radius: ['40%', '80%'],
  307. center: ['25%', '50%'],
  308. type: 'pie',
  309. z: 10,
  310. label: {
  311. show: false,
  312. },
  313. labelLine: {
  314. show: false,
  315. },
  316. itemStyle: {
  317. opacity: 0.7,
  318. },
  319. padAngle: 5,
  320. data: serie.data.map((e) => ({
  321. name: `${e[0]}: ${e[1]}`,
  322. value: e[1],
  323. })),
  324. });
  325. curr.push({
  326. radius: ['40%', '55%'],
  327. center: ['25%', '50%'],
  328. type: 'pie',
  329. z: 5,
  330. label: {
  331. show: false,
  332. },
  333. labelLine: {
  334. show: false,
  335. },
  336. animation: false,
  337. tooltip: {
  338. show: false,
  339. },
  340. padAngle: 6,
  341. data: serie.data.map((e) => ({
  342. name: `${e[0]}: ${e[1]}`,
  343. value: e[1],
  344. })),
  345. });
  346. return curr;
  347. }, []),
  348. };
  349. }
  350. // 柱状图
  351. if (type === 'bar') {
  352. return {
  353. textStyle,
  354. grid: {
  355. top: 50,
  356. bottom: dataZoom.length ? 70 : 50,
  357. },
  358. legend: {
  359. textStyle,
  360. show: legend.show,
  361. },
  362. tooltip: {
  363. trigger: 'item',
  364. },
  365. dataZoom: baseDataZoom,
  366. xAxis: xAxis.map((e) => {
  367. return {
  368. ...e,
  369. type: 'category',
  370. axisLabel: {
  371. interval: 0,
  372. width: baseDataZoom.length ? 100 : 800 / get(baseSeries, '[0].data.length', 1),
  373. overflow: 'break',
  374. },
  375. };
  376. }),
  377. yAxis: yAxis.map((e, i) => {
  378. return {
  379. ...e,
  380. splitLine: {
  381. lineStyle: {
  382. opacity: i === 0 ? 0.1 : 0,
  383. },
  384. },
  385. };
  386. }),
  387. series: baseSeries.reduce((curr: EChartsOption[], serie, index) => {
  388. const colors = ['#66ffff', '#ffff66'];
  389. if (baseSeries.length === 1) {
  390. curr.push({
  391. ...serie,
  392. type: 'pictorialBar',
  393. symbol: 'circle',
  394. symbolPosition: 'end',
  395. symbolSize: [20, 20],
  396. symbolOffset: [0, -10],
  397. barGap: '-100%',
  398. yAxisIndex: index,
  399. itemStyle: {
  400. color: colors[index % colors.length],
  401. },
  402. });
  403. }
  404. curr.push({
  405. ...serie,
  406. type: 'bar',
  407. silent: true,
  408. yAxisIndex: index,
  409. barWidth: 20,
  410. barGap: '100%',
  411. label: {
  412. show: true,
  413. textStyle,
  414. position: 'top',
  415. },
  416. itemStyle: {
  417. color: new graphic.LinearGradient(0, 0, 0, 1, [
  418. { offset: 0, color: colors[index % colors.length] },
  419. { offset: 0.2, color: colors[index % colors.length] },
  420. { offset: 1, color: `${colors[index % colors.length]}22` },
  421. ]),
  422. borderRadius: [5, 5, 0, 0],
  423. },
  424. });
  425. return curr;
  426. }, []),
  427. };
  428. }
  429. // 折线图和上面的柱状图类似
  430. if (type === 'line') {
  431. return {
  432. textStyle,
  433. legend: {
  434. show: legend.show,
  435. top: 10,
  436. right: 10,
  437. textStyle,
  438. },
  439. grid: {
  440. left: 60,
  441. top: 40,
  442. right: 60,
  443. bottom: dataZoom.length ? 70 : 30,
  444. },
  445. dataZoom: baseDataZoom,
  446. xAxis: xAxis.map((e) => {
  447. return {
  448. ...e,
  449. type: 'category',
  450. axisLabel: {
  451. width: 100,
  452. overflow: 'break',
  453. },
  454. };
  455. }),
  456. yAxis: yAxis.map((e, i) => {
  457. return {
  458. ...e,
  459. splitLine: {
  460. lineStyle: {
  461. opacity: i === 0 ? 0.1 : 0,
  462. },
  463. },
  464. };
  465. }),
  466. series: baseSeries.map((serie) => {
  467. return {
  468. ...serie,
  469. type: 'line',
  470. };
  471. }),
  472. };
  473. }
  474. // 平滑曲线图
  475. if (type === 'line_smooth') {
  476. return {
  477. textStyle,
  478. legend: {
  479. show: legend.show,
  480. top: 10,
  481. textStyle,
  482. },
  483. grid: {
  484. left: 60,
  485. right: 60,
  486. bottom: dataZoom.length ? 70 : 30,
  487. },
  488. dataZoom: baseDataZoom,
  489. xAxis: xAxis.map((e) => {
  490. return {
  491. ...e,
  492. type: 'category',
  493. };
  494. }),
  495. yAxis: yAxis.map((e, i) => {
  496. return {
  497. ...e,
  498. splitLine: {
  499. lineStyle: {
  500. opacity: i === 0 ? 0.1 : 0,
  501. },
  502. },
  503. };
  504. }),
  505. series: baseSeries.map((serie) => {
  506. return {
  507. ...serie,
  508. type: 'line',
  509. smooth: true,
  510. itemStyle: {
  511. opacity: 0,
  512. },
  513. };
  514. }),
  515. };
  516. }
  517. // 折线区域图,即折线下面的区域填满
  518. if (type === 'line_area') {
  519. return {
  520. textStyle,
  521. legend: {
  522. textStyle,
  523. show: legend.show,
  524. },
  525. grid: {
  526. left: 50,
  527. top: 50,
  528. right: 50,
  529. bottom: dataZoom.length ? 70 : 50,
  530. },
  531. dataZoom: baseDataZoom,
  532. xAxis: xAxis.map((e) => {
  533. return {
  534. ...e,
  535. type: 'category',
  536. boundaryGap: false,
  537. };
  538. }),
  539. yAxis: yAxis.map((e, i) => {
  540. return {
  541. ...e,
  542. splitLine: {
  543. lineStyle: {
  544. color: '#fff',
  545. opacity: i === 0 ? 0.1 : 0,
  546. },
  547. },
  548. };
  549. }),
  550. series: baseSeries.map((serie, index) => {
  551. const colors = ['#66ffff', '#6666ff'];
  552. let color;
  553. if (serie.color) {
  554. color = serie.color;
  555. } else {
  556. color = colors[index % colors.length];
  557. }
  558. return {
  559. ...serie,
  560. type: 'line',
  561. symbol: 'none',
  562. endLabel: { distance: 0 },
  563. lineStyle: { color: color },
  564. areaStyle: {
  565. origin: 'auto',
  566. color: new graphic.LinearGradient(0, 0, 0, 1, [
  567. { offset: 0, color: color },
  568. { offset: 0.2, color: color },
  569. { offset: 1, color: `${color}22` },
  570. ]),
  571. },
  572. };
  573. }),
  574. };
  575. }
  576. if (type === 'line_bar') {
  577. return {
  578. textStyle,
  579. legend: {
  580. textStyle,
  581. show: legend.show,
  582. top: 10,
  583. right: 10,
  584. },
  585. grid: {
  586. left: 40,
  587. top: 50,
  588. right: 40,
  589. bottom: dataZoom.length ? 70 : 10,
  590. show: false,
  591. },
  592. dataZoom: baseDataZoom,
  593. xAxis: xAxis.map((e) => {
  594. return {
  595. ...e,
  596. type: 'category',
  597. };
  598. }),
  599. yAxis: yAxis.map((e) => {
  600. return {
  601. ...e,
  602. };
  603. }),
  604. series: baseSeries.map((serie, i) => {
  605. return {
  606. ...serie,
  607. type: i % 2 ? 'line' : 'bar',
  608. smooth: true,
  609. barWidth: 20,
  610. };
  611. }),
  612. };
  613. }
  614. // 堆叠柱状图
  615. if (type === 'bar_stack') {
  616. return {
  617. textStyle,
  618. tooltip: {
  619. trigger: 'axis',
  620. axisPointer: {
  621. type: 'shadow',
  622. },
  623. },
  624. grid: {
  625. top: 50,
  626. bottom: 30,
  627. },
  628. legend: {
  629. textStyle,
  630. show: legend.show,
  631. },
  632. xAxis: xAxis.map((e) => {
  633. return {
  634. ...e,
  635. type: 'category',
  636. };
  637. }),
  638. yAxis: yAxis.map((e) => {
  639. return {
  640. ...e,
  641. splitLine: {
  642. lineStyle: {
  643. color: 'rgba(21,80,126,0.3)',
  644. type: 'dashed',
  645. },
  646. },
  647. };
  648. }),
  649. series: baseSeries.map((serie) => {
  650. return {
  651. ...serie,
  652. type: 'bar',
  653. stack: 'stackME',
  654. barMaxWidth: '24',
  655. emphasis: {
  656. focus: 'series',
  657. },
  658. label: {
  659. show: true,
  660. position: 'top', //在上方显示
  661. textStyle: {
  662. //数值样式
  663. color: '#fff',
  664. fontSize: 14,
  665. },
  666. },
  667. };
  668. }),
  669. color: ['#F56731', '#00E8FF'],
  670. };
  671. }
  672. // 柱状图,圆柱形样式
  673. if (type === 'bar_cylinder') {
  674. return {
  675. textStyle,
  676. grid: {
  677. top: 40,
  678. bottom: 50,
  679. },
  680. legend: {
  681. textStyle,
  682. show: legend.show,
  683. },
  684. tooltip: {
  685. trigger: 'item',
  686. },
  687. xAxis: xAxis.map((e) => {
  688. return {
  689. ...e,
  690. type: 'category',
  691. axisLabel: {
  692. interval: 0,
  693. width: (domWidth - 100) / get(baseSeries, '[0].data.length', 1),
  694. overflow: 'break',
  695. },
  696. };
  697. }),
  698. yAxis: yAxis.map((e, i) => {
  699. return {
  700. ...e,
  701. splitLine: {
  702. lineStyle: {
  703. opacity: i === 0 ? 0.1 : 0,
  704. },
  705. },
  706. };
  707. }),
  708. series: baseSeries.reduce((curr: EChartsOption[], serie, index) => {
  709. // const colors = ['#66ffff', '#00ff66', '#ffff66'];
  710. const colors = ['#73C0DE', '#ED6666', '#5470C6', '#91CC75', '#FAC858', '#17d1b2', '#2ae271', '#11bce7', '#c127f0', '#ee125b'];
  711. if (baseSeries.length === 1) {
  712. curr.push({
  713. ...serie,
  714. type: 'pictorialBar',
  715. symbol: 'circle',
  716. symbolPosition: 'end',
  717. symbolSize: [20, 10],
  718. symbolOffset: [0, -5],
  719. barGap: '-100%',
  720. yAxisIndex: index,
  721. itemStyle: {
  722. color: ({ dataIndex }) => colors[dataIndex % colors.length],
  723. },
  724. });
  725. curr.push({
  726. ...serie,
  727. type: 'pictorialBar',
  728. symbol: 'circle',
  729. symbolPosition: 'start',
  730. symbolSize: [20, 10],
  731. symbolOffset: [0, 5],
  732. barGap: '-100%',
  733. yAxisIndex: index,
  734. itemStyle: {
  735. color: ({ dataIndex }) => colors[dataIndex % colors.length],
  736. },
  737. });
  738. }
  739. curr.push({
  740. ...serie,
  741. type: 'bar',
  742. // silent: true,
  743. yAxisIndex: index,
  744. barWidth: 20,
  745. barGap: '100%',
  746. itemStyle: {
  747. color: ({ dataIndex }) =>
  748. new graphic.LinearGradient(0, 0, 0, 1, [
  749. { offset: 0, color: `${colors[dataIndex % colors.length]}44` },
  750. { offset: 0.5, color: colors[dataIndex % colors.length] },
  751. { offset: 1, color: colors[dataIndex % colors.length] },
  752. ]),
  753. borderRadius: [5, 5, 0, 0],
  754. },
  755. label: {
  756. show: true, //开启显示
  757. position: 'top', //在上方显示
  758. textStyle: {
  759. //数值样式
  760. color: '#ffffffbb',
  761. fontSize: 13,
  762. },
  763. formatter: function (obj) {
  764. return obj['data'][1];
  765. },
  766. },
  767. });
  768. return curr;
  769. }, []),
  770. };
  771. }
  772. // 柱状图,圆柱形样式
  773. if (type === 'bar_cylinder_wide') {
  774. return {
  775. textStyle,
  776. grid: {
  777. top: 40,
  778. bottom: 50,
  779. },
  780. legend: {
  781. textStyle,
  782. show: legend.show,
  783. },
  784. tooltip: {
  785. trigger: 'item',
  786. },
  787. xAxis: xAxis.map((e) => {
  788. return {
  789. ...e,
  790. type: 'category',
  791. axisLabel: {
  792. interval: 0,
  793. width: (domWidth - 100) / get(baseSeries, '[0].data.length', 1),
  794. overflow: 'break',
  795. },
  796. };
  797. }),
  798. yAxis: yAxis.map((e, i) => {
  799. return {
  800. ...e,
  801. splitLine: {
  802. lineStyle: {
  803. opacity: i === 0 ? 0.1 : 0,
  804. },
  805. },
  806. };
  807. }),
  808. series: baseSeries.reduce((curr: EChartsOption[], serie, index) => {
  809. // const colors = ['#66ffff', '#00ff66', '#ffff66'];
  810. const colors = ['#73C0DE', '#ED6666', '#5470C6', '#91CC75', '#FAC858', '#17d1b2', '#2ae271', '#11bce7', '#c127f0', '#ee125b'];
  811. if (baseSeries.length === 1) {
  812. curr.push({
  813. ...serie,
  814. type: 'pictorialBar',
  815. symbol: 'circle',
  816. symbolPosition: 'end',
  817. symbolSize: [50, 10],
  818. symbolOffset: [0, -5],
  819. barGap: '-100%',
  820. yAxisIndex: index,
  821. itemStyle: {
  822. color: ({ dataIndex }) => colors[dataIndex % colors.length],
  823. },
  824. });
  825. curr.push({
  826. ...serie,
  827. type: 'pictorialBar',
  828. symbol: 'circle',
  829. symbolPosition: 'start',
  830. symbolSize: [50, 10],
  831. symbolOffset: [0, 5],
  832. barGap: '-100%',
  833. yAxisIndex: index,
  834. itemStyle: {
  835. color: ({ dataIndex }) => colors[dataIndex % colors.length],
  836. },
  837. });
  838. }
  839. curr.push({
  840. ...serie,
  841. type: 'bar',
  842. // silent: true,
  843. yAxisIndex: index,
  844. barWidth: 50,
  845. barGap: '100%',
  846. itemStyle: {
  847. color: ({ dataIndex }) =>
  848. new graphic.LinearGradient(0, 0, 0, 1, [
  849. { offset: 0, color: `${colors[dataIndex % colors.length]}44` },
  850. { offset: 0.5, color: colors[dataIndex % colors.length] },
  851. { offset: 1, color: colors[dataIndex % colors.length] },
  852. ]),
  853. borderRadius: [5, 5, 0, 0],
  854. },
  855. label: {
  856. show: true, //开启显示
  857. position: 'top', //在上方显示
  858. textStyle: {
  859. //数值样式
  860. color: '#ffffffbb',
  861. fontSize: 13,
  862. },
  863. formatter: function (obj) {
  864. return obj['data'][1];
  865. },
  866. },
  867. });
  868. return curr;
  869. }, []),
  870. };
  871. }
  872. return {};
  873. };
  874. watch(
  875. () => props.chartData,
  876. () => {
  877. initCharts();
  878. },
  879. {
  880. immediate: true,
  881. }
  882. );
  883. function initCharts() {
  884. const o = genChartOption();
  885. setOptions(o as EChartsOption, false);
  886. }
  887. </script>