index.vue 17 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595
  1. <template>
  2. <div id="deviceList">
  3. <el-row class="right">
  4. <span style="float: left">当前选择的分区类型:{{ space.spaceName }}</span>
  5. <el-select
  6. v-model="showType"
  7. @change="initTable"
  8. style="width: 140px; margin-right: 10px; vertical-align: bottom"
  9. >
  10. <el-option
  11. v-for="item in showTypes"
  12. :key="item.value"
  13. :label="item.label"
  14. :value="item.value"
  15. ></el-option>
  16. </el-select>
  17. <span>增加</span>
  18. <el-input-number
  19. v-model="addNum"
  20. :controls="false"
  21. style="width: 50px"
  22. :min="1"
  23. :max="50"
  24. ></el-input-number>
  25. <span>个{{ space.spaceName }}</span>
  26. <el-button @click="handleAddTableRow">增加</el-button>
  27. </el-row>
  28. <div class="tableBox">
  29. <div
  30. class="center middle_sty"
  31. style="flex: 2"
  32. v-show="tableData && !tableData.length"
  33. >
  34. <p>
  35. <i class="icon-wushuju iconfont"></i>
  36. 暂无数据
  37. </p>
  38. </div>
  39. <div class="tableLeft" v-show="tableData && tableData.length">
  40. <handson-table ref="table"></handson-table>
  41. </div>
  42. </div>
  43. <el-row class="center">
  44. <el-button
  45. type="primary"
  46. size="medium"
  47. @click="handleCreateTableData"
  48. :disabled="createDisable"
  49. class="create_button"
  50. >创建业务空间</el-button
  51. >
  52. </el-row>
  53. <!--二维码弹窗 -->
  54. <qrcode
  55. :dialog="myDialog"
  56. :qrcodeUrl="qrcodeUrl"
  57. :addBody="true"
  58. ref="qrcode"
  59. ></qrcode>
  60. <!--上传图片-->
  61. <upload-img-dialog
  62. :read="false"
  63. @changeFile="imgChange"
  64. :keysArr="imgsArr"
  65. :dialog="myDialog"
  66. />
  67. <!-- 多选枚举类型维护 -->
  68. <menum
  69. :dialog="myDialog"
  70. :updateInfoData="updateInfoData"
  71. :updateInput="updateInput"
  72. @change="handleChangeMenum"
  73. ></menum>
  74. <!-- 不支持的输入方式 -->
  75. <el-dialog
  76. title="临时维护信息点"
  77. :visible.sync="myDialog.update"
  78. @close="handleCloseUpdate"
  79. width="670px"
  80. >
  81. <el-row>
  82. 该信息点未定义对应组件,现在维护数据不确定后续是否可用。如确实需要维护,请点击
  83. <el-link
  84. @click="updateInputShow = true"
  85. type="primary"
  86. :underline="false"
  87. >继续维护</el-link
  88. >
  89. </el-row>
  90. <el-row style="margin-top: 20px" v-show="updateInputShow">
  91. <el-input
  92. type="textarea"
  93. :autosize="{ minRows: 4, maxRows: 8 }"
  94. placeholder="请输入内容"
  95. v-model="updateInput"
  96. ></el-input>
  97. </el-row>
  98. <span slot="footer" class="dialog-footer">
  99. <el-button @click="myDialog.update = false">取 消</el-button>
  100. <el-button type="primary" @click="handleClickUpdate">确 认</el-button>
  101. </span>
  102. </el-dialog>
  103. </div>
  104. </template>
  105. <script>
  106. import tools from "@/utils/old-adm/scan/tools";
  107. import showTools from "@/utils/old-adm/handsontable/notShow";
  108. import session from "@/utils/storageUtil";
  109. import buildFloor from "@/utils/old-adm/handsontable/buildFloorData";
  110. import handsonTable from "@/components/old-adm/common/handsontable";
  111. import menum from "@/components/old-adm/dialogs/list/menum";
  112. import logicConfig from "@/logicConfig";
  113. import ScanController from "@/controller/old-adm/ScanController";
  114. import Dic from "@/controller/old-adm/dicController";
  115. import { mapGetters } from "vuex";
  116. import qrcode from "@/components/old-adm/ledger/components/lib/qrcode";
  117. import uploadImgDialog from "@/components/old-adm/dialogs/list/uploadImgDialog";
  118. export default {
  119. components: {
  120. qrcode, //二维码页面
  121. uploadImgDialog,
  122. handsonTable,
  123. menum,
  124. },
  125. data() {
  126. return {
  127. addNum: 1,
  128. onlyRead: false,
  129. showTypes: [
  130. { value: "Visible", label: "只看需交付的" },
  131. { value: "all", label: "全部" },
  132. ],
  133. tableHeader: [],
  134. tableData: session.get("spaceAddData")
  135. ? session.get("spaceAddData").length
  136. ? session.get("spaceAddData")
  137. : [{}]
  138. : [{}],
  139. copyTableData: [],
  140. category: {}, //路由参数
  141. myDialog: {
  142. qrcode: false, //二维码弹窗
  143. uploadImgs: false, //上传单个图片
  144. update: false, //临时维护信息点
  145. menum: false, //临时多选枚举类型信息点
  146. },
  147. inputMap: {
  148. flowBuild: {
  149. editable: true,
  150. code: "flowBuild",
  151. name: "建筑楼层",
  152. path: "flowBuild",
  153. category: "STATIC",
  154. dataType: "ENUM",
  155. },
  156. }, //信息点和输入方式映射表
  157. updateInputShow: false, //是否显示临时维护输入框
  158. updateInfoData: {}, //当前信息点信息
  159. updateInfoPoint: "", //临时维护信息点
  160. updateInput: "", //临时维护信息点值
  161. qrcodeUrl: "", //二维码图片地址
  162. imgsArr: [], //临时保存的图片key数组
  163. showType: this.$route.query.showType,
  164. buildFloorData: [],
  165. createDisable: false,
  166. };
  167. },
  168. computed: {
  169. ...mapGetters("layout", ["projectId", "secret", "userId"]),
  170. flowBuild() {
  171. let buildFloorSelectd = this.space.buildFloorSelectd;
  172. if (buildFloorSelectd && buildFloorSelectd.length == 0) {
  173. return "";
  174. } else if (buildFloorSelectd && buildFloorSelectd.length == 1) {
  175. let build = buildFloorSelectd[0];
  176. if (build == "all" || build == "noKnow") {
  177. return "";
  178. } else {
  179. return build;
  180. }
  181. } else if (buildFloorSelectd && buildFloorSelectd.length == 2) {
  182. let build = buildFloorSelectd[0],
  183. floor = buildFloorSelectd[1];
  184. if (build && (floor == "all" || floor == "noKnow")) {
  185. return build;
  186. } else {
  187. return `${build}-${floor}`;
  188. }
  189. }
  190. },
  191. },
  192. created() {
  193. this.space = this.$route.query;
  194. buildFloor.getData(this.buildFloorData);
  195. this.getTableHeader();
  196. },
  197. methods: {
  198. // 获取表头数据(初始化表格)
  199. async getTableHeader() {
  200. let params = {
  201. orders: "sort asc, name desc",
  202. pageNumber: 1,
  203. pageSize: 1000,
  204. type: this.space.spaceType,
  205. };
  206. Dic.getDataDictionary(params, (res) => {
  207. this.tableHeader = res.content;
  208. this.tableHeader.forEach((item) => {
  209. if (item.path) {
  210. this.inputMap[item.path] = item;
  211. }
  212. });
  213. this.tableData[0].flowBuild = this.flowBuild;
  214. this.initTable();
  215. });
  216. },
  217. // 创建业务空间数据
  218. async handleCreateTableData() {
  219. let newData = this.tableData.filter((item) => {
  220. let keys = Object.keys(item);
  221. keys.map((key) => {
  222. //将值为空字符串的属性删除
  223. if (item[key] == "") {
  224. delete item[key];
  225. }
  226. });
  227. let newK = Object.keys(item);
  228. if (newK.length) {
  229. return item;
  230. }
  231. });
  232. if (!newData.length) {
  233. this.$message("创建信息为空,请录入信息后再创建!");
  234. return;
  235. }
  236. let flag = false;
  237. newData.map((item) => {
  238. if (!item.localName) {
  239. flag = true;
  240. }
  241. // 处理建筑-楼层
  242. if (item.flowBuild) {
  243. let bid = item.flowBuild.split("-");
  244. if (bid[0] && bid[0] != "all" && bid[0] != "noKnow") {
  245. item.buildingId = item.flowBuild.split("-")[0];
  246. if (bid[1]) {
  247. if (bid[1] != "noKnow" && bid[1] != "all") {
  248. item.floorId = item.flowBuild.split("-")[1];
  249. }
  250. }
  251. }
  252. }
  253. item.classCode = this.space.spaceType;
  254. });
  255. if (flag) {
  256. this.$message.info("存在业务空间的本地名称为空,请检查");
  257. return;
  258. }
  259. //待接口修改为关联创建即可修改
  260. let param = {
  261. content: newData,
  262. };
  263. this.createDisable = true;
  264. ScanController.zoneCreate(param, (res) => {
  265. this.createDisable = false;
  266. switch (res.result) {
  267. case logicConfig.resultObj.success:
  268. this.$message.success("创建成功");
  269. session.remove("spaceAddData");
  270. this.$router.push({
  271. name: "spacelist",
  272. params: {
  273. zone: this.space.otherType
  274. ? this.space.otherType
  275. : this.space.spaceType,
  276. buildFloorSelectd: ["all"],
  277. isMyTab: "2",
  278. otherType: this.space.otherType ? this.space.spaceType : "",
  279. },
  280. });
  281. break;
  282. default:
  283. this.$message.error("创建失败");
  284. break;
  285. }
  286. });
  287. },
  288. // 删除表格行
  289. handleDeleteTableRow(a, b, c, d) {
  290. if (d && d === "ContextMenu.removeRow") {
  291. this.$message.success("删除成功");
  292. this.formaTableData();
  293. }
  294. },
  295. // 添加行
  296. handleAddTableRow() {
  297. let addRowLength = this.addNum;
  298. for (let i = 0; i < addRowLength; i++) {
  299. this.tableData.push({ flowBuild: this.flowBuild });
  300. }
  301. this.initTable();
  302. this.formaTableData();
  303. },
  304. //修改
  305. handleUpdataTable(changeData, source) {
  306. if (!this.onlyRead && source != "ObserveChanges.change") {
  307. this.formaTableData();
  308. }
  309. },
  310. //保存去掉空字段的新增数据
  311. formaTableData() {
  312. let newData = this.tableData.filter((item) => {
  313. let keys = Object.keys(item);
  314. keys.map((key) => {
  315. //将值为空字符串的属性删除
  316. if (item[key] == "") {
  317. delete item[key];
  318. }
  319. });
  320. if (keys.length && Object.keys(item).length) {
  321. return item;
  322. }
  323. });
  324. session.set("spaceAddData", newData);
  325. },
  326. //格式化表头显示的数据
  327. formatHeaderData(list) {
  328. let arr = tools.copyArr(list);
  329. let data = showTools.headerTextFilter(
  330. arr,
  331. "space",
  332. this.onlyRead,
  333. this.showType,
  334. true
  335. );
  336. data.unshift("所属建筑楼层");
  337. return data;
  338. },
  339. formatHeaderType(list) {
  340. //格式化表头头映射的数据
  341. let arr = tools.copyArr(list);
  342. let data = showTools.headerTypeFilter(
  343. arr,
  344. "space",
  345. this.onlyRead,
  346. this.showType,
  347. true
  348. );
  349. data.unshift({
  350. data: "flowBuild",
  351. renderer: tools.customDropdownRenderer,
  352. editor: "chosen",
  353. chosenOptions: {
  354. data: this.buildFloorData,
  355. },
  356. });
  357. return data;
  358. },
  359. initTable() {
  360. //实例化表格
  361. let settings = {
  362. data: this.tableData,
  363. colHeaders: this.formatHeaderData(this.tableHeader),
  364. columns: this.formatHeaderType(this.tableHeader),
  365. rowHeights: 30,
  366. fillHandle: "vertical", //允许纵向填充
  367. maxRows: this.tableData.length,
  368. contextMenu: this.onlyRead
  369. ? false
  370. : {
  371. items: {
  372. remove_row: {
  373. name: "删除业务空间",
  374. },
  375. },
  376. },
  377. // 事件
  378. afterChange: this.handleUpdataTable, //修改后
  379. afterFilter: this.trimmedRows, //排序前
  380. afterRemoveRow: this.handleDeleteTableRow, //右键删除
  381. afterOnCellMouseDown: this.handleTdClick, //鼠标点击
  382. };
  383. this.$nextTick(() => {
  384. this.tableExample = this.$refs.table.init(settings);
  385. });
  386. },
  387. //去除数组中相同的元素
  388. array_diff(a, b) {
  389. for (var i = 0; i < b.length; i++) {
  390. for (var j = 0; j < a.length; j++) {
  391. if (a[j] == b[i]) {
  392. a.splice(j, 1);
  393. j = j - 1;
  394. }
  395. }
  396. }
  397. return a;
  398. },
  399. //表格点击事件
  400. handleTdClick(el, rowArr) {
  401. //点击的是表头
  402. if (rowArr.row < 0) {
  403. return;
  404. }
  405. //被筛选过后的数组
  406. let trimmedArr = this.trimmedRows();
  407. //是否启用了排序
  408. let isSort = this.tableExample.getPlugin("columnSorting").isSorted();
  409. if (trimmedArr.length && isSort) {
  410. let sortArr =
  411. this.myHotArr.getPlugin("columnSorting").rowsMapper.__arrayMap;
  412. let infos = this.tableData[trimmedArr[sortArr[rowArr.row]]];
  413. this.getInfors(infos, { row: sortArr[rowArr.row], col: rowArr.col });
  414. } else if (isSort) {
  415. //排序后的数组
  416. let sortArr =
  417. this.tableExample.getPlugin("columnSorting").rowsMapper.__arrayMap;
  418. let infos = this.tableData[sortArr[rowArr.row]];
  419. this.getInfors(infos, { row: sortArr[rowArr.row], col: rowArr.col });
  420. } else if (trimmedArr.length) {
  421. let infos = this.tableData[trimmedArr[rowArr.row]];
  422. this.getInfors(infos, { row: trimmedArr[rowArr.row], col: rowArr.col });
  423. } else {
  424. let infos = this.tableData[rowArr.row];
  425. this.getInfors(infos, rowArr);
  426. }
  427. },
  428. //获取被筛选掉的行号
  429. trimmedRows() {
  430. var plugin = this.tableExample.getPlugin("trimRows").trimmedRows;
  431. let dataLength = this.tableData.length;
  432. let dataArr = new Array();
  433. for (let i = 0; i < dataLength; i++) {
  434. dataArr.push(i);
  435. }
  436. if (plugin.length <= 0) {
  437. dataArr = undefined;
  438. } else {
  439. dataArr = this.array_diff(dataArr, plugin);
  440. }
  441. return dataArr || [];
  442. },
  443. getInfors(infos, row) {
  444. let val = this.tableExample.colToProp(row.col);
  445. this.row = row.row;
  446. this.messKey = val;
  447. let inputData = this.inputMap[val];
  448. switch (val) {
  449. //空间二维码图片
  450. case "qRCodePicgg":
  451. this.qrcodeUrl = this.tableData[row.row].qRCodePic;
  452. if (!!this.qrcodeUrl) {
  453. this.myDialog.qrcode = true;
  454. } else {
  455. this.$message("此空间没有二维码");
  456. }
  457. break;
  458. case "pic":
  459. let Pdata = tools.dataForKey(this.tableData[row.row], val);
  460. this.imgsArr = Pdata ? Pdata : [];
  461. this.myDialog.uploadImgs = true;
  462. break;
  463. default:
  464. break;
  465. }
  466. // 维护多选枚举值
  467. if (!this.onlyRead && inputData.dataType == "MENUM") {
  468. this.updateInfoData = inputData;
  469. this.updateInfoPoint = val;
  470. this.updateInput = tools.dataForKey(this.tableData[row.row], val);
  471. this.myDialog.menum = true;
  472. return false;
  473. }
  474. if (!this.onlyRead && !inputData.editable) {
  475. this.$confirm("该信息点的值为自动生成,不可人工维护!", "提示", {
  476. confirmButtonText: "我知道了",
  477. showCancelButton: false,
  478. type: "warning",
  479. center: true,
  480. }).then(() => {
  481. return false;
  482. });
  483. }
  484. if (
  485. !this.onlyRead &&
  486. showTools.inputModeArr.indexOf(inputData.dataType) == "-1"
  487. ) {
  488. this.updateInfoPoint = val;
  489. this.updateInput = tools.dataForKey(this.tableData[row.row], val);
  490. this.myDialog.update = true;
  491. return false;
  492. }
  493. },
  494. //关闭临时维护弹窗回调
  495. handleCloseUpdate() {
  496. this.updateInputShow = false;
  497. this.updateInfoPoint = "";
  498. this.updateInput = "";
  499. },
  500. //更新临时维护信息点
  501. handleClickUpdate() {
  502. tools.setDataForKey(
  503. this.tableData[this.row],
  504. this.updateInfoPoint,
  505. this.updateInput
  506. );
  507. this.handleUpdataTable(
  508. [[this.row, this.updateInfoPoint, null, this.updateInput]],
  509. "edit"
  510. );
  511. this.updateInputShow = false;
  512. this.myDialog.update = false;
  513. this.updateInput = "";
  514. },
  515. //多选枚举类型值修改
  516. handleChangeMenum(newValue) {
  517. tools.setDataForKey(
  518. this.tableData[this.row],
  519. this.updateInfoPoint,
  520. newValue
  521. );
  522. this.handleUpdataTable(
  523. [[this.row, this.updateInfoPoint, null, newValue]],
  524. "edit"
  525. );
  526. this.myDialog.menum = false;
  527. this.updateInput = "";
  528. },
  529. utilToKey(key, name, data, messName) {
  530. if (key == name) {
  531. this.setDataToMain(data[key], messName, this.row);
  532. }
  533. },
  534. //上传图片弹窗触发事件
  535. imgChange(keys) {
  536. this.setDataToMain(keys, "pic", this.row);
  537. },
  538. //判断是否有值,有值赋值
  539. setDataToMain(data, key, row) {
  540. if (!!data && data != "--") {
  541. if (!!this.tableData[row]) {
  542. //铭牌照片特殊处理
  543. tools.setDataForKey(this.tableData[row], key, data);
  544. // this.tableData[row][key] = data;
  545. } else {
  546. this.tableData[row] = {};
  547. tools.setDataForKey(this.tableData[row], key, data);
  548. }
  549. } else {
  550. tools.setDataForKey(this.tableData[row], key, "");
  551. }
  552. },
  553. },
  554. };
  555. </script>
  556. <style lang="less" scoped>
  557. #deviceList {
  558. overflow: hidden;
  559. height: 100%;
  560. background-color: #fff;
  561. padding: 10px;
  562. position: relative;
  563. .right {
  564. background: #fff;
  565. }
  566. .search-header {
  567. overflow: hidden;
  568. padding: 0 10px 10px 10px;
  569. border-bottom: 1px solid #bcbcbc;
  570. }
  571. .tableBox {
  572. display: flex;
  573. height: calc(100% - 100px);
  574. margin-top: 10px;
  575. .tableLeft {
  576. flex: 1;
  577. }
  578. }
  579. .create_button {
  580. margin-top: 10px;
  581. }
  582. }
  583. </style>