index.vue 14 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462
  1. <template>
  2. <div
  3. :class="className"
  4. :style="{ height: '100%', width: width, padding: '15px' }"
  5. >
  6. <div class="select-container">
  7. <el-form
  8. style="display: grid; grid-template-columns: 1fr 1fr 1fr; grid-gap: 5px"
  9. label-width="100px"
  10. >
  11. <el-form-item label="评估指标">
  12. <el-select v-model="matchForm.dataKey" style="width: 100%">
  13. <el-option
  14. v-for="key in dataKeys"
  15. :key="key"
  16. :value="key"
  17. :label="keyToChinese[key]"
  18. >
  19. {{ keyToChinese[key] }}
  20. </el-option>
  21. </el-select>
  22. </el-form-item>
  23. <el-form-item label="行业">
  24. <el-select
  25. v-model="matchForm.industry"
  26. style="width: 100%"
  27. filterable
  28. >
  29. <el-option
  30. v-for="item in industryData"
  31. :key="item.key"
  32. :label="item.value"
  33. :value="item.key"
  34. >
  35. </el-option>
  36. </el-select>
  37. </el-form-item>
  38. <el-form-item label="企业分类">
  39. <el-select
  40. v-model="matchForm.companyType"
  41. style="width: 100%"
  42. clearable
  43. >
  44. <el-option
  45. v-for="item in enterTypeList"
  46. :key="item.number"
  47. :value="item.number"
  48. :label="keyToChineseEtype(item.number)"
  49. >
  50. </el-option>
  51. </el-select>
  52. </el-form-item>
  53. <el-form-item label="相差值阈值">
  54. <el-input
  55. v-model="matchForm.threshold"
  56. placeholder="请输入内容"
  57. style="width: 100%"
  58. ></el-input>
  59. </el-form-item>
  60. <!-- 用电:月度;用气:年度;用人:季度 -->
  61. <el-form-item label="年度">
  62. <el-select v-model="matchForm.year" style="width: 100%" filterable>
  63. <el-option
  64. v-for="key in yearsOptions"
  65. :key="key"
  66. :label="key"
  67. :value="key"
  68. >
  69. </el-option>
  70. </el-select>
  71. </el-form-item>
  72. <el-form-item label="月度" v-if="matchForm.dataKey === 'powerConsume'">
  73. <div style="display: flex; gap: 5px">
  74. <el-select v-model="matchForm.elecStartMonth" filterable>
  75. <el-option
  76. v-for="key in monthOptions"
  77. :key="key"
  78. :label="key"
  79. :value="key"
  80. >
  81. </el-option>
  82. </el-select>
  83. <span>-</span>
  84. <el-select v-model="matchForm.elecEndMonth" filterable>
  85. <el-option
  86. v-for="key in monthOptions"
  87. :key="key"
  88. :label="key"
  89. :value="key"
  90. >
  91. </el-option>
  92. </el-select>
  93. </div>
  94. </el-form-item>
  95. <el-form-item
  96. label="季度"
  97. v-if="matchForm.dataKey === 'employeeNumber'"
  98. >
  99. <div style="display: flex; gap: 5px">
  100. <el-select v-model="matchForm.manStartSeason" filterable>
  101. <el-option
  102. v-for="key in seasonOptions"
  103. :key="key"
  104. :label="key"
  105. :value="key"
  106. >
  107. </el-option>
  108. </el-select>
  109. <span>-</span>
  110. <el-select v-model="matchForm.manEndSeason" filterable>
  111. <el-option
  112. v-for="key in seasonOptions"
  113. :key="key"
  114. :label="key"
  115. :value="key"
  116. >
  117. </el-option>
  118. </el-select>
  119. </div>
  120. </el-form-item>
  121. <el-form-item>
  122. <el-button
  123. type="primary"
  124. icon="el-icon-search"
  125. @click="submit"
  126. size="mini"
  127. >搜索</el-button
  128. >
  129. </el-form-item>
  130. </el-form>
  131. </div>
  132. <el-table :data="tableData" border>\
  133. code: "2422"
  134. enterpriseName: "企业1"
  135. location: "A"
  136. trueThreshold: -0.24883973894126177
  137. typeName: "A"
  138. typeNum: "1"
  139. <el-table-column prop="enterpriseName" label="企业名称" sortable></el-table-column>
  140. <el-table-column
  141. prop="location"
  142. label="坐落地"
  143. width="200" sortable
  144. ></el-table-column>
  145. <el-table-column
  146. prop="code"
  147. label="行业代码"
  148. width="200"
  149. ></el-table-column>
  150. <el-table-column
  151. prop="typeName"
  152. label="企业分类"
  153. width="200"
  154. ></el-table-column>
  155. <el-table-column
  156. prop="trueThreshold"
  157. label="相差值"
  158. width="200" sortable
  159. ></el-table-column>
  160. </el-table>
  161. <el-pagination
  162. v-show="total > 0"
  163. :total="total"
  164. :page.sync="matchForm.pageNum"
  165. :limit.sync="matchForm.pageSize"
  166. @pagination="submit"
  167. >
  168. </el-pagination>
  169. </div>
  170. </template>
  171. <script>
  172. import { analysisMatch } from "@/api/analysis/analysis";
  173. import { getYearData } from "@/api/home";
  174. import { listEtypeAll } from "@/api/etype/etype";
  175. import { listAllIndustry } from "@/api/industry/industry"; // 导入行业数据的接口
  176. export default {
  177. props: {
  178. className: {
  179. type: String,
  180. default: "chart",
  181. },
  182. width: {
  183. type: String,
  184. default: "100%",
  185. },
  186. height: {
  187. type: String,
  188. default: "400px",
  189. },
  190. },
  191. data() {
  192. return {
  193. total: 0,
  194. matchForm: {
  195. pageNum: 1,
  196. pageSize: 10,
  197. dataKey: "powerConsume", //评估指标
  198. industry: "", //行业
  199. companyType: "", //企业分类
  200. threshold: null, //相差值阈值
  201. // 用电
  202. year: "", //年度 (用电和用人都用这个年度)
  203. elecStartMonth: "", //起始月
  204. elecEndMonth: "", //结束月
  205. // 用人
  206. manStartSeason: "", //起始季度
  207. manEndSeason: "", //结束季度
  208. },
  209. dataKeys: ["powerConsume", "energyConsume", "employeeNumber"], // 可选项
  210. keyToChinese: {
  211. landArea: "用地面积",
  212. totalIndustrialValue: "工业总产值",
  213. gdp: "工业增加值",
  214. taxableIncome: "应税收入",
  215. paidTax: "实缴税金",
  216. mainBusinessIncome: "主营业务收入",
  217. employeeNumber: "从业人员数",
  218. profit: "利润总额",
  219. ownerEquity: "所有者权益",
  220. funding: "研发经费",
  221. energyConsume: "能源消费量",
  222. powerConsume: "电力消费量",
  223. },
  224. seasonMap: new Map(),
  225. monthMap: new Map(),
  226. yearsOptions: [],
  227. enterTypeList: [],
  228. enterTypeList: [],
  229. industryData: [],
  230. industryMap: new Map(),
  231. tableData: [],
  232. };
  233. },
  234. mounted() {
  235. this.$nextTick(() => {
  236. this.fetchData();
  237. });
  238. },
  239. methods: {
  240. fetchData() {
  241. // 获取年份,获取季度、月度
  242. this.getYearData();
  243. this.getAllIndustry();
  244. this.getEtypeAll();
  245. },
  246. getYearData() {
  247. return getYearData().then((res) => {
  248. this.yearsOptions = res.rows[0].years;
  249. this.seasonMap = this.generateSeason(this.yearsOptions);
  250. this.monthMap = this.generateMonthlyData(this.yearsOptions);
  251. });
  252. },
  253. getAllIndustry() {
  254. return listAllIndustry().then((response) => {
  255. if (response?.rows) {
  256. this.industryData = response.rows.map((item) => ({
  257. key: item.code,
  258. value: item.code + item.industryName,
  259. }));
  260. }
  261. });
  262. },
  263. getEtypeAll() {
  264. return listEtypeAll().then((res) => {
  265. this.enterTypeList = res;
  266. // console.log(this.enterTypeList);
  267. });
  268. },
  269. generateSeason(years) {
  270. const currentYear = new Date().getFullYear();
  271. const currentMonth = new Date().getMonth() + 1;
  272. const season = new Map();
  273. years.forEach((year) => {
  274. if (year === currentYear) {
  275. // 当前年份,判断当前月份
  276. if (currentMonth < 1) {
  277. season.set(year, []);
  278. } else if (currentMonth < 4) {
  279. season.set(year, [1]); // 第一季度
  280. } else if (currentMonth < 7) {
  281. season.set(year, [1, 2]); // 前两季度
  282. } else if (currentMonth < 10) {
  283. season.set(year, [1, 2, 3]); // 前三季度
  284. } else {
  285. season.set(year, [1, 2, 3, 4]); // 四个季度
  286. }
  287. } else if (year < currentYear) {
  288. // 过去的年份都包含四个季度
  289. season.set(year, [1, 2, 3, 4]);
  290. } else {
  291. // 未来的年份
  292. season.set(year, []);
  293. }
  294. });
  295. return season;
  296. },
  297. generateMonthlyData(years) {
  298. const currentYear = new Date().getFullYear();
  299. const currentMonth = new Date().getMonth() + 1;
  300. const monthlyData = new Map();
  301. years.forEach((year) => {
  302. if (year === currentYear) {
  303. // 当前年份,判断当前月份
  304. const months = Array.from({ length: currentMonth }, (_, i) => i + 1);
  305. monthlyData.set(year, months); // 设置当前年份的已过月份
  306. } else if (year < currentYear) {
  307. // 过去的年份包含所有12个月
  308. monthlyData.set(year, [1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 12]);
  309. } else {
  310. // 未来的年份
  311. monthlyData.set(year, []);
  312. }
  313. });
  314. return monthlyData;
  315. },
  316. keyToChineseEtype(num) {
  317. const item = this.enterTypeList.find((element) => element.number === num);
  318. return item ? item.name : null;
  319. },
  320. submit() {
  321. // 清除无关字段
  322. this.clearUnrelatedFields();
  323. // 必填项校验
  324. if (this.validateForm()) {
  325. // this.$message.success("提交成功");
  326. // console.log(this.matchForm);
  327. analysisMatch(this.matchForm).then(res=>{
  328. // console.log(res,"test")
  329. if(res && res?.rows){
  330. this.total = res.total;
  331. this.tableData = res.rows.map(item => {
  332. return {
  333. ...item,
  334. trueThreshold: parseFloat(item.trueThreshold).toFixed(4) // 转换并保留4位小数
  335. };
  336. });
  337. }
  338. })
  339. }
  340. },
  341. clearUnrelatedFields() {
  342. const { dataKey } = this.matchForm;
  343. if (dataKey === "powerConsume") {
  344. this.matchForm.manStartSeason = "";
  345. this.matchForm.manEndSeason = "";
  346. } else if (dataKey === "employeeNumber") {
  347. this.matchForm.elecStartMonth = "";
  348. this.matchForm.elecEndMonth = "";
  349. } else if (dataKey === "energyConsume") {
  350. this.matchForm.manStartSeason = "";
  351. this.matchForm.manEndSeason = "";
  352. this.matchForm.elecStartMonth = "";
  353. this.matchForm.elecEndMonth = "";
  354. }
  355. },
  356. validateForm() {
  357. const { matchForm } = this;
  358. // 行业校验
  359. if (!matchForm.industry) {
  360. this.$message.error("请选择行业");
  361. return false;
  362. }
  363. // 阈值校验
  364. const threshold = parseFloat(matchForm.threshold);
  365. if (isNaN(threshold)) {
  366. this.$message.error("请填写有效的相差值阈值");
  367. return false;
  368. }
  369. matchForm.threshold = threshold; // 转换为数字
  370. // 年度必填校验
  371. if (!matchForm.year) {
  372. this.$message.error("请选择年度");
  373. return false;
  374. }
  375. // 针对 powerConsume 的校验
  376. if (matchForm.dataKey === "powerConsume") {
  377. if (!matchForm.elecStartMonth) {
  378. this.$message.error("请选择起始月度");
  379. return false;
  380. }
  381. if (!matchForm.elecEndMonth) {
  382. this.$message.error("请选择结束月度");
  383. return false;
  384. }
  385. if (matchForm.elecStartMonth > matchForm.elecEndMonth) {
  386. this.$message.error("起始月度必须小于等于结束月度");
  387. return false;
  388. }
  389. }
  390. // 针对 employeeNumber 的校验
  391. if (matchForm.dataKey === "employeeNumber") {
  392. if (!matchForm.manStartSeason) {
  393. this.$message.error("请选择起始季度");
  394. return false;
  395. }
  396. if (!matchForm.manEndSeason) {
  397. this.$message.error("请选择结束季度");
  398. return false;
  399. }
  400. if (matchForm.manStartSeason > matchForm.manEndSeason) {
  401. this.$message.error("起始季度必须小于等于结束季度");
  402. return false;
  403. }
  404. }
  405. return true; // 所有校验通过
  406. },
  407. },
  408. computed: {
  409. seasonOptions() {
  410. const options = this.matchForm.year
  411. ? this.seasonMap.get(this.matchForm.year)
  412. : [];
  413. return options || []; // 以防得到 undefined
  414. },
  415. monthOptions() {
  416. const options = this.matchForm.year
  417. ? this.monthMap.get(this.matchForm.year)
  418. : [];
  419. return options || []; // 以防得到 undefined
  420. },
  421. },
  422. };
  423. </script>
  424. <style scoped>
  425. .select-container {
  426. margin: 10px 0;
  427. font-size: 16px; /* 调整字号大小 */
  428. }
  429. .select-container select {
  430. padding: 10px 20px; /* 增加内边距 */
  431. margin-right: 10px; /* 增加右边距 */
  432. border: 1px solid #121315; /* 蓝色边框 */
  433. border-radius: 4px; /* 圆角边框 */
  434. background-color: white; /* 背景色 */
  435. color: #121315; /* 文字颜色 */
  436. font-size: 16px; /* 调整字号大小 */
  437. cursor: pointer; /* 鼠标悬停时的指针样式 */
  438. outline: none; /* 移除焦点时的轮廓 */
  439. }
  440. .select-container select:hover {
  441. border-color: #1a7cc8; /* 鼠标悬停时的边框颜色 */
  442. }
  443. .select-container select:focus {
  444. border-color: #121315; /* 焦点时的边框颜色 */
  445. box-shadow: 0 0 0 2px rgba(64, 159, 255, 0.2); /* 焦点时的阴影效果 */
  446. }
  447. </style>