index.vue 22 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729
  1. <template>
  2. <div class="adm-device">
  3. <statistics :statistics-msg="statisticsMsg"/>
  4. <div class="operation">
  5. <el-cascader :options="list" clearable v-model="deviceType" :props="optionProps"
  6. @change="handleChangeDevice"
  7. class="adm-select"></el-cascader>
  8. <admSearch @SearchValue="SearchValue"/>
  9. <el-button type="default" @click="addDevice" class="adm-btn">添加设备</el-button>
  10. </div>
  11. <div class="hr"></div>
  12. <div class="content">
  13. <el-tabs v-model="activeName" type="card" @tab-click="handleClick">
  14. <el-tab-pane v-for="(item,index) in paneMsg" :key="index" :label="item" :name="index.toString()"/>
  15. </el-tabs>
  16. <div class="table" v-loading="loading">
  17. <template v-if="deviceType.length > 0">
  18. <admMultiTable :currentHeader="currentHeader"
  19. @handleCurrentEdit="handleCurrentEdit"
  20. :tableData="tableData" :headersStage="headersStage"/>
  21. <Pagination v-if="tableData.length > 0" :paginationList="paginationList"
  22. @handleCurrentChange="handleCurrentChange"
  23. @handleSizeChange="handleSizeChange"/>
  24. </template>
  25. <div v-else class="void align">
  26. <svg-icon name="void" :width="String(120)" :height="String(123)"/>
  27. <p class="void-title">暂无内容</p>
  28. <p class="void-tips">可点击左上角选择设备类型</p>
  29. </div>
  30. </div>
  31. </div>
  32. <!-- 添加/编辑 设备-->
  33. <el-scrollbar style="height:400px;">
  34. <el-dialog
  35. :title="deviceMsg"
  36. :width="!isWidth ? '20%':''"
  37. :visible.sync="dialogVisible"
  38. @close="close">
  39. <template slot="title">
  40. <div class="alertTitle">
  41. <span>{{ deviceMsg }}</span>
  42. <el-button type="text">
  43. <span class="el-icon-question"/>
  44. <a href="/datacenter/object/equip/downloads" download="维护帮助"> 维护帮助</a>
  45. </el-button>
  46. </div>
  47. </template>
  48. <template v-if="deviceMsg == '添加设备'">
  49. <template v-if="next">
  50. <div class="align " :style="{ 'height': isWidth ? '400px':'200px' }">
  51. <span class="text ">设备类别</span>
  52. <el-cascader :options="list" clearable v-model="deviceVal"
  53. :props="optionProps"
  54. class="adm-select"></el-cascader>
  55. </div>
  56. <el-button type="primary" class="fr" @click="handleNext">下一步</el-button>
  57. </template>
  58. <template v-else-if="displayLocation && deviceVal">
  59. <deviceGraph ref="deviceGraph" :equip="curEquip" @goBack="goBack"/>
  60. <span slot="footer" class="dialog-footer">
  61. <el-button type="primary" @click="saveLocation">确定</el-button>
  62. <el-button type="default" @click="resetLocation">重置</el-button>
  63. </span>
  64. </template>
  65. <template v-else>
  66. <dataForm :deviceHeaders="deviceHeaders" ref="dataForm" :currRowContent="currRowContent"/>
  67. <span slot="footer" class="dialog-footer">
  68. <el-button type="primary" @click="handleDataForm">确 定</el-button>
  69. <el-button @click="handlePosition">维护位置</el-button>
  70. </span>
  71. </template>
  72. </template>
  73. <template v-else>
  74. <template v-if="displayLocation && deviceVal">
  75. <deviceGraph ref="deviceGraph" :equip="curEquip" @goBack="goBack"/>
  76. <span slot="footer" class="dialog-footer">
  77. <el-button type="primary" @click="saveLocation">确定</el-button>
  78. <el-button type="default" @click="resetLocation">重置</el-button>
  79. </span>
  80. </template>
  81. <template v-else>
  82. <dataForm :deviceHeaders="deviceHeaders" ref="dataForm" :currRowContent="currRowContent"/>
  83. <span slot="footer" class="dialog-footer">
  84. <el-button type="danger" style="float: left" @click="deleteDevice">删除设备</el-button>
  85. <el-button type="primary" @click="handleDataForm">确 定</el-button>
  86. <el-button @click="handlePosition">维护位置</el-button>
  87. </span>
  88. </template>
  89. </template>
  90. </el-dialog>
  91. </el-scrollbar>
  92. </div>
  93. </template>
  94. <script lang="ts">
  95. import {Component, Vue, Watch} from "vue-property-decorator";
  96. import {AdmMultiTable, AdmSearch, dataForm, Pagination, Statistics} from '../components/index'
  97. import {allDevice, BeatchQueryParam, dictInfo} from "@/api/equipComponent";
  98. import {createEquip, queryCount, queryEquip, updateEquip, deleteEquip} from "@/api/datacenter";
  99. import {UserModule} from "@/store/modules/user";
  100. import deviceGraph from "./components/deviceGraph.vue"
  101. import tools from "@/utils/maintain"
  102. @Component({
  103. name: 'adm-device',
  104. components: {Statistics, AdmSearch, AdmMultiTable, Pagination, dataForm, deviceGraph}
  105. })
  106. export default class extends Vue {
  107. optionProps = {
  108. value: 'code',
  109. label: 'aliasName',
  110. children: 'children'
  111. }
  112. // loading
  113. loading = false
  114. // 统计信息对象
  115. private statisticsMsg = {
  116. title: '全部设备',
  117. total: 0
  118. }
  119. // 设置高度
  120. isWidth = false
  121. // 设备类值
  122. deviceType = ''
  123. // 弹窗设备类值
  124. deviceVal = ''
  125. // 维护位置
  126. displayLocation = false
  127. // 表头信息集合
  128. headerInformation = {}
  129. // 表头阶段信息结合
  130. headersStage = {}
  131. // 当前阶段表头信息点集合
  132. all = []
  133. // 搜索内容
  134. inputSearch = ''
  135. // 下拉数据
  136. list = []
  137. // 弹窗开关
  138. dialogVisible = false
  139. // tabs数据
  140. paneMsg = []
  141. // 当前tabs值
  142. activeName = 0
  143. // 分页
  144. paginationList = {
  145. page: 1,
  146. size: 50,
  147. sizes: [10, 30, 50, 100, 150, 200],
  148. total: 0
  149. }
  150. // 下一步
  151. next = true
  152. // 弹窗 title
  153. deviceMsg = ''
  154. // 默认当前阶段
  155. currentHeader = ''
  156. // 主体数据
  157. tableData = []
  158. codeToDataSource = {}
  159. deviceHeaders = {}
  160. // 当前行数据
  161. currRowContent = {}
  162. // 维护位置开关
  163. maintain = ''
  164. // 传到维护位置的设备信息
  165. curEquip = {}
  166. // 项目id
  167. get projectId(): string {
  168. return UserModule.projectId
  169. }
  170. created() {
  171. this.deviceList();
  172. this.dataCount()
  173. }
  174. //查询统计数量
  175. dataCount() {
  176. queryCount({}).then(res => {
  177. this.statisticsMsg.total = res.count
  178. })
  179. }
  180. // 设备类数据
  181. deviceList() {
  182. allDevice({}).then(res => {
  183. this.list = res.content
  184. console.log(res)
  185. })
  186. }
  187. handleChangeDevice() {
  188. if (this.deviceType[1]) {
  189. this.loading = true
  190. let param = {
  191. category: this.deviceType[1]
  192. }
  193. let param2 = {
  194. filters: this.deviceType[1] ? `classCode='${this.deviceType[1]}'` : undefined,
  195. pageNumber: this.paginationList.page,
  196. pageSize: this.paginationList.size,
  197. orders: "createTime desc, id asc",
  198. projectId: this.projectId,
  199. cascade: [{"name": "floor",}, {"name": "objectInfo"}]
  200. }
  201. if (this.inputSearch != '') {
  202. param2.filters += `;codeName contain '${this.inputSearch}' or systemCategory contain '${this.inputSearch}' or bimTypeId contain '${this.inputSearch}' or localId contain '${this.inputSearch}'`
  203. }
  204. let promise = new Promise(resolve => {
  205. dictInfo(param).then((res: []) => {
  206. resolve(res)
  207. })
  208. })
  209. let promise2 = new Promise(resolve => {
  210. queryEquip(param2).then((res: []) => {
  211. resolve(res)
  212. })
  213. })
  214. Promise.all([promise, promise2]).then(res => {
  215. this.loading = false
  216. // 类型下信息点,默认设计阶段
  217. this.headerInformation = res[0] // 获取表头
  218. // this.tableData = res[1].content // 主体数据
  219. this.tableData = res[1].content
  220. this.paneMsg = res[0].dictStages.map(i => i.name)
  221. this.currentHeader = this.paneMsg[this.activeName]
  222. this.headerStage()
  223. this.paginationList.total = res[1].total
  224. })
  225. } else {
  226. this.headerInformation = {}
  227. }
  228. }
  229. async handleNext() {
  230. if (this.deviceVal[1]) {
  231. this.next = false
  232. this.isWidth = true
  233. let param = {
  234. category: this.deviceVal[1]
  235. }
  236. await dictInfo(param).then(res => {
  237. const basicInfos = [{path: 'bimTypeName', aliasName: '构件分类名称', category: "STATIC", editable: true},
  238. {path: 'localId', aliasName: '本地编码', category: "STATIC", editable: true},
  239. {path: 'floor.localName', editable: false, aliasName: '所属楼层', category: "STATIC"},
  240. {
  241. path: 'onSpace',
  242. editable: false,
  243. aliasName: '所在空间',
  244. dataType: 'STRING',
  245. category: "STATIC",
  246. render: (obj: any) => {
  247. return obj?.objectInfo && obj.objectInfo.map((item: any) => item.localName).join(',') || ''
  248. }
  249. }]
  250. this.deviceHeaders = {
  251. basicInfos,
  252. dictStages: res.dictStages
  253. }
  254. })
  255. } else {
  256. console.log(5)
  257. }
  258. }
  259. headerStage() {
  260. let pic = [], base = []
  261. if (Object.keys(this.headerInformation).length > 0) {
  262. this.headerInformation.dictStages.forEach(item => {
  263. if (this.currentHeader == item.name) {
  264. item.infos && item.infos.forEach(val => {
  265. if (val.dataType == 'ATTACHMENT') {
  266. pic.push(val)
  267. } else {
  268. base.push(val)
  269. }
  270. })
  271. }
  272. })
  273. }
  274. // this.headersStage = {
  275. // basicInfos: {
  276. // name: '基础信息台账',
  277. // data: this.headerInformation.basicInfos
  278. // },
  279. // dictStages: {
  280. // name: this.currentHeader,
  281. // data: pic.length > 0 ? [...base, ...pic] : [...base]
  282. // }
  283. // }
  284. // todo 固定写死基础信息台账
  285. this.headersStage = {
  286. basicInfos: {
  287. name: '基础信息台账',
  288. data: [
  289. {path: 'bimTypeName', aliasName: '构件分类名称', category: "STATIC"},
  290. {path: 'localId', aliasName: '本地编码', category: "STATIC"},
  291. {path: 'floor.localName', editable: false, aliasName: '所属楼层', category: "STATIC"},
  292. {
  293. path: 'onSpace',
  294. editable: false,
  295. aliasName: '所在空间',
  296. dataType: 'STRING',
  297. category: "STATIC",
  298. render: (obj: any) => {
  299. return obj?.objectInfo && obj.objectInfo.map((item: any) => item.localName).join(',') || ''
  300. }
  301. }
  302. ],
  303. },
  304. dictStages: {
  305. name: this.currentHeader,
  306. data: pic.length > 0 ? [...base, ...pic] : [...base]
  307. }
  308. }
  309. // 信息点集合
  310. this.all = [...this.headersStage.basicInfos.data, ...this.headersStage.dictStages.data]
  311. this.codeToDataSource = {}
  312. this.all.forEach(item => {
  313. if (item.dataSource) {
  314. try {
  315. this.codeToDataSource[item.code] = {}
  316. item.dataSource.forEach(dic => {
  317. this.codeToDataSource[item.code][dic.code] = dic.name;
  318. })
  319. } catch (e) {
  320. console.log(e);
  321. }
  322. }
  323. });
  324. this.getBatch(this.tableData)
  325. }
  326. // 查询动态数据
  327. getBatch(data) {
  328. let param = {
  329. groupCode: 'WD',
  330. appId: 'datacenter',
  331. projectId: this.projectId,
  332. data: []
  333. }
  334. this.all.forEach(head => {
  335. if (head.category != 'STATIC') {
  336. data.forEach(item => {
  337. let cur = tools.dataForKey(item, head.path)
  338. if (cur) {
  339. param.data.push({
  340. objectId: item.id,
  341. infoCode: head.code
  342. })
  343. }
  344. })
  345. }
  346. })
  347. if (param.data.length > 0) {
  348. BeatchQueryParam(param).then(res => {
  349. this.tableData = data.map(item => {
  350. res.data.map(child => {
  351. if (item.id == child.objectId) {
  352. if (!!child.data || child.data == 0) {
  353. this.all.map(head => {
  354. if (head.code == child.infoCode) {
  355. let contentVal = child.data
  356. if (this.codeToDataSource[child.infoCode]) {
  357. contentVal = this.codeToDataSource[child.infoCode][child.data]
  358. }
  359. tools.setDataForKey(item, head.path, contentVal);
  360. }
  361. })
  362. } else {
  363. this.all.map(head => {
  364. if (head.code == child.infoCode) {
  365. tools.setDataForKey(
  366. item,
  367. head.path,
  368. child.error ? child.value
  369. ? "表号功能号格式错误"
  370. : "表号功能号不存在"
  371. : "暂未采集到实时数据"
  372. );
  373. }
  374. });
  375. }
  376. }
  377. })
  378. return item
  379. })
  380. })
  381. }
  382. }
  383. // 维护阶段 tabs
  384. async handleClick(val: any) {
  385. this.currentHeader = val.label
  386. await this.headerStage()
  387. }
  388. // 搜索
  389. SearchValue(val: string) {
  390. this.inputSearch = val
  391. this.handleChangeDevice(this.deviceType[1])
  392. }
  393. // 当前分页
  394. handleCurrentChange(val: number) {
  395. console.log(val)
  396. this.paginationList.page = val
  397. this.handleChangeDevice(this.deviceType[1])
  398. }
  399. handleSizeChange(val: number) {
  400. this.paginationList.size = val
  401. this.handleChangeDevice(this.deviceType[1])
  402. }
  403. // 添加设备
  404. addDevice() {
  405. this.deviceMsg = '添加设备'
  406. this.dialogVisible = true
  407. this.currRowContent = {}
  408. }
  409. // 维护位置
  410. handlePosition() {
  411. this.currentRow = this.$refs.dataForm.form
  412. this.$refs.dataForm.submitForm(this.handlePositionSave)
  413. }
  414. handlePositionSave() {
  415. this.curEquip = tools.formatData(this.$refs.dataForm.form)
  416. this.displayLocation = true
  417. }
  418. // 添加 事件处理
  419. handleDataForm() {
  420. this.$refs.dataForm.submitForm(this.handleDataFormSave)
  421. }
  422. // 删除设备
  423. deleteDevice() {
  424. deleteEquip([{id: this.currRowContent.id}]).then(res => {
  425. if (res.result == 'success') {
  426. this.$message.success('删除成功')
  427. this.handleChangeDevice()
  428. this.dialogVisible = false;
  429. }
  430. })
  431. }
  432. handleDataFormSave() {
  433. const eq = tools.formatData(this.$refs.dataForm.form);
  434. if (eq.id) {
  435. //更新
  436. this.handleUpdateEquip(eq);
  437. } else {
  438. eq.classCode = this.deviceVal[1]
  439. // 创建
  440. this.handleCreateEquip(eq);
  441. }
  442. }
  443. // 编辑当前行
  444. handleCurrentEdit(val: object) {
  445. this.deviceMsg = '编辑设备'
  446. this.currRowContent = val
  447. this.handleNext()
  448. this.dialogVisible = true
  449. }
  450. // close
  451. close() {
  452. this.next = true
  453. this.isWidth = false
  454. if (this.deviceType) {
  455. this.deviceVal = this.deviceType
  456. } else {
  457. this.deviceVal = ''
  458. }
  459. this.displayLocation = false
  460. }
  461. // 取消
  462. cancelLocation() {
  463. // @ts-ignore
  464. this.$refs.deviceGraph.cancelLocation()
  465. }
  466. // 保存
  467. saveLocation() {
  468. // @ts-ignore
  469. const data = this.$refs.deviceGraph.getLocation()
  470. if (data) {
  471. this.curEquip.bimLocation = `${data.x},${data.y},${data.z}`;
  472. this.curEquip.buildingId = data.buildingId;
  473. this.curEquip.floorId = data.floorId;
  474. }
  475. if (this.curEquip.id) {
  476. //更新
  477. this.handleUpdateEquip(this.curEquip);
  478. } else {
  479. this.curEquip.classCode = this.deviceVal[1]
  480. // 创建
  481. this.handleCreateEquip(this.curEquip);
  482. }
  483. }
  484. // 更新设备
  485. handleUpdateEquip(obj) {
  486. let pa;
  487. if (Array.isArray(obj)) {
  488. pa = {content: obj}
  489. } else {
  490. pa = {content: [obj]}
  491. }
  492. updateEquip(pa).then(res => {
  493. if (res.result == 'success') {
  494. this.$message.success('更新成功');
  495. this.dialogVisible = false;
  496. this.handleChangeDevice()
  497. }
  498. })
  499. }
  500. // 创建设备
  501. handleCreateEquip(obj: any) {
  502. let pa;
  503. if (Array.isArray(obj)) {
  504. pa = {content: obj}
  505. } else {
  506. pa = {content: [obj]}
  507. }
  508. createEquip(pa).then(res => {
  509. if (res.result == 'success') {
  510. this.$message.success('创建成功');
  511. this.dialogVisible = false;
  512. this.handleChangeDevice()
  513. }
  514. })
  515. }
  516. // 重置
  517. resetLocation() {
  518. // @ts-ignore
  519. this.$refs.deviceGraph.resetLocation()
  520. }
  521. // 返回
  522. goBack() {
  523. this.displayLocation = false
  524. this.currRowContent = this.currentRow
  525. }
  526. @Watch("deviceType", {immediate: true, deep: true})
  527. handleDeviceMsg() {
  528. this.deviceVal = this.deviceType
  529. }
  530. }
  531. </script>
  532. <style lang="scss" scoped>
  533. $margin: 12px;
  534. $border: 1px solid #E1E7EA;
  535. .align {
  536. display: flex;
  537. align-items: center;
  538. justify-content: center;
  539. flex-direction: column;
  540. flex-wrap: wrap;
  541. .text {
  542. margin-right: 150px;
  543. margin-bottom: 10px;
  544. }
  545. }
  546. .adm-device {
  547. background: #fff;
  548. padding: 12px;
  549. height: 100%;
  550. .el-dialog__header {
  551. padding: 10px;
  552. }
  553. .operation {
  554. margin: 12px 0;
  555. .adm-select {
  556. margin-right: $margin;
  557. }
  558. .adm-btn {
  559. float: right;
  560. }
  561. }
  562. .hr {
  563. background: #E1E7EA;
  564. color: #E1E7EA;
  565. width: 100%;
  566. height: 1px;
  567. margin-bottom: 16px;
  568. }
  569. .content {
  570. position: relative;
  571. height: calc(100% - 140px);
  572. .table {
  573. border-left: $border;
  574. border-right: $border;
  575. border-bottom: $border;
  576. height: calc(100% - 41px);
  577. padding: 12px;
  578. padding-bottom: 50px;
  579. .void {
  580. margin-top: 200px;
  581. }
  582. .void-title {
  583. color: #333333;
  584. line-height: 21px;
  585. font-size: 16px;
  586. }
  587. .void-tips {
  588. color: #9CA1A9;
  589. line-height: 22px;
  590. font-size: 14px;
  591. }
  592. }
  593. .adm-multi-table {
  594. }
  595. }
  596. .adm-pagination {
  597. right: 10px;
  598. position: absolute;
  599. bottom: 10px;
  600. }
  601. }
  602. </style>
  603. <style lang="scss">
  604. .adm-device {
  605. .el-dialog__header {
  606. padding: 10px;
  607. }
  608. .el-select, .el-date-editor.el-input, .el-date-editor.el-input__inner {
  609. width: 100%;
  610. }
  611. }
  612. .data-form {
  613. height: 430px;
  614. }
  615. .text {
  616. color: #000000;
  617. margin-bottom: 10px;
  618. }
  619. .el-tabs__header {
  620. margin: 0;
  621. }
  622. .el-dialog__header {
  623. border-bottom: 1px solid #D8D8D8;
  624. }
  625. .fr {
  626. float: right;
  627. }
  628. .dialog-button {
  629. float: right;
  630. margin-left: 10px;
  631. margin-top: 30px;
  632. }
  633. .el-dialog {
  634. .el-dialog__body {
  635. padding: 20px;
  636. max-height: 643px !important;
  637. min-height: 100px;
  638. overflow-y: auto;
  639. overflow-x: hidden;
  640. }
  641. }
  642. .alertTitle {
  643. overflow: hidden;
  644. line-height: 32px;
  645. button {
  646. float: right;
  647. margin-right: 40px;
  648. }
  649. }
  650. </style>