radar-chart.vue 12 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467
  1. <template>
  2. <div class="chart" :id="id"></div>
  3. </template>
  4. <script>
  5. import util from "@/helper/util.js";
  6. import partten from "@/helper/partten.js";
  7. import * as echarts from "echarts";
  8. export default {
  9. name: "radar-chart",
  10. componentName: "radar-chart",
  11. props: {
  12. // 宽度 默认9.722vh
  13. width: {
  14. type: String,
  15. default: "100%",
  16. },
  17. // 高度 默认9.722vh
  18. height: {
  19. type: String,
  20. default: "7.4074vh",
  21. },
  22. // 标题
  23. title: {
  24. type: String,
  25. default: "",
  26. },
  27. // 值
  28. value: {
  29. type: Object,
  30. default: () => {
  31. return {
  32. indicator: [
  33. "北(0.0/0.0)",
  34. "北北西(0.0/0.0)",
  35. "北西(0.0/0.0)",
  36. "西北西(0.0/0.0)",
  37. "西(0.0/0.0)",
  38. "西南西(0.0/0.0)",
  39. "南西(0.0/0.0)",
  40. "南南西(0.0/0.0)",
  41. "南(0.0/0.0)",
  42. "南南东(0.0/0.0)",
  43. "东南(0.0/0.0)",
  44. "东南东(0.0/0.0)",
  45. "东(0.0/0.0)",
  46. "东北东(0.0/0.0)",
  47. "北东(0.0/0.0)",
  48. "北北东(0.0/0.0)",
  49. ],
  50. data: [
  51. {
  52. value: [
  53. 44200, 14200, 20000, 35000, 50000, 38000, 44200, 14200, 20000,
  54. 35000, 50000, 38000, 44200, 14200, 20000, 35000, 50000, 38000,
  55. 20000, 35000, 50000, 38000,
  56. ],
  57. name: "NAME",
  58. },
  59. ],
  60. };
  61. },
  62. },
  63. },
  64. data() {
  65. return {
  66. id: "",
  67. chart: null,
  68. lineStyles: [],
  69. green: {
  70. areaStyle: {
  71. color: "rgba(165,228,175, 0.9)",
  72. },
  73. lineStyle: {
  74. color: "rgba(255,255,255, 0.85)",
  75. },
  76. itemStyle: {
  77. color: "rgba(165,228,175, 0.5)",
  78. borderColor: "rgba(255,255,255, 0.5)",
  79. borderWidth: 0.5,
  80. },
  81. },
  82. blue: {
  83. areaStyle: {
  84. color: "rgba(75,85,174, 0.9)",
  85. },
  86. lineStyle: {
  87. color: "rgba(255,255,255, 0.85)",
  88. },
  89. itemStyle: {
  90. color: "rgba(75,85,174, 0.9)",
  91. borderColor: "rgba(255,255,255, 0.5)",
  92. borderWidth: 0.5,
  93. },
  94. },
  95. };
  96. },
  97. computed: {},
  98. methods: {
  99. initChart() {
  100. let themeName = "";
  101. let theme = this.$store.state.themeName;
  102. if (theme == "dark" || theme == "light") {
  103. themeName = theme;
  104. } else {
  105. themeName = theme.split(" ")[1];
  106. }
  107. let chart = echarts.init(this.$el);
  108. let maxValue = -1;
  109. if (themeName === "dark") {
  110. this.lineStyles = [this.green, this.blue];
  111. } else {
  112. this.lineStyles = [this.blue, this.green];
  113. }
  114. if (this.value.data)
  115. this.value.data.forEach((item, index) => {
  116. item.value.forEach((value) => {
  117. if (value > maxValue) {
  118. maxValue = value;
  119. }
  120. });
  121. item.areaStyle =
  122. this.lineStyles[index % this.lineStyles.length].areaStyle;
  123. item.lineStyle =
  124. this.lineStyles[index % this.lineStyles.length].lineStyle;
  125. item.itemStyle =
  126. this.lineStyles[index % this.lineStyles.length].itemStyle;
  127. });
  128. maxValue *= 1.5;
  129. let indicator = [];
  130. if (this.value.indicator)
  131. this.value.indicator.forEach((item) => {
  132. indicator.push({ name: item, max: maxValue });
  133. });
  134. let baseSize = 65;
  135. let option = {
  136. grid: {
  137. left: 0,
  138. right: 0,
  139. bottom: 0,
  140. top: 0,
  141. },
  142. tooltip: {
  143. trigger: "item",
  144. backgroundColor:
  145. this.$store.state.themeName === "dark"
  146. ? "rgba(0,0,0,0.4)"
  147. : "rgba(255,255,255,0.5)",
  148. borderColor:
  149. this.$store.state.themeName === "dark"
  150. ? partten.getColor("gray")
  151. : "#000",
  152. textStyle: {
  153. color: this.$store.state.themeName === "dark" ? "#fff" : "#000",
  154. fontSize: util.vh(16),
  155. },
  156. position: function (pos, params, dom, rect, size) {
  157. // 鼠标在左侧时 tooltip 显示到右侧,鼠标在右侧时 tooltip 显示到左侧。
  158. var obj = { top: 60 };
  159. obj[["left", "right"][+(pos[0] < size.viewSize[0] / 2)]] = 5;
  160. return obj;
  161. },
  162. extraCssText: "position: absolute; margin-top:-20%;",
  163. // extraCssText: "max-width:140px;position:sticky;top:0;left:0"
  164. },
  165. radar: [
  166. // 最低层 90
  167. {
  168. radius: baseSize + "%",
  169. center: ["50%", "50%"],
  170. splitNumber: 1,
  171. nameGap: "4",
  172. name: {
  173. textStyle: {
  174. color:
  175. this.$store.state.themeName === "dark"
  176. ? partten.getColor("gray")
  177. : "#000",
  178. fontSize: 12,
  179. padding: [0, 16],
  180. },
  181. },
  182. axisLine: {
  183. lineStyle: {
  184. color:
  185. this.$store.state.themeName === "dark"
  186. ? partten.getColor("gray") + 40
  187. : "#000" + 40,
  188. },
  189. },
  190. splitLine: {
  191. lineStyle: {
  192. width: 1,
  193. color:
  194. this.$store.state.themeName === "dark"
  195. ? partten.getColor("gray") + 40
  196. : "#000" + 40,
  197. },
  198. },
  199. splitArea: {
  200. areaStyle: {
  201. color: "transparent",
  202. },
  203. },
  204. indicator: indicator,
  205. },
  206. // 次外层 80 - 90
  207. {
  208. radius: ["55%", "65%"],
  209. center: ["50%", "50%"],
  210. startAngle: 90,
  211. splitNumber: 2,
  212. name: {
  213. show: false,
  214. },
  215. axisLine: {
  216. lineStyle: {
  217. color:
  218. this.$store.state.themeName === "dark"
  219. ? partten.getColor("gray") + 40
  220. : "#000" + 40,
  221. shadowBlur: 1,
  222. shadowColor: "#fff",
  223. shadowOffsetX: 0.5,
  224. shadowOffsetY: 1,
  225. },
  226. },
  227. splitLine: {
  228. lineStyle: {
  229. width: 1,
  230. color:
  231. this.$store.state.themeName === "dark"
  232. ? partten.getColor("gray") + 40
  233. : "#000" + 40,
  234. shadowColor: "#fff",
  235. shadowBlur: 0,
  236. shadowOffsetX: 0.5,
  237. shadowOffsetY: 0.5,
  238. },
  239. },
  240. splitArea: {
  241. areaStyle: {
  242. color: "transparent",
  243. },
  244. },
  245. indicator: indicator,
  246. },
  247. // 渐变层 40 - 80
  248. {
  249. radius: ["30%", "55%"],
  250. center: ["50%", "50%"],
  251. splitNumber: 1,
  252. name: {
  253. show: false,
  254. },
  255. axisLine: {
  256. lineStyle: {
  257. color:
  258. this.$store.state.themeName === "dark"
  259. ? partten.getColor("gray") + 40
  260. : "#000" + 40,
  261. },
  262. },
  263. splitLine: {
  264. lineStyle: {
  265. width: 1,
  266. color:
  267. this.$store.state.themeName === "dark"
  268. ? partten.getColor("gray")
  269. : "#000",
  270. },
  271. },
  272. splitArea: {
  273. areaStyle: {
  274. shadowBlur: 4,
  275. color: {
  276. type: "radial",
  277. x: 0.5,
  278. y: 0.5,
  279. r: 0.5,
  280. colorStops: [
  281. {
  282. offset: 0.5,
  283. color: "transparent", // 0% 处的颜色
  284. },
  285. {
  286. offset: 1,
  287. color:
  288. this.$store.state.themeName === "dark"
  289. ? partten.getColor("green") + 60
  290. : partten.getColor("deepblue") + 60, // 100% 处的颜色
  291. },
  292. ],
  293. global: false, // 缺省为 false
  294. },
  295. },
  296. },
  297. indicator: indicator,
  298. },
  299. // 渐变层 0 - 40
  300. {
  301. radius: ["0%", "30%"],
  302. center: ["50%", "50%"],
  303. splitNumber: 1,
  304. name: {
  305. show: false,
  306. },
  307. axisLine: {
  308. lineStyle: {
  309. color:
  310. this.$store.state.themeName === "dark"
  311. ? partten.getColor("gray") + 40
  312. : "#000" + 40,
  313. },
  314. },
  315. splitLine: {
  316. lineStyle: {
  317. width: 1,
  318. color:
  319. this.$store.state.themeName === "dark"
  320. ? partten.getColor("gray")
  321. : "#000",
  322. },
  323. },
  324. splitArea: {
  325. areaStyle: {
  326. shadowBlur: 4,
  327. color: {
  328. type: "radial",
  329. x: 0.5,
  330. y: 0.5,
  331. r: 0.5,
  332. colorStops: [
  333. {
  334. offset: 0.5,
  335. color: "transparent", // 0% 处的颜色
  336. },
  337. {
  338. offset: 1,
  339. color:
  340. this.$store.state.themeName === "dark"
  341. ? partten.getColor("green") + 60
  342. : partten.getColor("deepblue") + 60, // 100% 处的颜色
  343. },
  344. ],
  345. global: false, // 缺省为 false
  346. },
  347. },
  348. },
  349. indicator: indicator,
  350. },
  351. // 内层 0 - 50
  352. {
  353. radius: "50%",
  354. center: ["50%", "50%"],
  355. splitNumber: 1,
  356. name: {
  357. show: false,
  358. },
  359. axisLine: {
  360. lineStyle: {
  361. color:
  362. this.$store.state.themeName === "dark"
  363. ? partten.getColor("gray") + 40
  364. : "#000" + 40,
  365. },
  366. },
  367. splitLine: {
  368. lineStyle: {
  369. width: 1,
  370. color:
  371. this.$store.state.themeName === "dark"
  372. ? partten.getColor("gray")
  373. : "#000",
  374. },
  375. },
  376. splitArea: {
  377. areaStyle: {
  378. shadowBlur: 4,
  379. color: "transparent",
  380. },
  381. },
  382. indicator: indicator,
  383. },
  384. // 内层 0 - 45
  385. {
  386. radius: "35%",
  387. center: ["50%", "50%"],
  388. splitNumber: 1,
  389. name: {
  390. show: false,
  391. },
  392. axisLine: {
  393. lineStyle: {
  394. color:
  395. this.$store.state.themeName === "dark"
  396. ? partten.getColor("gray") + 40
  397. : "#000" + 40,
  398. },
  399. },
  400. splitLine: {
  401. lineStyle: {
  402. width: 1,
  403. color:
  404. this.$store.state.themeName === "dark"
  405. ? partten.getColor("gray")
  406. : "#000",
  407. },
  408. },
  409. splitArea: {
  410. areaStyle: {
  411. shadowBlur: 4,
  412. color: "transparent",
  413. },
  414. },
  415. indicator: indicator,
  416. },
  417. ],
  418. series: [
  419. {
  420. name: this.title,
  421. type: "radar",
  422. data: this.value.data,
  423. },
  424. ],
  425. };
  426. console.log(option);
  427. chart.setOption(option);
  428. },
  429. },
  430. created() {
  431. this.id = "pie-chart-" + util.newGUID();
  432. },
  433. mounted() {
  434. this.$nextTick(() => {
  435. this.$el.style.width = this.width;
  436. this.$el.style.height = this.height;
  437. this.initChart();
  438. });
  439. },
  440. updated() {
  441. this.$nextTick(() => {
  442. this.initChart();
  443. });
  444. },
  445. watch: {
  446. "$store.state.themeName"() {
  447. this.initChart();
  448. },
  449. },
  450. };
  451. </script>
  452. <style lang="less" scoped>
  453. .chart {
  454. width: 100%;
  455. height: 100%;
  456. display: block;
  457. margin: auto;
  458. position: relative;
  459. }
  460. </style>