sandianscore.vue 19 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596
  1. <template>
  2. <!-- 得分水平分布-企业水平分析 - littlegreen - 补充form和label -->
  3. <div
  4. :class="className"
  5. :style="{ height: '100%', width: width, padding: '20px' }"
  6. >
  7. <div class="select-container">
  8. <el-form
  9. style="
  10. display: grid;
  11. grid-template-columns: 1fr 1fr 1fr;
  12. align-items: center;
  13. grid-gap: 20px;
  14. "
  15. label-width="100px"
  16. >
  17. <el-form-item label="评估指标" style="margin-bottom: 0">
  18. <el-select v-model="selectedDataKey" style="width: 100%">
  19. <el-option
  20. v-for="key in dataKeys"
  21. :key="key"
  22. :value="key"
  23. :label="keyToChinese[key]"
  24. >
  25. {{ keyToChinese[key] }}
  26. </el-option>
  27. </el-select>
  28. </el-form-item>
  29. <el-form-item label="年度" style="margin-bottom: 0"
  30. ><el-select v-model="selectedYear" style="width: 100%">
  31. <el-option v-for="year in availableYears" :key="year" :value="year">
  32. {{ year }}
  33. </el-option>
  34. </el-select></el-form-item
  35. >
  36. <el-form-item label="行业代码" style="margin-bottom: 0"
  37. ><el-input
  38. v-model="selectedCode"
  39. placeholder="请输入行业代码"
  40. clearable
  41. style="width: 100%"
  42. /></el-form-item>
  43. <el-form-item label="检测区间" style="margin-bottom: 0">
  44. <div style="display: flex">
  45. <el-input v-model="detectMin" placeholder="最小值" clearable />
  46. <span style="margin: 0 10px">~</span>
  47. <el-input v-model="detectMax" placeholder="最大值" clearable />
  48. </div>
  49. </el-form-item>
  50. <!-- <el-form-item label="数据分布区间" style="margin-bottom:0"
  51. ><el-input
  52. v-model="selectedRange"
  53. placeholder="请设置数据分布区间"
  54. clearable
  55. style="width: 240px"
  56. /></el-form-item> -->
  57. <div style="display: inline; padding-left: 10px">
  58. 当前行业: {{ industryMap.get(selectedCode) }}
  59. </div>
  60. <el-form-item>
  61. <el-button
  62. type="primary"
  63. icon="el-icon-search"
  64. @click="updateChart"
  65. >搜索</el-button>
  66. </el-form-item>
  67. </el-form>
  68. </div>
  69. <div ref="chart" :style="{ height: height, width: width }"></div>
  70. <el-table :data="tableData" border style="width: 100%">
  71. <el-table-column
  72. prop="industryName"
  73. label="行业名称"
  74. width="200"
  75. ></el-table-column>
  76. <el-table-column prop="year" label="年度" width="200"> </el-table-column>
  77. <el-table-column prop="enterpriseId" label="企业ID"></el-table-column>
  78. <el-table-column
  79. prop="value"
  80. label="Z-score得分"
  81. width="200"
  82. ></el-table-column>
  83. <el-table-column
  84. prop="level"
  85. label="水平评价"
  86. width="200"
  87. ></el-table-column>
  88. </el-table>
  89. </div>
  90. </template>
  91. <script>
  92. import * as echarts from "echarts";
  93. import { listScore } from "@/api/score/score"; // 导入得分数据的接口
  94. import { listIndustry } from "@/api/industry/industry"; // 导入行业数据的接口
  95. require("echarts/theme/macarons"); // echarts theme
  96. import resize from "./mixins/resize";
  97. export default {
  98. mixins: [resize],
  99. props: {
  100. className: {
  101. type: String,
  102. default: "chart",
  103. },
  104. width: {
  105. type: String,
  106. default: "100%",
  107. },
  108. height: {
  109. type: String,
  110. default: "400px",
  111. },
  112. },
  113. data() {
  114. return {
  115. chart: null,
  116. chartData: [],
  117. industryData: [],
  118. industryMap: new Map(),
  119. selectedDataKey: "totalIndustrialValueScore", // 默认选择的字段
  120. selectedCode: null,
  121. dataKeys: [
  122. // 'mainBusinessScore',
  123. // 'landAreaScore',
  124. "totalIndustrialValueScore",
  125. // 'gdpScore',
  126. "taxableIncomeScore",
  127. "paidTaxScore",
  128. // 'mainBusinessIncomeScore',
  129. // 'employeeNumberScore',
  130. // 'profitScore',
  131. // 'ownerEquityScore',
  132. "fundingScore",
  133. "energyConsumeScore",
  134. "powerConsumeScore",
  135. // 其他得分字段...
  136. ], // 可选项
  137. keyToChinese: {
  138. mainBusinessScore: "主营业务活动得分",
  139. landAreaScore: "用地面积得分",
  140. totalIndustrialValueScore: "工业总产值得分",
  141. gdpScore: "工业增加值得分",
  142. taxableIncomeScore: "应税收入得分",
  143. paidTaxScore: "实缴税金得分",
  144. mainBusinessIncomeScore: "主营业务收入得分",
  145. employeeNumberScore: "从业人员数得分",
  146. profitScore: "利润总额得分",
  147. ownerEquityScore: "所有者权益得分",
  148. fundingScore: "研发经费得分",
  149. energyConsumeScore: "能源消费量得分",
  150. powerConsumeScore: "电力消费量得分",
  151. },
  152. selectedYear: null,
  153. queryParams: {
  154. pageNum: 1,
  155. pageSize: 2000000, // 默认 2000000 条(全部一次性数据)
  156. enterpriseName: null,
  157. location: null,
  158. code: null,
  159. mainBusiness: null,
  160. landArea: null,
  161. totalIndustrialValue: null,
  162. gdp: null,
  163. taxableIncome: null,
  164. paidTax: null,
  165. mainBusinessIncome: null,
  166. employeeNumber: null,
  167. profit: null,
  168. ownerEquity: null,
  169. funding: null,
  170. energyConsume: null,
  171. year: null,
  172. month: null,
  173. },
  174. industryQueryParams: {
  175. pageNum: 1,
  176. pageSize: 2000000,
  177. industryName: null,
  178. code: null,
  179. }, // 查询行业信息
  180. availableYears: [],
  181. selectedRange: 0.1, // 不能为 0, - littlegreen - 固定0.1
  182. // littlegreen - 检测区间 和 表格数据
  183. detectMin: null,
  184. detectMax: null,
  185. tableData: [],
  186. };
  187. },
  188. mounted() {
  189. this.$nextTick(() => {
  190. this.fetchData();
  191. });
  192. },
  193. beforeDestroy() {
  194. if (this.chart) {
  195. this.chart.dispose();
  196. }
  197. },
  198. methods: {
  199. fetchData() {
  200. listScore(this.queryParams)
  201. .then((response) => {
  202. this.chartData = response.rows;
  203. this.availableYears = [
  204. ...new Set(this.chartData.map((item) => item.year)),
  205. ];
  206. this.selectedYear = this.availableYears[0] || null;
  207. // this.selectedCode = [
  208. // ...new Set(this.x.map((item) => item.code)),
  209. // ][0];
  210. // littlegreen - 解决报错
  211. if (Array.isArray(this.chartData)) {
  212. this.selectedCode = [
  213. ...new Set(this.chartData.map((item) => item.code)),
  214. ][0];
  215. } else {
  216. console.warn("this.x is not an array or is undefined.");
  217. this.selectedCode = null; // 或根据需求设置一个默认值
  218. }
  219. this.initChart();
  220. })
  221. .catch((error) => {
  222. console.error("Error fetching data:", error);
  223. });
  224. listIndustry(this.industryQueryParams)
  225. .then((response) => {
  226. this.industryData = response.rows;
  227. // console.log("industry_info" + this.industryData);
  228. this.industryMap = this.industryData.reduce((map, item) => {
  229. const key = `${item.code}`; // 键是行业代码
  230. map.set(key, item.industryName); // 将键和对应的行业名称存储到Map中
  231. return map;
  232. }, new Map());
  233. })
  234. .catch((error) => {
  235. console.error("Error fetching data:", error);
  236. });
  237. },
  238. initChart() {
  239. if (this.chart) {
  240. this.chart.dispose();
  241. }
  242. this.chart = echarts.init(this.$refs.chart, "macarons");
  243. this.updateChart();
  244. },
  245. updateChart() {
  246. const that = this;
  247. if (!this.chart) {
  248. return;
  249. }
  250. let flag = this.checkNumber();
  251. if (flag == 2) {
  252. return;
  253. }
  254. // console.log(flag);
  255. let filteredData = this.chartData;
  256. if (this.selectedYear) {
  257. filteredData = filteredData.filter(
  258. (item) =>
  259. item.year === this.selectedYear && item.code === this.selectedCode
  260. );
  261. }
  262. const values = filteredData.map((item) => item[this.selectedDataKey]);
  263. const minVal = Math.min(...values).toFixed(2);
  264. const maxVal = Math.max(...values).toFixed(2);
  265. console.log(filteredData, "filteredData");
  266. if (
  267. this.selectedRange === null ||
  268. this.selectedRange === "" ||
  269. isNaN(+this.selectedRange)
  270. )
  271. this.selectedRange = "0.1";
  272. else if (Number(this.selectedRange) <= 0) this.selectedRange = "0.1";
  273. const binCount = Math.ceil(
  274. (maxVal - minVal) / Number(this.selectedRange)
  275. );
  276. let maxNum = -1;
  277. // littlegreen - 传入最大值、最小值、间隔,获得区间列表
  278. function generateRangeArray(x1, x2, interval) {
  279. x1 = parseFloat(x1);
  280. x2 = parseFloat(x2);
  281. interval = parseFloat(interval);
  282. const result = [];
  283. let current = x1;
  284. while (current < x2) {
  285. let next;
  286. if (current == x1) {
  287. current = Math.floor(current * 10) / 10;
  288. }
  289. next = parseFloat((current + interval).toFixed(2));
  290. // 保留两位小数
  291. result.push({
  292. name: `${current}-${next}`,
  293. min: current,
  294. max: next,
  295. });
  296. current = next;
  297. }
  298. // 最后一个区间
  299. if (current <= x2) {
  300. result.push({
  301. name: `${current}-${x2}`,
  302. min: current,
  303. max: x2,
  304. });
  305. }
  306. return result;
  307. }
  308. // 生成区间标签和频次
  309. const histogramData = [];
  310. let result = generateRangeArray(minVal, maxVal, 0.1);
  311. // littlegreen - 获取不同区间的数据
  312. result.forEach((item) => {
  313. let count = 0;
  314. count += values.filter(
  315. (value) => value >= item.min && value < item.max
  316. ).length;
  317. if (item.max === maxVal) {
  318. count += values.filter((value) => value === item.max).length;
  319. }
  320. histogramData.push({ name: item.name, value: count });
  321. });
  322. // for (let i = 0; i < binCount; i++) {
  323. // const x0 = minVal + i * Number(this.selectedRange);
  324. // const x1 = Math.min(x0 + Number(this.selectedRange), maxVal).toFixed(2);
  325. // // const label = `${x0}-${x1}`;
  326. // // let count = 0;
  327. // // count += values.filter((value) => value >= x0 && value < x1).length;
  328. // // if (x1 === maxVal) {
  329. // // count += values.filter((value) => value === maxVal).length;
  330. // // }
  331. // // if (count > 0) {
  332. // // if (count > maxNum) {
  333. // // maxNum = count;
  334. // // }
  335. // // histogramData.push({ name: label, value: count });
  336. // // }
  337. // }
  338. const maxHistogramValue = Math.max(
  339. ...histogramData.map((item) => item.value)
  340. );
  341. const totalHistogramValue = histogramData.reduce(
  342. (sum, item) => sum + item.value,
  343. 0
  344. );
  345. // const quadraticData = this.generateQuadraticData(binCount, maxHistogramValue, totalHistogramValue);
  346. // littlegreen - 去掉line和scatter,增加x轴拖动,根据检测区间修改柱状的颜色
  347. const option = {
  348. color: ["rgba(245,0,0,0.6)"],
  349. dataZoom: [
  350. {
  351. type: "inside", // 内置于坐标系中
  352. start: 0,
  353. end: 100,
  354. xAxisIndex: [0],
  355. },
  356. ],
  357. tooltip: {
  358. trigger: "axis",
  359. axisPointer: {
  360. type: "shadow",
  361. },
  362. formatter: (params) => {
  363. // 使用箭头函数来保持 this 的上下文
  364. const seriesName = params[0].name; // 系列名称
  365. const dataIndex = params[0].dataIndex; // 数据索引
  366. const value = params[0].data; // 实际的数据值
  367. const xAxisLabel = params[0].axisValueLabel; // x轴的标签
  368. const yAxisLabel = params[0].seriesName;
  369. // 构建自定义的 tooltip 内容,包含年份
  370. return `${xAxisLabel}<br> ${yAxisLabel}: ${value}`;
  371. },
  372. },
  373. legend: {
  374. data: [
  375. `${this.selectedYear}年 ${
  376. this.keyToChinese[this.selectedDataKey]
  377. }企业数统计`,
  378. ],
  379. },
  380. grid: {
  381. left: "3%",
  382. right: "4%",
  383. bottom: "50px",
  384. containLabel: true,
  385. },
  386. xAxis: {
  387. type: "category",
  388. name: "分布区间",
  389. data: histogramData.map((item) => item.name),
  390. axisLabel: {
  391. fontSize: 12,
  392. interval: 0,
  393. rotate: 20,
  394. },
  395. },
  396. yAxis: {
  397. name: "企业数",
  398. type: "value",
  399. minInterval: 1,
  400. },
  401. series: [
  402. {
  403. name: `${this.selectedYear}年 ${
  404. this.keyToChinese[this.selectedDataKey]
  405. }企业数统计`,
  406. type: "bar",
  407. barWidth: "30%",
  408. itemStyle: {
  409. // color: "rgba(245,0,0,0.6)",
  410. normal: {
  411. color: function (params) {
  412. const value = params.dataIndex; // 获取当前柱子在数据中的索引
  413. if (flag == 1) {
  414. if (
  415. result[value].min >= that.detectMin &&
  416. result[value].max <= that.detectMax
  417. ) {
  418. return "#ffcc00"; // 变更颜色为黄色
  419. }
  420. return "rgba(245,0,0,0.6)";
  421. }
  422. return "rgba(245,0,0,0.6)";
  423. // 默认颜色
  424. },
  425. },
  426. },
  427. data: histogramData.map((item) => item.value),
  428. label: {
  429. show: false, // 将show属性设置为false,去掉柱子上的数字
  430. },
  431. },
  432. // {
  433. // type: "scatter",
  434. // symbol: "circle",
  435. // symbolSize: 10,
  436. // data: histogramData.map((item, index) => [index, item.value]),
  437. // itemStyle: {
  438. // color: "red",
  439. // },
  440. // },
  441. // {
  442. // type: "line",
  443. // data: histogramData.map((item, index) => [index, item.value]),
  444. // lineStyle: {
  445. // color: "#1a7cc8",
  446. // },
  447. // symbol: "none",
  448. // },
  449. ],
  450. };
  451. this.isChartShow = true;
  452. this.chart.setOption(option);
  453. this.chart.on("click", function (param) {
  454. if (
  455. typeof param.dataIndex === "number" &&
  456. param.dataIndex >= 0 &&
  457. param.dataIndex < result.length
  458. ) {
  459. let dataIndexValue = result[param.dataIndex];
  460. const filteredArray = filteredData.filter((item) => {
  461. const isMaxValEqual = result[1] === maxVal;
  462. const lowerBoundCheck =
  463. item[that.selectedDataKey] >= dataIndexValue.min;
  464. const upperBoundCheck = isMaxValEqual
  465. ? item[that.selectedDataKey] <= dataIndexValue.max
  466. : item[that.selectedDataKey] < dataIndexValue.max;
  467. return lowerBoundCheck && upperBoundCheck;
  468. });
  469. that.tableData = filteredArray.map((item) => {
  470. return {
  471. industryName: item.code,
  472. enterpriseId: item.enterpriseName,
  473. year: item.year,
  474. value: item[that.selectedDataKey],
  475. level:"高"
  476. };
  477. });
  478. }
  479. // const filteredArray = filteredData.filter((item) => {
  480. // const isMaxValEqual = result[1] === maxVal;
  481. // const lowerBoundCheck = item[that.selectedDataKey] >= dataIndexValue.min;
  482. // const upperBoundCheck = isMaxValEqual
  483. // ? item[that.selectedDataKey] <= dataIndexValue.max
  484. // : item[that.selectedDataKey] < dataIndexValue.max;
  485. // return lowerBoundCheck && upperBoundCheck;
  486. // })
  487. // that.tableData = filteredArray.map(item => {
  488. // return {
  489. // industryName: item.code,
  490. // enterpriseId: item.enterpriseName,
  491. // year: item.year,
  492. // value: item[that.selectedDataKey]
  493. // }
  494. // })
  495. });
  496. },
  497. generateQuadraticData(binCount, maxHistogramValue, totalHistogramValue) {
  498. const a = -0.01;
  499. const h = binCount / 2;
  500. const k = maxHistogramValue * 0.75;
  501. const sideValue = (maxHistogramValue * 0.15) / 2;
  502. const data = [];
  503. for (let i = 0; i < binCount; i++) {
  504. const x = i;
  505. const y = a * (x - h) ** 2 + k;
  506. // 调整两侧的值
  507. if (i === 0 || i === binCount - 1) {
  508. data.push([x, sideValue]);
  509. } else {
  510. data.push([x, y]);
  511. }
  512. }
  513. return data;
  514. },
  515. // littlegreen - 检测检测区间是否填写正确
  516. checkNumber() {
  517. const maxInput = parseFloat(this.detectMax);
  518. const minInput = parseFloat(this.detectMin);
  519. let flag;
  520. if (!this.detectMin && !this.detectMax) {
  521. return (flag = 0);
  522. }
  523. if (isNaN(maxInput) || isNaN(minInput)) {
  524. this.$message({
  525. message: "请输入有效的数字。",
  526. type: "error",
  527. });
  528. return (flag = 2);
  529. }
  530. // 检查 minInput 是否大于 maxInput
  531. if (minInput >= maxInput) {
  532. this.$message({
  533. message: "监测区间最大值不可小于或等于最小值",
  534. type: "error",
  535. });
  536. return (flag = 2);
  537. }
  538. return (flag = 1);
  539. },
  540. },
  541. watch: {
  542. // selectedDataKey() {
  543. // this.updateChart();
  544. // },
  545. // selectedYear() {
  546. // this.updateChart();
  547. // },
  548. // selectedRange() {
  549. // setTimeout(this.updateChart, 2500);
  550. // },
  551. // selectedCode() {
  552. // this.updateChart();
  553. // },
  554. },
  555. };
  556. </script>
  557. <style scoped>
  558. .select-container {
  559. margin: 10px 0;
  560. font-size: 16px; /* 调整字号大小 */
  561. }
  562. .select-container select {
  563. padding: 10px 20px; /* 增加内边距 */
  564. margin-right: 10px; /* 增加右边距 */
  565. border: 1px solid #121315; /* 蓝色边框 */
  566. border-radius: 4px; /* 圆角边框 */
  567. background-color: white; /* 背景色 */
  568. color: #121315; /* 文字颜色 */
  569. font-size: 16px; /* 调整字号大小 */
  570. cursor: pointer; /* 鼠标悬停时的指针样式 */
  571. outline: none; /* 移除焦点时的轮廓 */
  572. }
  573. .select-container select:hover {
  574. border-color: #1a7cc8; /* 鼠标悬停时的边框颜色 */
  575. }
  576. .select-container select:focus {
  577. border-color: #121315; /* 焦点时的边框颜色 */
  578. box-shadow: 0 0 0 2px rgba(64, 159, 255, 0.2); /* 焦点时的阴影效果 */
  579. }
  580. </style>