BaseFormMeasure.vue 14 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399
  1. <template>
  2. <a-card :bordered="false" v-show="visible" class="card">
  3. <a-row :gutter="48" style="position:fixed;bottom:150px;z-index:999;display:flex; justify-content: center;width: 90%;">
  4. <a-col :md="48" :sm="48">
  5. <span>
  6. <a-button type="primary" @click="save()">保存</a-button>
  7. <a-button style="margin-left: 8px" type="default" @click="handleCancel()">返回</a-button>
  8. </span>
  9. </a-col>
  10. </a-row>
  11. <div>
  12. <div class="tables">
  13. <div style="text-align:center;padding-bottom:10px;font-size:32px;">时代思康检定表</div>
  14. <div style="display:flex; justify-content: space-between;">
  15. <div> 检定人:{{ username }}</div>
  16. </div>
  17. <table>
  18. <thead border="1px">
  19. <tr>
  20. <th width="100px">序号</th>
  21. <th width="150px">是否在库</th>
  22. <th width="150px">设备名称</th>
  23. <th width="150px">型号</th>
  24. <th width="150px">出厂编号</th>
  25. <!-- <th width="150px">是否是子设备</th>-->
  26. <th width="150px">设备位置</th>
  27. <th width="350px">选择父设备</th>
  28. <th width="150px">设备位号</th>
  29. <th width="150px">检定周期</th>
  30. <th width="150px">检定日期</th>
  31. <th width="150px">下次检定日期</th>
  32. <th width="150px">检定单号</th>
  33. <th width="150px">检定单位</th>
  34. <th width="150px">备注</th>
  35. <th width="150px">操作</th>
  36. </tr>
  37. </thead>
  38. <tbody v-if="ListForm.length>0">
  39. <tr v-for="(item,i) in ListForm" :key="item.id">
  40. <td>{{ i+1 }}</td>
  41. <td> <a-switch v-model="item.sbStatus" >
  42. <a-icon slot="checkedChildren" type="check" />
  43. <a-icon slot="unCheckedChildren" type="close" />
  44. </a-switch></td>
  45. <td>{{ item.sbName }}</td>
  46. <td>{{ item.sbModel }}</td>
  47. <td><div> {{ item.zzh }}</div></td>
  48. <!-- <td>
  49. <a-switch v-model="item.isChild" >
  50. <a-icon slot="checkedChildren" type="check" />
  51. <a-icon slot="unCheckedChildren" type="close" />
  52. </a-switch>
  53. </td>-->
  54. <td>
  55. <div v-if="item.sbStatus">{{ item.sbPositionName }}</div>
  56. <a-tree-select
  57. v-else
  58. style="width: 150px"
  59. :dropdownStyle="{ maxHeight: '400px', overflow: 'auto' }"
  60. :treeData="treeData"
  61. :treeNodeFilterProp="'title'"
  62. :showSearch="true"
  63. v-model="item.sbPositionId"
  64. @change="handleSbPosition(item)"
  65. placeholder="请选择"
  66. >
  67. </a-tree-select>
  68. </td>
  69. <td>
  70. <div v-if="!item.sbStatus" style="width:250px">
  71. <a-input
  72. disabled
  73. style="width: 60%"
  74. v-model="item.parentSbName"/>
  75. <a-button type="primary" @click="handleSbSelect(item.sbPositionId,i)">选择</a-button>
  76. </div>
  77. <div v-else>无</div>
  78. </td>
  79. <td>
  80. <div v-if="item.sbStatus"> {{ item.sbPositionNo }}</div>
  81. <div v-else style="width:250px">
  82. <a-input
  83. style="width: 60%"
  84. v-model="item.sbPositionNo"/>
  85. <a-button v-if="item.isChild === 1" type="primary" @click="handleSbNoSelect(item.sbParentId,i)">选择</a-button>
  86. </div>
  87. </td>
  88. <td><a-input v-model="item.checkPeriod" suffix="月" /></td>
  89. <td>
  90. <a-date-picker
  91. placeholder="日期"
  92. style="width: 150px"
  93. :format="BaseTool.Date.PICKER_NORM_DATE_PATTERN"
  94. v-model="item.lastDate" />
  95. </td>
  96. <td>{{ validity(item.lastDate,item.checkPeriod) }}</td>
  97. <td><a-input style="width: 150px" v-model="item.no" /></td>
  98. <td><a-input style="width: 150px" v-model="item.requirement" /></td>
  99. <td><a-input style="width: 150px" v-model="item.remark" /></td>
  100. <td>
  101. <a-button type="link" @click="uploadImg(item,i)">上传图片</a-button>
  102. <a-button type="link" @click="uploadFile(item,i)">上传文件</a-button>
  103. <a-modal v-model="uploadVisible" title="上传图片" :footer="null" @cancel="uploadVisible = false">
  104. <a-upload
  105. :action="uploadUrl"
  106. list-type="picture-card"
  107. :file-list="defaultApplicationFileList"
  108. @change="handleApplicationFileChange"
  109. @preview="handlePreview"
  110. accept="image/*"
  111. :headers="headers"
  112. >
  113. <div>
  114. <a-icon type="plus" />
  115. <div class="ant-upload-text">
  116. Upload
  117. </div>
  118. </div>
  119. </a-upload>
  120. </a-modal>
  121. <a-modal v-model="uploadFiledVisible" title="上传文件" :footer="null" @cancel="uploadFiledVisible = false">
  122. <a-upload
  123. :action="uploadUrl"
  124. :multiple="true"
  125. :file-list="defaultCheckFileList"
  126. @change="handleCheckFileChange"
  127. :headers="headers"
  128. >
  129. <a-button> <a-icon type="upload" /> 上传 </a-button>
  130. </a-upload>
  131. </a-modal>
  132. <a-modal :visible="previewVisible" :footer="null" @cancel="previewVisible = false" @ok="previewVisible = false">
  133. <img alt="example" style="width: 100%" :src="previewImage" />
  134. </a-modal>
  135. </td>
  136. </tr>
  137. </tbody>
  138. </table>
  139. </div>
  140. </div>
  141. <sb-info-select-modal ref="sbInfoSelectModal" @selected="handleSbSelectd"/>
  142. <sb-position-no-modal ref="sbPositionNoModal" @selected="handleSbNoSelectd"/>
  143. </a-card>
  144. </template>
  145. <script>
  146. import { getSbPositionTree } from '@/api/sb/position'
  147. import SbInfoSelectModal from '@/views/sb/info/modules/SbInfoSelectModal'
  148. import SbPositionNoModal from '@/views/sb/location/modules/LocationSelectModal'
  149. import { importMeasureInStore } from '@/api/sb/info'
  150. import { uploadUrl } from '@/api/upms/file'
  151. import Vue from 'vue'
  152. import { ACCESS_TOKEN } from '@/store/mutation-types'
  153. export default {
  154. name: 'BaseFillGatherTask',
  155. components: {
  156. SbInfoSelectModal,
  157. SbPositionNoModal
  158. },
  159. data () {
  160. return {
  161. uploadUrl: uploadUrl,
  162. defaultApplicationFileList: [],
  163. defaultCheckFileList: [],
  164. checkFileList: [], // 文档
  165. applicationFileList: [],
  166. headers: {
  167. Authorization: 'Bearer ' + Vue.ls.get(ACCESS_TOKEN)
  168. },
  169. visible: false,
  170. username: this.$store.getters.userInfo.username,
  171. uploadVisible: false,
  172. uploadFiledVisible: false,
  173. previewVisible: false,
  174. previewImage: '',
  175. treeData: [],
  176. sbParentOPt: 0,
  177. model: {},
  178. listPage: {
  179. pageNum: 1,
  180. pageSize: 10,
  181. total: 0
  182. },
  183. pagination: {
  184. pageNum: 1,
  185. pageSize: 10,
  186. total: 0
  187. },
  188. imgId: '',
  189. fileId: '',
  190. ListForm: [],
  191. id: '',
  192. updatesInfo: []
  193. // 下拉框map
  194. }
  195. },
  196. props: {
  197. },
  198. created () {
  199. // 下拉框map
  200. },
  201. methods: {
  202. validity (lastDate, checkPeriod) {
  203. if (lastDate === '') return ''
  204. lastDate = this.BaseTool.Date.formatter(lastDate, this.BaseTool.Date.PICKER_NORM_DATE_PATTERN).split('-')
  205. lastDate[1] = (parseInt(lastDate[1]) + parseInt(checkPeriod)) % 12
  206. lastDate[0] = +lastDate[0] + Math.floor((parseInt(lastDate[1]) + parseInt(checkPeriod)) / 12)
  207. if (lastDate[1] === 0) {
  208. lastDate[1] = 12
  209. }
  210. return lastDate.join('-')
  211. },
  212. base (record, type) {
  213. this.visible = true
  214. this.model = record
  215. this.ListForm = record.map(item => {
  216. const data = {
  217. sbName: item.name,
  218. sbModel: item.model,
  219. checkPeriod: item.checkPeriod || 0,
  220. isChild: item.isChild,
  221. parentSbName: item.parentSbName,
  222. sbParentId: item.parentId,
  223. sbPositionId: item.positionId,
  224. sbPositionName: item.positionName,
  225. no: '',
  226. lastDate: '',
  227. sbStatus: false,
  228. sbPositionNo: item.positionNo,
  229. requirement: '',
  230. remark: '',
  231. name: this.username,
  232. sbId: item.id,
  233. type,
  234. zzh: item.zzh,
  235. checkImgList: [],
  236. checkFileList: []
  237. }
  238. return data
  239. })
  240. getSbPositionTree().then(res => {
  241. this.treeData = res.data
  242. })
  243. },
  244. handleSbPosition (item) {
  245. item.parentSbName = ''
  246. item.sbParentId = ''
  247. item.sbPositionNo = ''
  248. },
  249. handleSbSelect (positionId, i) {
  250. if (positionId === '' || positionId === null) {
  251. this.$message.warning('请选择设备位置!')
  252. return
  253. }
  254. this.sbParentOPt = i
  255. this.$refs.sbInfoSelectModal.base({}, { isChild: this.DictCache.VALUE.SB_IS_CHILD.IS_PARENT, positionId })
  256. },
  257. handleSbNoSelect (sbId, i) {
  258. if (sbId === '' || sbId === null) {
  259. this.$message.warning('请选择父设备!')
  260. return
  261. }
  262. this.sbParentOPt = i
  263. this.$refs.sbPositionNoModal.base({ sbId })
  264. },
  265. handleSbSelectd (keys, rows) {
  266. console.log(keys, rows)
  267. this.ListForm[ this.sbParentOPt ].sbParentId = keys[0]
  268. this.ListForm[ this.sbParentOPt ].parentSbName = rows[0].name
  269. this.ListForm[ this.sbParentOPt ].sbPositionNo = ''
  270. },
  271. handleSbNoSelectd (keys, rows) {
  272. console.log(keys, rows)
  273. this.ListForm[ this.sbParentOPt ].sbPositionNo = rows[0].no
  274. },
  275. uploadFile (val, i) {
  276. this.uploadFiledVisible = true
  277. this.fileId = i
  278. console.log(val)
  279. this.defaultCheckFileList = this.BaseTool.UPLOAD.transImg(val.checkFileList)
  280. },
  281. uploadImg (val, i) {
  282. this.uploadVisible = true
  283. console.log(val)
  284. this.imgId = i
  285. this.defaultApplicationFileList = this.BaseTool.UPLOAD.transImg(val.checkImgList)
  286. },
  287. getBase64 (file) {
  288. return new Promise((resolve, reject) => {
  289. const reader = new FileReader()
  290. reader.readAsDataURL(file)
  291. reader.onload = () => resolve(reader.result)
  292. reader.onerror = error => reject(error)
  293. })
  294. },
  295. async handlePreview (file) {
  296. if (!file.url && !file.preview) {
  297. file.preview = await this.getBase64(file.originFileObj)
  298. }
  299. this.previewImage = file.url || file.preview
  300. this.previewVisible = true
  301. },
  302. handleApplicationFileChange (info) {
  303. this.defaultApplicationFileList = info.fileList
  304. this.applicationFileList = this.setFileList(info, 11)
  305. this.ListForm[this.imgId].checkImgList = this.applicationFileList
  306. },
  307. handleCheckFileChange (info) {
  308. this.defaultCheckFileList = info.fileList
  309. this.checkFileList = this.setFileList(info, 32)
  310. this.ListForm[this.fileId].checkFileList = this.checkFileList
  311. console.log(this.checkFileList)
  312. },
  313. setFileList (info, type) {
  314. const file = info.file
  315. const fileList = info.fileList
  316. if (file.status === 'done') {
  317. return this.BaseTool.UPLOAD.getUploadFileDTO(fileList, type)
  318. } else if (file.status === 'removed') {
  319. return this.BaseTool.UPLOAD.getUploadFileDTO(fileList, type)
  320. } else if (file.status === 'error') {
  321. this.$message.error('上传失败')
  322. return []
  323. }
  324. },
  325. save () {
  326. let status = false
  327. this.ListForm.forEach(item => {
  328. switch (true) {
  329. case item.sbStatus || (item.requirement === '' && (item.lastDate === '' || item.lastDate === null)) :
  330. break
  331. case item.requirement === '':
  332. this.$message.warning(item.sbName + '的检定单位不能为空!')
  333. status = true
  334. break
  335. case item.lastDate === '' || item.lastDate === null:
  336. this.$message.warning(item.sbName + '的检定日期不能为空!')
  337. status = true
  338. break
  339. }
  340. item.lastDate = item.lastDate !== '' && item.lastDate !== null ? this.BaseTool.Date.formatter(item.lastDate, this.BaseTool.Date.PICKER_NORM_DATE_PATTERN) : ''
  341. item.checkFileList.forEach(item => {
  342. item.id = null
  343. })
  344. item.checkImgList.forEach(item => {
  345. item.id = null
  346. })
  347. })
  348. if (status) return
  349. const params = {
  350. sbMeasureLogDTOList: this.ListForm.filter(item => !(!item.sbStatus && item.requirement === '' && item.lastDate === ''))
  351. }
  352. console.log(params)
  353. importMeasureInStore(params).then(res => {
  354. this.$message.success('检定完成!')
  355. this.handleCancel()
  356. })
  357. },
  358. handleCancel () {
  359. this.visible = false
  360. this.ListForm = []
  361. this.$emit('ok')
  362. }
  363. }
  364. }
  365. </script>
  366. <style lang="less" scoped>
  367. .tables{
  368. margin: 20px 0px;
  369. width: 100%;
  370. overflow-x:auto;
  371. table {
  372. min-width:100%;
  373. margin: 0 auto;
  374. border: 1px solid #D6D6D6;
  375. border-collapse: collapse;
  376. font-size: 16px;
  377. font-weight: 400;
  378. table-layout:fixed;
  379. }
  380. th{
  381. background: #fafafa;
  382. font-weight: 500;
  383. }
  384. th,
  385. td {
  386. border: 1px solid #D6D6D6;
  387. text-align: center;
  388. padding: 5px 10px;
  389. white-space: nowrap;
  390. }
  391. }
  392. </style>