sandianscore.vue 24 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734
  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 type="primary" icon="el-icon-search" @click="updateChart"
  62. >搜索</el-button
  63. >
  64. </el-form-item>
  65. </el-form>
  66. </div>
  67. <div ref="chart" :style="{ height: height, width: width }"></div>
  68. <el-table :data="tableData" border style="width: 100%">
  69. <el-table-column
  70. prop="industryName"
  71. label="行业名称"
  72. width="200"
  73. ></el-table-column>
  74. <el-table-column prop="year" label="年度" width="200"> </el-table-column>
  75. <el-table-column prop="enterpriseId" label="企业ID"></el-table-column>
  76. <el-table-column
  77. prop="value"
  78. label="Z-score得分"
  79. width="200"
  80. ></el-table-column>
  81. <el-table-column
  82. prop="level"
  83. label="水平评价"
  84. width="200"
  85. ></el-table-column>
  86. </el-table>
  87. </div>
  88. </template>
  89. <script>
  90. import * as echarts from "echarts";
  91. import { listScore } from "@/api/score/score"; // 导入得分数据的接口
  92. import { listIndustry } from "@/api/industry/industry"; // 导入行业数据的接口
  93. require("echarts/theme/macarons"); // echarts theme
  94. import resize from "./mixins/resize";
  95. export default {
  96. mixins: [resize],
  97. props: {
  98. className: {
  99. type: String,
  100. default: "chart",
  101. },
  102. width: {
  103. type: String,
  104. default: "100%",
  105. },
  106. height: {
  107. type: String,
  108. default: "400px",
  109. },
  110. },
  111. data() {
  112. return {
  113. chart: null,
  114. chartData: [],
  115. industryData: [],
  116. industryMap: new Map(),
  117. selectedDataKey: "totalIndustrialValueScore", // 默认选择的字段
  118. selectedCode: null,
  119. dataKeys: [
  120. // 'mainBusinessScore',
  121. // 'landAreaScore',
  122. "totalIndustrialValueScore",
  123. // 'gdpScore',
  124. "taxableIncomeScore",
  125. "paidTaxScore",
  126. // 'mainBusinessIncomeScore',
  127. // 'employeeNumberScore',
  128. // 'profitScore',
  129. // 'ownerEquityScore',
  130. "fundingScore",
  131. "energyConsumeScore",
  132. "powerConsumeScore",
  133. // 其他得分字段...
  134. ], // 可选项
  135. keyToChinese: {
  136. mainBusinessScore: "主营业务活动得分",
  137. landAreaScore: "用地面积得分",
  138. totalIndustrialValueScore: "工业总产值得分",
  139. gdpScore: "工业增加值得分",
  140. taxableIncomeScore: "应税收入得分",
  141. paidTaxScore: "实缴税金得分",
  142. mainBusinessIncomeScore: "主营业务收入得分",
  143. employeeNumberScore: "从业人员数得分",
  144. profitScore: "利润总额得分",
  145. ownerEquityScore: "所有者权益得分",
  146. fundingScore: "研发经费得分",
  147. energyConsumeScore: "能源消费量得分",
  148. powerConsumeScore: "电力消费量得分",
  149. },
  150. keyToLevel: {
  151. totalIndustrialValueScore: "totalIndustrialValueLevel",
  152. taxableIncomeScore: "taxableIncomeLevel",
  153. paidTaxScore: "paidTaxLevel",
  154. fundingScore: "fundingLevel",
  155. energyConsumeScore: "energyConsumeLevel",
  156. powerConsumeScore: "powerConsumeLevel",
  157. },
  158. selectedYear: null,
  159. queryParams: {
  160. pageNum: 1,
  161. pageSize: 2000000, // 默认 2000000 条(全部一次性数据)
  162. enterpriseName: null,
  163. location: null,
  164. code: null,
  165. mainBusiness: null,
  166. landArea: null,
  167. totalIndustrialValue: null,
  168. gdp: null,
  169. taxableIncome: null,
  170. paidTax: null,
  171. mainBusinessIncome: null,
  172. employeeNumber: null,
  173. profit: null,
  174. ownerEquity: null,
  175. funding: null,
  176. energyConsume: null,
  177. year: null,
  178. month: null,
  179. },
  180. industryQueryParams: {
  181. pageNum: 1,
  182. pageSize: 2000000,
  183. industryName: null,
  184. code: null,
  185. }, // 查询行业信息
  186. availableYears: [],
  187. selectedRange: 0.1, // 不能为 0, - littlegreen - 固定0.1
  188. // littlegreen - 检测区间 和 表格数据
  189. detectMin: null,
  190. detectMax: null,
  191. tableData: [],
  192. };
  193. },
  194. mounted() {
  195. this.$nextTick(() => {
  196. this.fetchData();
  197. });
  198. },
  199. beforeDestroy() {
  200. if (this.chart) {
  201. this.chart.dispose();
  202. }
  203. },
  204. methods: {
  205. fetchData() {
  206. listScore(this.queryParams)
  207. .then((response) => {
  208. this.chartData = response.rows;
  209. this.availableYears = [
  210. ...new Set(this.chartData.map((item) => item.year)),
  211. ];
  212. this.selectedYear = this.availableYears[0] || null;
  213. // this.selectedCode = [
  214. // ...new Set(this.x.map((item) => item.code)),
  215. // ][0];
  216. // littlegreen - 解决报错
  217. if (Array.isArray(this.chartData)) {
  218. this.selectedCode = [
  219. ...new Set(this.chartData.map((item) => item.code)),
  220. ][0];
  221. } else {
  222. console.warn("this.x is not an array or is undefined.");
  223. this.selectedCode = null; // 或根据需求设置一个默认值
  224. }
  225. this.initChart();
  226. })
  227. .catch((error) => {
  228. console.error("Error fetching data:", error);
  229. });
  230. listIndustry(this.industryQueryParams)
  231. .then((response) => {
  232. this.industryData = response.rows;
  233. // console.log("industry_info" + this.industryData);
  234. this.industryMap = this.industryData.reduce((map, item) => {
  235. const key = `${item.code}`; // 键是行业代码
  236. map.set(key, item.industryName); // 将键和对应的行业名称存储到Map中
  237. return map;
  238. }, new Map());
  239. })
  240. .catch((error) => {
  241. console.error("Error fetching data:", error);
  242. });
  243. },
  244. initChart() {
  245. if (this.chart) {
  246. this.chart.dispose();
  247. }
  248. this.chart = echarts.init(this.$refs.chart, "macarons");
  249. this.updateChart();
  250. },
  251. updateChart() {
  252. const that = this;
  253. if (!this.chart) {
  254. return;
  255. }
  256. let flag = this.checkNumber();
  257. if (flag == 2) {
  258. return;
  259. }
  260. // console.log(flag);
  261. let filteredData = this.chartData;
  262. if (this.selectedYear) {
  263. filteredData = filteredData.filter(
  264. (item) =>
  265. item.year === this.selectedYear && item.code === this.selectedCode
  266. );
  267. }
  268. const values = filteredData.map((item) => item[this.selectedDataKey]);
  269. console.log(filteredData, values, "filteredData")
  270. const minVal = Math.min(...values).toFixed(2);
  271. const maxVal = Math.max(...values).toFixed(2);
  272. // console.log(minVal, maxVal)
  273. // console.log(
  274. // filteredData,
  275. // "filteredData",
  276. // this.selectedDataKey,
  277. // values,
  278. // "values"
  279. // );
  280. // console.log(filteredData, "filteredData");
  281. if (
  282. this.selectedRange === null ||
  283. this.selectedRange === "" ||
  284. isNaN(+this.selectedRange)
  285. )
  286. this.selectedRange = "0.1";
  287. else if (Number(this.selectedRange) <= 0) this.selectedRange = "0.1";
  288. const binCount = Math.ceil(
  289. (maxVal - minVal) / Number(this.selectedRange)
  290. );
  291. let maxNum = -1;
  292. // littlegreen - 传入最大值、最小值、间隔,获得区间列表
  293. function generateRangeArray(x1, x2, interval) {
  294. // 确保输入为浮点数
  295. x1 = parseFloat(x1);
  296. x2 = parseFloat(x2);
  297. interval = parseFloat(interval);
  298. const result = [];
  299. // 从最接近 x1 的更小的值开始
  300. let current = x1;
  301. // console.log(current)
  302. while (current < x2) {
  303. let next = parseFloat((current + interval).toFixed(2));
  304. // 保留两位小数,并添加区间到结果
  305. result.push({
  306. name: `${current.toFixed(2)}-${next.toFixed(2)}`,
  307. min: current,
  308. max: next,
  309. });
  310. current = next;
  311. }
  312. // 如果最后一个区间包含最大值,则加入
  313. if (current < x2) {
  314. result.push({
  315. name: `${current.toFixed(2)}-${x2.toFixed(2)}`,
  316. min: current,
  317. max: x2,
  318. });
  319. }
  320. return result;
  321. }
  322. // function generateRangeArray(x1, x2, interval) {
  323. // x1 = parseFloat(x1);
  324. // x2 = parseFloat(x2);
  325. // interval = parseFloat(interval);
  326. // const result = [];
  327. // let current = x1 - (x2 - x1);
  328. // while (current < x2) {
  329. // let next;
  330. // if (current == x1) {
  331. // current = Math.floor(current * 10) / 10;
  332. // }
  333. // next = parseFloat((current + interval).toFixed(2));
  334. // // 保留两位小数
  335. // result.push({
  336. // name: `${current}-${next}`,
  337. // min: current,
  338. // max: next,
  339. // });
  340. // current = next;
  341. // }
  342. // // 最后一个区间
  343. // if (current <= x2) {
  344. // result.push({
  345. // name: `${current}-${x2}`,
  346. // min: current,
  347. // max: x2,
  348. // });
  349. // }
  350. // return result;
  351. // }
  352. // 生成区间标签和频次
  353. const histogramData = [];
  354. let result = generateRangeArray(minVal, maxVal, 0.1);
  355. // littlegreen - 获取不同区间的数据
  356. result.forEach((item) => {
  357. let count = 0;
  358. count += values.filter(
  359. (value) => value >= item.min && value < item.max
  360. ).length;
  361. // console.log(item.max, maxVal)
  362. if (item.max == maxVal) {
  363. count += values.filter((value) => value === item.max).length;
  364. }
  365. histogramData.push({ name: item.name, value: count });
  366. });
  367. const maxHistogramValue = Math.max(
  368. ...histogramData.map((item) => item.value)
  369. );
  370. const totalHistogramValue = histogramData.reduce(
  371. (sum, item) => sum + item.value,
  372. 0
  373. );
  374. //--------------------------正态分布----------------------------------
  375. // 计算均值
  376. const mean = values.reduce((acc, val) => acc + val, 0) / values.length;
  377. // console.log(mean.toFixed(2),"mean")
  378. // 计算标准差
  379. const stdDev = Math.sqrt(
  380. values.reduce((acc, val) => acc + Math.pow(val - mean, 2), 0) /
  381. values.length
  382. );
  383. // 创建间隔为0.1的区间
  384. const center = mean.toFixed(2)
  385. const interval = 0.1;
  386. let minLimit = Math.floor(Math.min(...values) * 10) / 10; // 向下取整到最近的0.1
  387. let maxLimit = Math.ceil(Math.max(...values) * 10) / 10; // 向上取整到最近的0.1
  388. console.log("中心", center,minLimit,maxLimit)
  389. if(center-minLimit != maxLimit-center){
  390. minLimit = Math.floor(minLimit - (maxLimit-center - (center-minLimit)))
  391. }
  392. const intervals = [];
  393. console.log(minLimit,maxLimit, interval)
  394. // 乘以1000 是将小数变成整数去计算,小数计算会有误差
  395. for (let i = minLimit * 1000 ; i <= maxLimit*1000 ; i += (interval*1000)) {
  396. intervals.push(parseFloat((i/1000).toFixed(2)))
  397. }
  398. // 计算每个间隔的频次
  399. const frequency = new Array(intervals.length - 1).fill(0); // 初始化频次数组
  400. let intervalArr = intervals.slice(0,-1)
  401. const intervalStrings = [];
  402. for (let i = 0; i < intervalArr.length; i++) {
  403. const start = intervalArr[i].toFixed(1); // 保留一位小数
  404. const end = ((intervalArr[i]*1000 + interval*1000)/1000).toFixed(1); // 保留一位小数
  405. intervalStrings.push({
  406. name: `${start}-${end}`,
  407. max: end,
  408. min: start
  409. }); // 构建区间字符串
  410. }
  411. console.log(intervals,intervalArr)
  412. values.forEach((value) => {
  413. for (let j = 0; j < intervalStrings.length - 1; j++) {
  414. // console.log(value)
  415. // 保证包含右边界
  416. // console.log(intervals[j].min, intervals[j].max)
  417. if (value >= intervalStrings[j].min && value < intervalStrings[j].max) {
  418. // console.log(intervals[j],intervals[j + 1])
  419. frequency[j]++;
  420. break; // 找到对应区间后可以跳出循环
  421. }
  422. }
  423. });
  424. // 确保处理最大值等于maxLimit的情况
  425. if (values.some((value) => value == maxLimit)) {
  426. frequency[frequency.length - 1]++;
  427. }
  428. // 计算正态分布值
  429. const normalDistributionValues = intervals
  430. .slice(0, -1)
  431. .map((intervalStart, index) => {
  432. const intervalEnd = intervals[index + 1];
  433. const midpoint = (intervalStart + intervalEnd) / 2; // 每个区间的中点
  434. return (
  435. (1 / (stdDev * Math.sqrt(2 * Math.PI))) *
  436. Math.exp(-0.5 * Math.pow((midpoint - mean) / stdDev, 2))
  437. );
  438. });
  439. // 输出结果
  440. // console.log("Intervals:", intervals.slice(0, -1)); // 显示间隔
  441. // console.log("Frequencies:", frequency);
  442. // console.log("Normal Distribution Values:", normalDistributionValues);
  443. //---------------------------------正态分布end---------------------------
  444. // const quadraticData = this.generateQuadraticData(binCount, maxHistogramValue, totalHistogramValue);
  445. // littlegreen - 去掉line和scatter,增加x轴拖动,根据检测区间修改柱状的颜色
  446. const option = {
  447. color: ["rgba(245,0,0,0.6)"],
  448. dataZoom: [
  449. {
  450. type: "inside", // 内置于坐标系中
  451. start: 0,
  452. end: 100,
  453. xAxisIndex: [0],
  454. },
  455. ],
  456. tooltip: {
  457. trigger: "axis",
  458. axisPointer: {
  459. type: "shadow",
  460. },
  461. formatter: (params) => {
  462. // 使用箭头函数来保持 this 的上下文
  463. const seriesName = params[0].name; // 系列名称
  464. const dataIndex = params[0].dataIndex; // 数据索引
  465. const value = params[0].data; // 实际的数据值
  466. const xAxisLabel = params[0].axisValueLabel; // x轴的标签
  467. const yAxisLabel = params[0].seriesName;
  468. // 构建自定义的 tooltip 内容,包含年份
  469. return `${xAxisLabel}</br> ${yAxisLabel}: ${value} `;
  470. },
  471. },
  472. legend: {
  473. data: [
  474. `${this.selectedYear}年 ${
  475. this.keyToChinese[this.selectedDataKey]
  476. }企业数统计`,
  477. "正态分布"
  478. ],
  479. },
  480. grid: {
  481. left: "3%",
  482. right: "4%",
  483. bottom: "50px",
  484. containLabel: true,
  485. },
  486. xAxis: {
  487. type: "category",
  488. name: "分布区间",
  489. // data: histogramData.map((item) => item.name),
  490. data: intervalStrings.map(item=>item.name),
  491. axisLabel: {
  492. fontSize: 12,
  493. interval: 0,
  494. rotate: 20,
  495. // padding: [0, 0, 0, -200]
  496. },
  497. },
  498. yAxis: [{
  499. name: "企业数",
  500. type: "value",
  501. minInterval: 1,
  502. },{
  503. name: '正态分布',
  504. type: 'value'
  505. }],
  506. series: [
  507. {
  508. name: `${this.selectedYear}年 ${
  509. this.keyToChinese[this.selectedDataKey]
  510. }企业数统计`,
  511. type: "bar",
  512. barWidth: "30%",
  513. itemStyle: {
  514. // color: "rgba(245,0,0,0.6)",
  515. normal: {
  516. color: function (params) {
  517. const value = params.dataIndex; // 获取当前柱子在数据中的索引
  518. if (flag == 1) {
  519. let min = parseFloat(intervalStrings[value].min)*1000
  520. let max = parseFloat(intervalStrings[value].max)*1000
  521. const detectMin = parseFloat(that.detectMin)*1000
  522. const detectMax = parseFloat(that.detectMax)*1000
  523. if (
  524. min >= detectMin &&
  525. max <= detectMax
  526. ) {
  527. return "#ffcc00"; // 变更颜色为黄色
  528. }else if(min == detectMax){
  529. return "#ffcc00";
  530. }
  531. return "rgba(245,0,0,0.6)";
  532. }
  533. return "rgba(245,0,0,0.6)";
  534. // 默认颜色
  535. },
  536. },
  537. },
  538. // data: histogramData.map((item) => item.value),
  539. data: frequency,
  540. label: {
  541. show: false, // 将show属性设置为false,去掉柱子上的数字
  542. },
  543. },
  544. {
  545. name:"正态分布",
  546. yAxisIndex: 1,
  547. type: "line",
  548. data: normalDistributionValues,
  549. lineStyle: {
  550. color: "#1a7cc8",
  551. },
  552. symbol: "none",
  553. },
  554. ],
  555. };
  556. this.isChartShow = true;
  557. this.chart.setOption(option);
  558. this.chart.on("click", function (param) {
  559. // console.log(param.dataIndex, intervalStrings)
  560. if(typeof param.dataIndex === "number" &&
  561. param.dataIndex >= 0 &&
  562. param.dataIndex < intervalStrings.length
  563. ){
  564. let dataIndexValue = intervalStrings[param.dataIndex];
  565. const filteredArray = filteredData.filter((item)=>{
  566. const isMaxValEqual = parseFloat(dataIndexValue.max) == parseFloat(maxVal);
  567. const lowerBoundCheck = item[that.selectedDataKey] >= dataIndexValue.min;
  568. const upperBoundCheck = isMaxValEqual
  569. ? item[that.selectedDataKey] <= dataIndexValue.max
  570. : item[that.selectedDataKey] < dataIndexValue.max;
  571. return lowerBoundCheck && upperBoundCheck;
  572. })
  573. // console.log(filteredData,filteredArray,that.selectedDataKey)
  574. that.tableData = filteredArray.map((item) => {
  575. return {
  576. industryName: item.code,
  577. enterpriseId: item.enterpriseName,
  578. year: item.year,
  579. value: item[that.selectedDataKey],
  580. level: item[that.keyToLevel[that.selectedDataKey]],
  581. };
  582. });
  583. }
  584. // if (
  585. // typeof param.dataIndex === "number" &&
  586. // param.dataIndex >= 0 &&
  587. // param.dataIndex < intervalStrings.length
  588. // ) {
  589. // let dataIndexValue = intervalStrings[param.dataIndex];
  590. // const filteredArray = filteredData.filter((item) => {
  591. // const isMaxValEqual = intervalStrings[1] === maxVal;
  592. // const lowerBoundCheck =
  593. // item[that.selectedDataKey] >= dataIndexValue.min;
  594. // const upperBoundCheck = isMaxValEqual
  595. // ? item[that.selectedDataKey] <= dataIndexValue.max
  596. // : item[that.selectedDataKey] < dataIndexValue.max;
  597. // return lowerBoundCheck && upperBoundCheck;
  598. // });
  599. // that.tableData = filteredArray.map((item) => {
  600. // return {
  601. // industryName: item.code,
  602. // enterpriseId: item.enterpriseName,
  603. // year: item.year,
  604. // value: item[that.selectedDataKey],
  605. // level: "高",
  606. // };
  607. // });
  608. // }
  609. // const filteredArray = filteredData.filter((item) => {
  610. // const isMaxValEqual = result[1] === maxVal;
  611. // const lowerBoundCheck = item[that.selectedDataKey] >= dataIndexValue.min;
  612. // const upperBoundCheck = isMaxValEqual
  613. // ? item[that.selectedDataKey] <= dataIndexValue.max
  614. // : item[that.selectedDataKey] < dataIndexValue.max;
  615. // return lowerBoundCheck && upperBoundCheck;
  616. // })
  617. // that.tableData = filteredArray.map(item => {
  618. // return {
  619. // industryName: item.code,
  620. // enterpriseId: item.enterpriseName,
  621. // year: item.year,
  622. // value: item[that.selectedDataKey]
  623. // }
  624. // })
  625. });
  626. },
  627. generateQuadraticData(binCount, maxHistogramValue, totalHistogramValue) {
  628. const a = -0.01;
  629. const h = binCount / 2;
  630. const k = maxHistogramValue * 0.75;
  631. const sideValue = (maxHistogramValue * 0.15) / 2;
  632. const data = [];
  633. for (let i = 0; i < binCount; i++) {
  634. const x = i;
  635. const y = a * (x - h) ** 2 + k;
  636. // 调整两侧的值
  637. if (i === 0 || i === binCount - 1) {
  638. data.push([x, sideValue]);
  639. } else {
  640. data.push([x, y]);
  641. }
  642. }
  643. return data;
  644. },
  645. // littlegreen - 检测检测区间是否填写正确
  646. checkNumber() {
  647. const maxInput = parseFloat(this.detectMax);
  648. const minInput = parseFloat(this.detectMin);
  649. let flag;
  650. if (!this.detectMin && !this.detectMax) {
  651. return (flag = 0);
  652. }
  653. if (isNaN(maxInput) || isNaN(minInput)) {
  654. this.$message({
  655. message: "请输入有效的数字。",
  656. type: "error",
  657. });
  658. return (flag = 2);
  659. }
  660. // 检查 minInput 是否大于 maxInput
  661. if (minInput >= maxInput) {
  662. this.$message({
  663. message: "监测区间最大值不可小于或等于最小值",
  664. type: "error",
  665. });
  666. return (flag = 2);
  667. }
  668. return (flag = 1);
  669. },
  670. },
  671. watch: {
  672. // selectedDataKey() {
  673. // this.updateChart();
  674. // },
  675. // selectedYear() {
  676. // this.updateChart();
  677. // },
  678. // selectedRange() {
  679. // setTimeout(this.updateChart, 2500);
  680. // },
  681. // selectedCode() {
  682. // this.updateChart();
  683. // },
  684. },
  685. };
  686. </script>
  687. <style scoped>
  688. .select-container {
  689. margin: 10px 0;
  690. font-size: 16px; /* 调整字号大小 */
  691. }
  692. .select-container select {
  693. padding: 10px 20px; /* 增加内边距 */
  694. margin-right: 10px; /* 增加右边距 */
  695. border: 1px solid #121315; /* 蓝色边框 */
  696. border-radius: 4px; /* 圆角边框 */
  697. background-color: white; /* 背景色 */
  698. color: #121315; /* 文字颜色 */
  699. font-size: 16px; /* 调整字号大小 */
  700. cursor: pointer; /* 鼠标悬停时的指针样式 */
  701. outline: none; /* 移除焦点时的轮廓 */
  702. }
  703. .select-container select:hover {
  704. border-color: #1a7cc8; /* 鼠标悬停时的边框颜色 */
  705. }
  706. .select-container select:focus {
  707. border-color: #121315; /* 焦点时的边框颜色 */
  708. box-shadow: 0 0 0 2px rgba(64, 159, 255, 0.2); /* 焦点时的阴影效果 */
  709. }
  710. </style>