ListItem.vue 39 KB

1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192939495969798991001011021031041051061071081091101111121131141151161171181191201211221231241251261271281291301311321331341351361371381391401411421431441451461471481491501511521531541551561571581591601611621631641651661671681691701711721731741751761771781791801811821831841851861871881891901911921931941951961971981992002012022032042052062072082092102112122132142152162172182192202212222232242252262272282292302312322332342352362372382392402412422432442452462472482492502512522532542552562572582592602612622632642652662672682692702712722732742752762772782792802812822832842852862872882892902912922932942952962972982993003013023033043053063073083093103113123133143153163173183193203213223233243253263273283293303313323333343353363373383393403413423433443453463473483493503513523533543553563573583593603613623633643653663673683693703713723733743753763773783793803813823833843853863873883893903913923933943953963973983994004014024034044054064074084094104114124134144154164174184194204214224234244254264274284294304314324334344354364374384394404414424434444454464474484494504514524534544554564574584594604614624634644654664674684694704714724734744754764774784794804814824834844854864874884894904914924934944954964974984995005015025035045055065075085095105115125135145155165175185195205215225235245255265275285295305315325335345355365375385395405415425435445455465475485495505515525535545555565575585595605615625635645655665675685695705715725735745755765775785795805815825835845855865875885895905915925935945955965975985996006016026036046056066076086096106116126136146156166176186196206216226236246256266276286296306316326336346356366376386396406416426436446456466476486496506516526536546556566576586596606616626636646656666676686696706716726736746756766776786796806816826836846856866876886896906916926936946956966976986997007017027037047057067077087097107117127137147157167177187197207217227237247257267277287297307317327337347357367377387397407417427437447457467477487497507517527537547557567577587597607617627637647657667677687697707717727737747757767777787797807817827837847857867877887897907917927937947957967977987998008018028038048058068078088098108118128138148158168178188198208218228238248258268278288298308318328338348358368378388398408418428438448458468478488498508518528538548558568578588598608618628638648658668678688698708718728738748758768778788798808818828838848858868878888898908918928938948958968978988999009019029039049059069079089099109119129139149159169179189199209219229239249259269279289299309319329339349359369379389399409419429439449459469479489499509519529539549559569579589599609619629639649659669679689699709719729739749759769779789799809819829839849859869879889899909919929939949959969979989991000100110021003100410051006100710081009101010111012101310141015101610171018101910201021102210231024102510261027102810291030103110321033103410351036103710381039104010411042104310441045104610471048104910501051105210531054105510561057105810591060106110621063106410651066106710681069107010711072107310741075107610771078107910801081108210831084108510861087108810891090109110921093109410951096109710981099110011011102110311041105110611071108110911101111111211131114111511161117111811191120112111221123112411251126112711281129113011311132113311341135113611371138113911401141114211431144114511461147114811491150115111521153115411551156115711581159116011611162116311641165116611671168116911701171117211731174117511761177117811791180118111821183118411851186118711881189119011911192119311941195119611971198119912001201120212031204120512061207120812091210121112121213
  1. <template>
  2. <div class="data-fill-list-box">
  3. <el-collapse v-model="ActiveKey" accordion @change="CollapseChange">
  4. <template v-for="(item,index) in listDatas" :key="item?.pKeyId">
  5. <el-collapse-item :name="`item-${index}-${item?.pKeyId}`" :disabled="item['isBussShow'] === 2" :id="`item-${index}-${item?.pKeyId}`">
  6. <template #title>
  7. <div class="hc-collapse-item-header">
  8. <div class="text-lg truncate item-title">{{item.nodeName}}</div>
  9. <div class="hc-extra-text-box">
  10. <HcTooltip keys="wbs_del_table" v-if="item['isCopeTab'] === 2">
  11. <el-button type="danger" plain :disabled="item['isBussShow'] === 2" @click.stop="delClick(item,index)">删除本表</el-button>
  12. </HcTooltip>
  13. <HcTooltip keys="wbs_copy_table">
  14. <el-button type="primary" plain :disabled="item['isBussShow'] === 2" @click.stop="copyClick(item,index)">复制本表</el-button>
  15. </HcTooltip>
  16. <HcTooltip keys="wbs_hide_table">
  17. <el-button type="primary" plain @click.stop="hideClick(item,index)">
  18. <template v-if="item['isBussShow'] === 1">隐藏本表</template>
  19. <template v-else>显示本表</template>
  20. </el-button>
  21. </HcTooltip>
  22. <HcTooltip keys="wbs_preview_table">
  23. <el-button type="info" plain disabled v-if="item['isBussShow'] === 2 || item['isTabPdf'] === 1">预览</el-button>
  24. <el-button type="primary" plain @click.stop="previewClick(item,index)" v-else>预览</el-button>
  25. </HcTooltip>
  26. <HcTooltip keys="wbs_upload_table">
  27. <el-button :type="item['tabFileType'] === 2?'success':'primary'" plain :disabled="item['isBussShow'] === 2" @click.stop="uploadClick(item,index)">
  28. <template v-if="item['tabFileType'] === 2">已上传</template>
  29. <template v-else>上传</template>
  30. </el-button>
  31. </HcTooltip>
  32. </div>
  33. </div>
  34. </template>
  35. <div class="data-fill-list-item-content">
  36. <div class="data-fill-table-form-box">
  37. <div class="hc-excel-table-form-view" :id="`table-form-${item?.pKeyId}`"/>
  38. <div class="hc-no-table-form" v-if="item?.isTableForm === false">
  39. <div class="table-form-no">
  40. <img :src="notableform" alt=""/>
  41. <div class="desc">暂无表单数据</div>
  42. </div>
  43. </div>
  44. </div>
  45. <div class="data-fill-table-tip-box">
  46. <div class="text-orange tip-title">
  47. <HcIcon name="error" fill ui="text-2xl"/>
  48. <span class="ml-1">提示</span>
  49. </div>
  50. <div class="text-gray-400 tip-item">1、灰色框代表可通过系统识别计算,公式自动引用,可通过公式计算少量数据,(表头数据及简单),也可只填写白色框数据</div>
  51. <div class="text-gray-400 tip-item">2、系统支持键盘中,shift + tab键向上一个填报框切换,tab向下一个填报框切换。暂不支持上下按键切换输入框</div>
  52. <div class="table-tip-foot">
  53. <div class="tip-left-btn">
  54. <HcTooltip keys="wbs_import_table">
  55. <div class="text-gray-400 dow-text">
  56. <HcIcon name="publish" ui="text-lg"/>
  57. <span class="ml-1">导入列表数据</span>
  58. </div>
  59. </HcTooltip>
  60. <HcTooltip keys="wbs_download_table">
  61. <div class="text-main dow-text">
  62. <HcIcon name="file_download" ui="text-lg"/>
  63. <span class="ml-1">下载导入模板</span>
  64. </div>
  65. </HcTooltip>
  66. </div>
  67. <div class="tip-right-btn">
  68. <HcTooltip keys="wbs_save_table">
  69. <el-button type="primary" hc-btn :disabled="item?.isTableForm === false" :loading="tableFormSaveLoading" @click="tableFormSaveClick(item,index)">
  70. <HcIcon name="save"/>
  71. <span>保存</span>
  72. </el-button>
  73. </HcTooltip>
  74. </div>
  75. </div>
  76. </div>
  77. </div>
  78. </el-collapse-item>
  79. </template>
  80. </el-collapse>
  81. </div>
  82. <!--右键菜单-->
  83. <HcContextMenu ref="contextMenuRef" :datas="tableFormMenu" @item-click="handleMenuSelect"/>
  84. <!--上传文件-->
  85. <HcDialog :show="uploadModal" title="上传文件" widths="38rem" :footer="false" @close="uploadModal = false">
  86. <HcUpload :fileList="fileListData" :datas="uploadData" @change='uploadChange' :tableTypeValue="tableTypeValue" :contractId="contractId" :baseData="baseData"/>
  87. </HcDialog>
  88. <!--插入特殊字符-->
  89. <HcDialog :show="specialModal" title="插入特殊字符" widths="600px" saveText="确认插入" @close="specialModal = false" @save="specialNodeClick">
  90. <el-form ref="specialFormRef" :model="specialFormModel" :rules="specialFormRules" label-width="0px" size="large" class="mb-6">
  91. <el-form-item prop="val" class="special-form-item">
  92. <el-input v-model="specialFormModel.val" ref="specialRef" id="specialId" placeholder="请选择特殊字符代码" clearable @blur="specialInputBlur"/>
  93. </el-form-item>
  94. </el-form>
  95. <el-row :gutter="20" style="margin: -10px;">
  96. <el-col :span="3" style="padding: 10px;" v-for="item in specialCharacters">
  97. <div class="special-box" @click="specialClick">
  98. <span class="font-EUDC" :title="`字符代码(C):${item !== 'K̅'?item.slice(2,7):'K̅'}`" v-html="item"/>
  99. </div>
  100. </el-col>
  101. </el-row>
  102. </HcDialog>
  103. <!--引用容器参数-->
  104. <HcDialog :show="vesselModal" title="引用容器参数" widths="84%" saveText="确认引用" isTable @close="vesselModalClose" @save="vesselModalSave">
  105. <div class="adding-form-dialog-box">
  106. <div class="dialog-tree-box">
  107. <el-scrollbar>
  108. <HcMenuSimple :datas="menus" :props="menuProps" :keys="menuKey" @change="menuChange"/>
  109. </el-scrollbar>
  110. </div>
  111. <div class="dialog-table-box">
  112. <div class="dialog-table">
  113. <HcTable ref="vesselTableRef" :column="vesselTableColumn" :datas="vesselTableData" :loading="vesselTableLoading" isCheck @selection-change="vesselTableSelection"/>
  114. </div>
  115. <div class="dialog-pages">
  116. <HcPages :pages="vesselTablePage" @change="vesselTablePageChange"/>
  117. </div>
  118. </div>
  119. </div>
  120. </HcDialog>
  121. <!--引用设备仪器-->
  122. <HcDialog :show="deviceModal" title="引用设备仪器" widths="84%" saveText="确认引用" isTable @close="deviceModalClose" @save="deviceModalSave">
  123. <!-- <HcTable ref="deviceTableRef" :column="deviceTableColumn" :datas="deviceTableData" :loading="deviceTableLoading" isCheck @selection-change="deviceTableSelection"/> -->
  124. <div class="adding-form-dialog-box">
  125. <div class="dialog-tree-box">
  126. <el-scrollbar>
  127. <HcMenuSimple :datas="equipmentmenus" :props="equipmentmenuProps" :keys="equipmentmenuKey" @change="equipmentmenuChange"/>
  128. </el-scrollbar>
  129. </div>
  130. <div class="dialog-table-box">
  131. <div class="dialog-table">
  132. <HcTable ref="deviceTableRef" :column="deviceTableColumn" :datas="deviceTableData" :loading="deviceTableLoading" isCheck @selection-change="deviceTableSelection"/>
  133. </div>
  134. <div class="dialog-pages">
  135. <HcPages :pages="equipmentPage" @change="equipmentTablePageChange"/>
  136. </div>
  137. </div>
  138. </div>
  139. </HcDialog>
  140. </template>
  141. <script setup>
  142. import {ref,watch,nextTick} from "vue";
  143. import {useAppStore} from "~src/store";
  144. import {useRouter, useRoute} from 'vue-router'
  145. import wbsApi from "~api/data-fill/wbs"
  146. import HcUpload from "./HcUpload.vue"
  147. import HTableForm from "~src/plugins/HTableForm"
  148. import dataApi from "~api/tentative/detect/test";
  149. import dataApi1 from "~api/tentative/parameter/container"
  150. import dataApi2 from "~api/tentative/device/approach"
  151. import {getClassList} from "~api/tentative";
  152. import notableform from '~src/assets/view/notableform.svg';
  153. import {utilsText, isType, formValidate, deepClone} from "vue-utils-plus"
  154. const router = useRouter()
  155. const useRoutes = useRoute()
  156. //路由参数
  157. const routerQuery = useRoutes?.query;
  158. const isaddType = routerQuery?.isaddType || false;
  159. //初始
  160. const props = defineProps({
  161. datas: {
  162. type: Array,
  163. default: () => ([])
  164. },
  165. status: {
  166. type: [String,Number],
  167. default: ''
  168. },
  169. baseData: {
  170. type: Object,
  171. default: () => ({})
  172. },
  173. deviceUseIds:{
  174. type: String,
  175. default: () => ('')
  176. },
  177. authBtnTabKey:{
  178. type: String,
  179. default: () => ('')
  180. },
  181. checkTableId:{
  182. type: String,
  183. default: () => ('')
  184. },
  185. tabTypeKey:{
  186. type: String,
  187. default: () => ('')
  188. },
  189. nodeIdvalue:{
  190. type: String,
  191. default: () => ('')
  192. }
  193. })
  194. const {isString, getObjNullValue, getArrValue} = isType()
  195. const {setPosInsert, setPosRange} = utilsText()
  196. const listDatas = ref(props.datas)
  197. const isStatus = ref(props.status)
  198. const baseData = ref(props.baseData)
  199. const authBtnTabKeyType = ref(props.authBtnTabKey)//所属方
  200. const useAppState = useAppStore()
  201. const projectId = ref(useAppState.getProjectId);
  202. const contractId = ref(useAppState.getContractId);
  203. const tabTypeKeyInfo=ref(props.tabTypeKey)
  204. const nodeIdvaluedata=ref(props.nodeIdvalue)
  205. //监听
  206. watch(() => [
  207. props.datas,
  208. props.tabTypeKey
  209. ], ([datas,TabTypeKey]) => {
  210. listDatas.value = datas
  211. tabTypeKeyInfo.value=TabTypeKey
  212. setFormDataNum(datas)
  213. })
  214. //监听
  215. watch(() => [
  216. props.status,
  217. props.baseData,
  218. props.nodeIdvalue
  219. ], ([val, base,NodeIdvalue]) => {
  220. //1 未填报,2待上报,3已上报
  221. isStatus.value = val
  222. baseData.value = base
  223. nodeIdvaluedata.value=NodeIdvalue
  224. })
  225. //渲染完成
  226. nextTick(() => {
  227. setFormDataNum(props.datas)
  228. })
  229. //获取pKeyId
  230. const getValString = (val) => {
  231. return val ? val + '' : ''
  232. }
  233. //获取表单初始数据
  234. const getFormDataInit = ({projectId, cid, pKeyId}) => {
  235. const { nodeId, contractId } = baseData.value
  236. return {
  237. projectId: projectId,
  238. contractId: cid || contractId,
  239. pkeyId: getValString(pKeyId),
  240. nodeId: nodeId
  241. }
  242. }
  243. //设置表单对象的数量
  244. const formData = ref([])
  245. const setFormDataNum = (datas) => {
  246. ActiveKey.value = ''
  247. let newArr = [];
  248. for (let i = 0; i < datas.length; i++) {
  249. newArr.push({
  250. ...getFormDataInit(datas[i]),
  251. isCollapseLoad: false,
  252. })
  253. }
  254. formData.value = newArr
  255. }
  256. //展开事件
  257. const ActiveKey = ref('')
  258. const formKeyIds = ref('')
  259. const CollapseChange = async (name) => {
  260. ActiveKey.value = name
  261. const names = name ? name.split('-') : []
  262. formData.value.forEach((changeitem)=>{
  263. if(changeitem.pkeyId===names[2]){
  264. changeitem.isCollapseLoad=true
  265. }
  266. })
  267. if (names.length > 0) {
  268. getOffsetTop(name)
  269. const index = names[1]
  270. const item = listDatas.value[index]
  271. emit('upcheckTableId', item.id)
  272. formKeyIds.value = getValString(item.pKeyId)
  273. console.log(nodeIdvaluedata.value,'nodeIdvaluedata.value');
  274. console.log(item,'item');
  275. if (!item.isTableFormRender) {
  276. await getBussDataInfo(item, index)
  277. }
  278. //渲染表单
  279. await getExcelHtml(item,index)
  280. } else {
  281. // await getExcelHtml(item,index)
  282. getOffsetTop()
  283. formKeyIds.value = ''
  284. }
  285. }
  286. //获取模板标签数据
  287. const formRegExpJson = ref({})
  288. const getExcelHtml = async (item,index) => {
  289. console.log('获取模板数据');
  290. const pkeyIds = getValString(item.pKeyId)
  291. if (pkeyIds) {
  292. const { id } = baseData.value
  293. console.log(nodeIdvaluedata.value,'nodeIdvaluedata.value');
  294. const {error, code, data} = await dataApi.getExcelHtml({
  295. id: id||nodeIdvaluedata.value,
  296. primaryKeyId: pkeyIds
  297. }, false)
  298. const resData = isString(data) ? data || '' : ''
  299. if (!error && code === 200 && resData) {
  300. item.isTableForm = true
  301. //渲染表单
  302. HTableForm.createForm({
  303. template: resData,
  304. tableForm: formData.value[index],
  305. appId: `#table-form-${pkeyIds}`,
  306. onRight: (event, KeyName) => {
  307. onRightClick(event, KeyName, index)
  308. },
  309. //表单正则效验
  310. onBlur: (event, key, reg, val, msg) => {
  311. setTableFormBlurReg(pkeyIds, event, key, reg, val, msg, item, index)
  312. }
  313. })
  314. item.isTableFormRender = true
  315. item.isRenderTableForm = true
  316. } else {
  317. item.isTableForm = false
  318. item.isRenderTableForm = true
  319. window?.$message?.warning('暂无表单')
  320. }
  321. } else {
  322. item.isTableForm = false
  323. item.isRenderTableForm = false
  324. window?.$message?.warning('pkeyId为空')
  325. }
  326. }
  327. //正则效验
  328. const setTableFormBlurReg = (pkeyId, event, key, reg, val, msg, item, index) => {
  329. const dom = document.getElementById(key)?.parentElement ?? ''
  330. if (dom) {
  331. if (val && reg) {
  332. let regx = new RegExp(reg);
  333. let state = regx.test(val);
  334. if (state) {
  335. delete formRegExpJson.value[pkeyId]
  336. dom.style = ''
  337. } else {
  338. formRegExpJson.value[pkeyId] = {key, reg, val, msg, state, nodeName: item.nodeName, itemId: `item-${index}-${item?.pKeyId}`}
  339. dom.style = '--el-input-border-color: #fe0000; box-shadow: 0 0 0 2px #fe0000 inset;'
  340. window?.$message?.warning(msg)
  341. }
  342. } else {
  343. delete formRegExpJson.value[pkeyId]
  344. dom.style = ''
  345. }
  346. }
  347. }
  348. //获取已填写的数据
  349. const getBussDataInfo = async (item, index) => {
  350. const pkeyIds = getValString(item.pKeyId)
  351. if (pkeyIds) {
  352. const { id } = baseData.value
  353. const {error, code, data} = await dataApi.getBussDataInfo({
  354. id: id||nodeIdvaluedata.value,
  355. pkeyId: pkeyIds
  356. }, false)
  357. data.forEach((item1)=>{
  358. const resData = getObjNullValue(item1)
  359. if (!error && code === 200 && resData) {
  360. HTableForm.setPickerKey(resData)
  361. const InitObj = getFormDataInit(item) //有数据,关联数据
  362. formData.value[index] = {
  363. ...resData, ...InitObj,
  364. isCollapseLoad: true
  365. }
  366. }else {
  367. formData.value[index] = {
  368. ...getFormDataInit(item),
  369. isCollapseLoad: true
  370. }
  371. }
  372. })
  373. // if (!error && code === 200 && resData) {
  374. // HTableForm.setPickerKey(resData)
  375. // const InitObj = getFormDataInit(item) //有数据,关联数据
  376. // formData.value[index] = {
  377. // ...resData, ...InitObj,
  378. // isCollapseLoad: true
  379. // }
  380. // } else {
  381. // formData.value[index] = {
  382. // ...getFormDataInit(item),
  383. // isCollapseLoad: true
  384. // }
  385. // }
  386. } else {
  387. window?.$message?.warning('pkeyId为空')
  388. }
  389. }
  390. const toBackClick = () => {
  391. router.push({
  392. path: '/tentative/detect/test',
  393. query: {}
  394. })
  395. }
  396. //单个保存
  397. const tableFormSaveLoading = ref(false)
  398. const tableFormSaveClick = async (item,index) => {
  399. if (isStatus.value !== '3') {
  400. if(tabTypeKeyInfo.value==='2'&&baseData.value['detectionResult']===''){
  401. window.$message?.warning('请选择是否合格')
  402. }else{
  403. const res = await saveExcelBussData(item,index)
  404. if (res) {
  405. if(!isaddType){
  406. await getBussPdfInfo(item)
  407. }else{
  408. //返回
  409. toBackClick()
  410. }
  411. renewData()
  412. }
  413. }
  414. } else {
  415. window?.$message?.warning('已上报的资料,不允许保存。')
  416. }
  417. }
  418. //保存表单数据
  419. const saveExcelBussData = async (item, index, showTip = true) => {
  420. if (!getObjNullValue(formRegExpJson.value)) {
  421. tableFormSaveLoading.value = true
  422. const InitObj = getFormDataInit(item)
  423. const {error, code} = await dataApi.saveExcelBussData({
  424. ...baseData.value,
  425. isBatchSave:0,
  426. dataInfo: {
  427. orderList: [{...formData.value[index], ...InitObj}]
  428. }
  429. })
  430. //处理数据
  431. tableFormSaveLoading.value = false
  432. if (!error && code === 200) {
  433. if(showTip) window?.$message?.success('保存成功')
  434. return true
  435. } else {
  436. return false
  437. }
  438. } else {
  439. window?.$message?.warning('请先修改完红色输入框的数据')
  440. return false
  441. }
  442. }
  443. //预览PDF
  444. const getBussPdfInfo = async ({pKeyId}, showTip = true) => {
  445. const pkeyIds = getValString(pKeyId)
  446. if (pkeyIds) {
  447. const { id } = baseData.value
  448. const {error, code, data} = await dataApi.getBussPdf({
  449. id: id,
  450. pKeyId: pkeyIds
  451. },false)
  452. if (!error && code === 200) {
  453. if (data) {
  454. window.open(data, '_blank')
  455. } else if(showTip) {
  456. window?.$message?.warning('PDF错误')
  457. }
  458. } else {
  459. if(showTip) {
  460. window?.$message?.warning(data.msg || '获取PDF失败')
  461. }
  462. }
  463. } else {
  464. window?.$message?.warning('pkeyId为空')
  465. }
  466. }
  467. //删除本表
  468. const delClick = async ({pKeyId}) => {
  469. const pkeyIds = getValString(pKeyId)
  470. if (pkeyIds) {
  471. const { id } = baseData.value
  472. const {error, code} = await dataApi.removeBussTabInfo({
  473. id: id,
  474. pKeyId: pkeyIds
  475. })
  476. if (!error && code === 200) {
  477. window?.$message?.success('操作成功')
  478. renewData()
  479. }
  480. } else {
  481. window?.$message?.warning('pkeyId为空')
  482. }
  483. }
  484. //复制本表
  485. const copyClick = async (item,index) => {
  486. const pkeyIds = getValString(item.pKeyId)
  487. if (pkeyIds) {
  488. if (isStatus.value !== '3') {
  489. if (!item.isRenderTableForm) {
  490. await copeBussTab(pkeyIds)
  491. } else if (!item.isTableForm) {
  492. window?.$message?.warning('暂无表单数据')
  493. } else if (item.isRenderTableForm) {
  494. const res = await saveExcelBussData(item,index,false)
  495. if (res) {
  496. await copeBussTab(pkeyIds)
  497. } else {
  498. window?.$message?.warning('复制本表操作失败')
  499. }
  500. } else {
  501. window?.$message?.warning(`数据异常了, isRenderTableForm: ${item.isRenderTableForm}, isTableForm: ${item.isTableForm}, pKeyId:${pkeyIds}`)
  502. }
  503. } else {
  504. window?.$message?.warning('已上报的资料,不允许复制')
  505. }
  506. } else {
  507. window?.$message?.warning('pkeyId为空')
  508. }
  509. }
  510. const copeBussTab = async (pkeyIds) => {
  511. const { id } = baseData.value
  512. const {error, code} = await dataApi.copyBussTab({
  513. id: id,
  514. pKeyId: pkeyIds
  515. })
  516. if (!error && code === 200) {
  517. window?.$message?.success('操作成功')
  518. renewData()
  519. }
  520. }
  521. //隐藏本表
  522. const hideClick = async ({pKeyId, isBussShow}) => {
  523. const pkeyIds = getValString(pKeyId)
  524. if (pkeyIds) {
  525. if (isStatus.value !== '3') {
  526. const { id } = baseData.value
  527. const isBussShows = isBussShow === 2 ? 1 : 2 //状态(1显示 2隐藏)
  528. const {error, code} = await dataApi.showBussTab({
  529. id: id,
  530. pKeyId: pkeyIds,
  531. status: isBussShows
  532. })
  533. if (!error && code === 200) {
  534. window?.$message?.success('操作成功')
  535. renewData()
  536. }
  537. } else {
  538. window?.$message?.warning('已上报的资料,不允许隐藏')
  539. }
  540. } else {
  541. window?.$message?.warning('pkeyId为空')
  542. }
  543. }
  544. //预览
  545. const previewClick = async (item,index) => {
  546. await getBussPdfInfo(item)
  547. }
  548. //上传变量
  549. const uploadModal = ref(false)
  550. const fileListData = ref([]);
  551. const uploadData = ref({})
  552. const tableTypeValue=ref('')
  553. //上传被点击
  554. const uploadClick = (item,index) => {
  555. const pkeyIds = item.pKeyId ? item.pKeyId + '' : ''
  556. const { id } = baseData.value
  557. const tableType = item.tableType ? item.tableType : ''
  558. tableTypeValue.value= item.tableType ? item.tableType : ''
  559. const classify = authBtnTabKeyType.value
  560. const keyName = `item-${index}-${pkeyIds}`
  561. if (pkeyIds) {
  562. if (isStatus.value !== '3' && item.isTableForm) {
  563. uploadModal.value = true
  564. uploadData.value = getFormDataInit(item,pkeyIds)
  565. uploadData.value.tableType =tableType
  566. uploadData.value.classify =classify
  567. uploadData.value.id =id
  568. console.log( uploadData.value,' uploadData.value');
  569. //获取文件列表
  570. getBussFileList(pkeyIds)
  571. } else if (!item.isRenderTableForm) {
  572. CollapseChange(keyName)
  573. window?.$message?.warning('请再次点击上传')
  574. } else if (!item.isTableForm) {
  575. window?.$message?.warning('暂无表单数据')
  576. } else {
  577. window?.$message?.warning('已上报的资料,不允许上传')
  578. }
  579. } else {
  580. window?.$message?.warning('pkeyId为空')
  581. }
  582. }
  583. //获取文件列表
  584. const getBussFileList = async (pkeyId) => {
  585. const { id } = baseData.value
  586. const {error, code, data} = await wbsApi.getBussFileList1({
  587. pkeyid: pkeyId,
  588. id:id
  589. })
  590. if (!error && code === 200) {
  591. fileListData.value = getArrValue(data)
  592. } else {
  593. fileListData.value = []
  594. }
  595. }
  596. //上传文件
  597. const uploadChange = async ({type}) => {
  598. if(type === 'success') {
  599. uploadModal.value = false
  600. renewData()
  601. } else if (type === 'del') {
  602. renewData()
  603. }
  604. }
  605. //相关变量
  606. const tableFormItemNode = ref({})
  607. //菜单数据
  608. const tableFormMenu = ref([
  609. {label: '容器参数', key: "vessel"},
  610. {label: '引用设备仪器', key: "device"},
  611. {label: '插入特殊字符', key: "special"},
  612. ])
  613. //鼠标右键事件
  614. const contextMenuRef = ref(null)
  615. const onRightClick = (event, KeyName, index) => {
  616. //取光标位置
  617. const specialDom = document.getElementById(KeyName + "")
  618. const startPos = specialDom?.selectionStart || 0
  619. const endPos = specialDom?.selectionEnd || 0
  620. //存储临时信息
  621. tableFormItemNode.value = {KeyName, index, startPos, endPos, pkeyId: formKeyIds.value}
  622. contextMenuRef.value?.showMenu(event) //展开菜单
  623. }
  624. //鼠标右键菜单被点击
  625. const handleMenuSelect = ({key}) => {
  626. if (key === 'vessel') {
  627. vesselModal.value = true
  628. getMenusData()
  629. } else if (key === 'special') {
  630. specialModalShow()
  631. } else if (key === 'device') {
  632. deviceModal.value = true
  633. getequipmentMenusData()
  634. }
  635. }
  636. //引用容器参数
  637. const vesselModal = ref(false)
  638. const vesselTableRef = ref(null)
  639. //引用容器参数菜单数据
  640. const menuProps = {
  641. key: 'id',
  642. label: 'containerName',
  643. }
  644. //引用容器参数菜单数据
  645. const equipmentmenuProps = {
  646. key: 'id',
  647. label: 'className',
  648. }
  649. const menus = ref([]);
  650. const getMenusData = async () => {
  651. const { data } = await dataApi1.queryClassification({
  652. projectId: projectId.value,
  653. contractId: contractId.value
  654. })
  655. const arr = getArrValue(data)
  656. menus.value = arr
  657. if (arr.length > 0) {
  658. const item = arr[0]
  659. console.log(item,'item');
  660. menuItem.value = item
  661. menuKey.value = item?.id
  662. getVesselTableData()
  663. }
  664. }
  665. //菜单被点击
  666. const menuKey = ref()
  667. const menuItem = ref({})
  668. const menuChange = (item) => {
  669. menuItem.value = item
  670. menuKey.value = item?.id
  671. getVesselTableData()
  672. }
  673. //表格数据
  674. const vesselTableColumn = ref([
  675. {key:'key_1', name: '容器编号'},
  676. ])
  677. const vesselTableData = ref([])
  678. const vesselTablePage = ref({current: 1, size: 20, total: 0})
  679. const vesselTablePageChange = ({current, size}) => {
  680. vesselTablePage.value.current = current
  681. vesselTablePage.value.size = size
  682. getVesselTableData()
  683. }
  684. //获取表格数据
  685. const vesselTableLoading = ref(false)
  686. const getVesselTableData = async () => {
  687. const {id, fieldList} = menuItem.value
  688. const fieldLists = getArrValue(fieldList)
  689. if (fieldLists.length > 0) {
  690. vesselTableLoading.value = true
  691. const { error, code, data } = await dataApi1.queryPage({
  692. projectId: projectId.value,
  693. contractId: contractId.value,
  694. containerId: menuKey.value,
  695. fieldKey:fieldList[0].fieldKey,
  696. size:vesselTablePage.value.size,
  697. current:vesselTablePage.value.current,
  698. })
  699. //处理数据
  700. vesselTableLoading.value = false
  701. if (!error && code === 200) {
  702. vesselTableData.value = getArrValue(data['records'])
  703. vesselTablePage.value.total = data.total || 0
  704. } else {
  705. vesselTableData.value = []
  706. vesselTablePage.value.total = 0
  707. }
  708. }
  709. }
  710. //多选
  711. const vesselTableKeys = ref([]);
  712. const vesselTableSelection = (rows) => {
  713. vesselTableKeys.value = rows
  714. }
  715. //确认引用
  716. const vesselModalSave = () => {
  717. if (vesselTableKeys.value.length>0) {
  718. const item = tableFormItemNode.value
  719. const form = formData.value[item.index]
  720. const val =[]
  721. vesselTableKeys.value.forEach((item)=>{
  722. val.push(item.key_1)
  723. })
  724. const newval=val.join('、')
  725. formData.value[item.index][item.KeyName] = setPosInsert(item.startPos, item.endPos, form[item.KeyName], newval)
  726. vesselModal.value = false
  727. let posVal = item.startPos + newval.length;
  728. nextTick(() => {
  729. setPosRange(item.KeyName, posVal)
  730. })
  731. }else{
  732. window?.$message?.warning('请先选择引用容器参数')
  733. }
  734. }
  735. //关闭
  736. const vesselModalClose = () => {
  737. vesselModal.value = false
  738. }
  739. const equipmentmenus = ref([]);
  740. const getequipmentMenusData = async () => {
  741. const { data } = await getClassList({
  742. projectId: projectId.value,
  743. contractId: contractId.value
  744. })
  745. const arr = getArrValue(data)
  746. equipmentmenus.value = arr
  747. if (arr.length > 0) {
  748. const item = arr[0]
  749. equipmentmenuItem.value = item
  750. equipmentmenuKey.value = item?.id
  751. getDeviceTableData()
  752. }
  753. }
  754. //菜单被点击
  755. const equipmentmenuKey = ref()
  756. const equipmentmenuItem = ref({})
  757. const equipmentmenuChange = (item) => {
  758. equipmentmenuItem.value = item
  759. equipmentmenuKey.value = item?.id
  760. getDeviceTableData()
  761. }
  762. //引用设备仪器
  763. const deviceModal = ref(false)
  764. const deviceTableRef = ref(null)
  765. //表格数据
  766. const deviceTableColumn = ref([
  767. {key:'deviceNumber', name: '设备编号'},
  768. {key:'deviceName', name: '设备仪器名称'},
  769. ])
  770. const deviceTableData = ref([])
  771. const equipmentPage = ref({current: 1, size: 20, total: 0})
  772. const equipmentTablePageChange = ({current, size}) => {
  773. equipmentPage.value.current = current
  774. equipmentPage.value.size = size
  775. getDeviceTableData()
  776. }
  777. //获取表格数据
  778. const deviceTableLoading = ref(false)
  779. const getDeviceTableData =async () => {
  780. deviceTableLoading.value = true
  781. const { error, code, data } = await dataApi2.queryPage({
  782. projectId: projectId.value,
  783. contractId: contractId.value,
  784. deviceClassId: equipmentmenuKey.value,
  785. size:equipmentPage.value.size,
  786. current:equipmentPage.value.current,
  787. })
  788. //处理数据
  789. deviceTableLoading.value = false
  790. if (!error && code === 200) {
  791. deviceTableData.value = getArrValue(data['records'])
  792. vesselTablePage.value.total = data.total || 0
  793. } else {
  794. deviceTableData.value = []
  795. equipmentPage.value.total = 0
  796. }
  797. }
  798. //多选
  799. const deviceTableKeys = ref([]);
  800. const deviceTableSelection = (rows) => {
  801. deviceTableKeys.value = rows
  802. }
  803. //确认引用
  804. const deviceModalSave = () => {
  805. if (deviceTableKeys.value.length>0) {
  806. const item = tableFormItemNode.value
  807. const form = formData.value[item.index]
  808. const val =[]
  809. const idarr=[]
  810. const listr=[]
  811. deviceTableKeys.value.forEach((item)=>{
  812. val.push(item.deviceNumber)
  813. idarr.push(item.id)
  814. listr.push(item.deviceNumber+"_"+item.deviceName)
  815. })
  816. const newval=listr.join('、');
  817. const idval=idarr.join(',')
  818. formData.value[item.index][item.KeyName] = setPosInsert(item.startPos, item.endPos, form[item.KeyName], newval)
  819. vesselModal.value = false
  820. let posVal = item.startPos + newval.length;
  821. nextTick(() => {
  822. setPosRange(item.KeyName, posVal)
  823. })
  824. deviceModal.value = false;
  825. emit('updeviceUseIds', idval)
  826. }else{
  827. window?.$message?.warning('请先选择引用容器设备')
  828. }
  829. }
  830. //关闭
  831. const deviceModalClose = () => {
  832. deviceModal.value = false
  833. }
  834. //插入特殊字符
  835. const specialModal = ref(false)
  836. const specialCharacters = ref([
  837. '&#57344;', "&#57345;", "&#57346;", "&#57347;", '&#8804;', '&#8805;', '&#8451;',
  838. '&#9312;', '&#9313;', '&#9314;', '&#9315;', '&#9316;', '&#9317;', '&#9318;', '&#9319;', '&#9320;', '&#9321;', '&#9322;', '&#9323;',
  839. '&#9324;', '&#9325;', '&#9326;', '&#9327;', '&#9328;', '&#9329;', '&#9330;', '&#9331;',
  840. "&#8544;", "&#8545;", "&#8546;", "&#8547;", "&#8548;", "&#8549;", "&#8550;", "&#8551;", "&#8552;", "&#8553;", "&#8554;", "&#8555;","K̅"
  841. ])
  842. //输入框验证
  843. const specialFormRef = ref(null)
  844. const specialFormModel = ref({val: ''})
  845. const specialFormRules = {
  846. val: {
  847. required: true,
  848. trigger: "blur",
  849. message: "请选择特殊字符代码"
  850. }
  851. }
  852. //显示插入特殊字符
  853. const specialRef = ref(null)
  854. const specialModalShow = () => {
  855. specialFormModel.value.val = ''
  856. specialModal.value = true
  857. nextTick(() => {
  858. specialRef.value?.focus();
  859. })
  860. }
  861. //失去焦点
  862. const specialPos = ref({start: 0, end: 0})
  863. const specialInputBlur = (e) => {
  864. specialPos.value = {
  865. start: e?.target?.selectionStart || 0,
  866. end: e?.target?.selectionEnd || 0
  867. }
  868. }
  869. //点击符号
  870. const specialClick = (event) => {
  871. const text = event?.target?.innerText ?? ''
  872. const start = specialPos.value.start
  873. const end = specialPos.value.end
  874. const form = specialFormModel.value.val
  875. specialFormModel.value.val = setPosInsert(start, end, form, text)
  876. specialRef.value?.focus();
  877. let posVal = start + text.length;
  878. nextTick(() => {
  879. setPosRange('specialId', posVal)
  880. })
  881. }
  882. //确认插入
  883. const specialNodeClick = async () => {
  884. const res = await formValidate(specialFormRef.value)
  885. if (res) {
  886. const item = tableFormItemNode.value
  887. const form = formData.value[item.index]
  888. const val = specialFormModel.value.val ?? ''
  889. formData.value[item.index][item.KeyName] = setPosInsert(item.startPos, item.endPos, form[item.KeyName], val)
  890. specialModal.value = false
  891. specialRef.value?.focus();
  892. let posVal = item.startPos + val.length;
  893. nextTick(() => {
  894. setPosRange(item.KeyName, posVal)
  895. })
  896. }
  897. }
  898. //事件
  899. const emit = defineEmits(['renew','offsetTop','updeviceUseIds','upcheckTableId'])
  900. //被点击
  901. const getOffsetTop = (key = '') => {
  902. if (key) {
  903. const dom = document.getElementById(key)
  904. emit('offsetTop', dom.offsetTop)
  905. } else {
  906. emit('offsetTop', 0)
  907. }
  908. }
  909. //通知数据更新
  910. const renewData = () => {
  911. emit('renew')
  912. ActiveKey.value = ''
  913. }
  914. //获取表单数据
  915. const getFormData = () => {
  916. const formArr = formData.value;
  917. console.log( formData.value,' formData.value');
  918. // return formArr.filter(({pkeyId, isCollapseLoad}) => {
  919. // return (pkeyId ?? '') !== '' && isCollapseLoad;
  920. // })
  921. return formArr.filter((item) => {
  922. // return (pkeyId ?? '') !== '' && isCollapseLoad;
  923. return item
  924. })
  925. }
  926. //获取表单效验数据
  927. const getFormRegExpJson = () => {
  928. return deepClone(formRegExpJson.value);
  929. }
  930. //获取当前展开项
  931. const getActiveKey = () => {
  932. return ActiveKey.value;
  933. }
  934. //设置当前展开项
  935. const setActiveKey = (key) => {
  936. return ActiveKey.value = key;
  937. }
  938. // 暴露出去
  939. defineExpose({
  940. getFormData,
  941. getFormRegExpJson,
  942. getActiveKey,
  943. setActiveKey
  944. })
  945. </script>
  946. <style lang="scss" scoped>
  947. .data-fill-list-box {
  948. position: relative;
  949. //margin-bottom: 25%;
  950. .hc-collapse-item-header {
  951. flex: 1;
  952. position: relative;
  953. margin-left: 46px;
  954. display: flex;
  955. align-items: center;
  956. .item-title {
  957. flex: 1;
  958. position: relative;
  959. user-select: none;
  960. color: #50545E;
  961. font-size: 16px;
  962. font-weight: 400;
  963. cursor: pointer;
  964. }
  965. .hc-extra-text-box {
  966. position: relative;
  967. padding-right: 24px;
  968. }
  969. }
  970. .data-fill-list-item-content {
  971. position: relative;
  972. display: flex;
  973. height: calc(100vh - 428px);
  974. .data-fill-table-form-box {
  975. position: relative;
  976. padding: 24px 20px;
  977. height: 100%;
  978. overflow: auto;
  979. flex: 1;
  980. .hc-no-table-form {
  981. position: relative;
  982. height: 100%;
  983. display: flex;
  984. justify-content: center;
  985. align-items: center;
  986. .table-form-no {
  987. position: relative;
  988. img {
  989. width: 350px;
  990. }
  991. .desc {
  992. text-align: center;
  993. font-size: 20px;
  994. color: #aaa;
  995. }
  996. }
  997. }
  998. }
  999. .data-fill-table-tip-box {
  1000. width: 240px;
  1001. position: relative;
  1002. border-left: 1px solid #E9E9E9;
  1003. padding: 20px 15px 80px;
  1004. .tip-title {
  1005. font-size: 16px;
  1006. margin-bottom: 10px;
  1007. display: flex;
  1008. align-items: center;
  1009. }
  1010. .tip-item {
  1011. margin-bottom: 20px;
  1012. }
  1013. .table-tip-foot {
  1014. position: absolute;
  1015. bottom: 15px;
  1016. right: 0;
  1017. left: 0;
  1018. display: flex;
  1019. align-items: center;
  1020. padding: 0 15px;
  1021. .tip-left-btn {
  1022. flex: 1;
  1023. .dow-text {
  1024. cursor: pointer;
  1025. display: flex;
  1026. align-items: center;
  1027. }
  1028. }
  1029. }
  1030. }
  1031. }
  1032. }
  1033. .special-box {
  1034. position: relative;
  1035. display: flex;
  1036. justify-content: center;
  1037. align-items: center;
  1038. border: 1px solid #eee;
  1039. border-radius: 3px;
  1040. height: 52px;
  1041. width: 52px;
  1042. cursor: pointer;
  1043. user-select: none;
  1044. transition: color .3s, background-color .3s;
  1045. &:hover {
  1046. color: var(--el-color-primary);
  1047. background-color: var(--el-color-primary-light-8);
  1048. }
  1049. .font-EUDC {
  1050. font-size: 22px;
  1051. }
  1052. }
  1053. </style>
  1054. <style lang="scss">
  1055. .data-fill-list-box {
  1056. .el-collapse {
  1057. --el-collapse-header-height: 60px;
  1058. border: 0;
  1059. .el-collapse-item {
  1060. margin: 0 0 16px;
  1061. background-color: #f1f5f8;
  1062. border: 1px solid #E9E9E9;
  1063. border-radius: 4px;
  1064. }
  1065. .el-collapse-item__header {
  1066. background-color: transparent;
  1067. font-weight: 400;
  1068. border-bottom: 0;
  1069. cursor: default;
  1070. font-size: 14px;
  1071. .el-collapse-item__arrow {
  1072. position: absolute;
  1073. color: #50545E;
  1074. cursor: pointer;
  1075. left: 20px;
  1076. margin: 0;
  1077. }
  1078. }
  1079. .el-collapse-item.is-active .el-collapse-item__header.is-active {
  1080. background-color: #E7EEF4;
  1081. }
  1082. .el-collapse-item__wrap {
  1083. background-color: transparent;
  1084. border-bottom: 0;
  1085. .el-collapse-item__content {
  1086. position: relative;
  1087. padding-bottom: 0;
  1088. font-size: 14px;
  1089. color: #50545E;
  1090. line-height: initial;
  1091. }
  1092. }
  1093. }
  1094. }
  1095. //插入特殊字符弹窗的输入框
  1096. .data-fill-list-box .data-fill-table-form-box td,
  1097. .data-fill-list-box .data-fill-table-form-box td .el-input .el-input__wrapper .el-input__inner,
  1098. .el-form-item.special-form-item .el-form-item__content .el-input .el-input__wrapper .el-input__inner {
  1099. font-family: "EUDC", 宋体, v-sans, system-ui, -apple-system, BlinkMacSystemFont, "Segoe UI", sans-serif, "Apple Color Emoji", "Segoe UI Emoji", "Segoe UI Symbol";
  1100. }
  1101. //引用容器参数弹窗
  1102. //关联试验数据
  1103. .adding-form-dialog-box {
  1104. position: relative;
  1105. height: 100%;
  1106. display: flex;
  1107. .dialog-tree-box {
  1108. position: relative;
  1109. border-right: 1px solid #EEEEEE;
  1110. width: 500px;
  1111. height: 100%
  1112. }
  1113. .dialog-table-box {
  1114. position: relative;
  1115. flex: 1;
  1116. height: 100%;
  1117. padding: 18px;
  1118. .dialog-search {
  1119. position: relative;
  1120. display: flex;
  1121. }
  1122. .dialog-table {
  1123. position: relative;
  1124. height: calc(100% - 68px);
  1125. padding: 18px 0;
  1126. }
  1127. .dialog-pages {
  1128. position: relative;
  1129. }
  1130. }
  1131. }
  1132. </style>