HcUpload.vue 16 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598
  1. <template>
  2. <div class="mb-1 p-2">
  3. <span class="text-orange">按住鼠标拖动文件可进行附件列表排序</span>
  4. </div>
  5. <el-upload
  6. ref="uploadRef"
  7. :accept="accept" :action="action" :before-remove="delUploadData" :before-upload="beforeUpload"
  8. :data="uploadData"
  9. :disabled="isCanuploadVal" :file-list="fileListData" :headers="getHeader()" :on-error="uploadError"
  10. :on-exceed="uploadExceed" :on-preview="uploadPreview" :on-progress="uploadprogress"
  11. :on-remove="uploadRemove" :on-success="uploadSuccess" class="hc-upload-border"
  12. :class="autoUpload === false ? 'hc-upload-border1' : 'hc-upload-border'"
  13. drag multiple
  14. :auto-upload="autoUpload"
  15. :on-change="handleFileChange"
  16. :show-file-list="false"
  17. >
  18. <!-- 使用file插槽自定义文件列表 -->
  19. <draggable
  20. v-model="fileListData"
  21. v-loading="pdfLoading"
  22. item-key="uid"
  23. handle=".drag-handle"
  24. class="file-list-container"
  25. @end="onDragEnd"
  26. >
  27. <template #item="{ element }">
  28. <div class="file-item">
  29. <HcIcon name="drag-move-2" class="drag-handle cursor-move" />
  30. <HcIcon name="file" class="file-icon" />
  31. <span class="file-name cursor-pointer" @click="previewUrl(element)">{{ element.name }}</span>
  32. <HcIcon
  33. name="close"
  34. class="float-right cursor-pointer text-red"
  35. @click.stop="handleRemove(element)"
  36. />
  37. </div>
  38. </template>
  39. </draggable>
  40. <template #trigger>
  41. <div v-loading="uploadDisabled" :element-loading-text="loadingText" class="hc-upload-loading h-full" @click.stop="beforesubmitUpload">
  42. <HcIcon name="backup" ui="text-5xl mt-4" />
  43. <div class="el-upload__text">拖动文件到这里 或 <em>点击这里选择文件</em></div>
  44. </div>
  45. </template>
  46. <template #tip>
  47. <div class="el-upload__tip" style="font-size: 14px;">
  48. {{ acceptTip }}
  49. </div>
  50. </template>
  51. </el-upload>
  52. <div class="mt-3" style="float: right;">
  53. <el-button v-if="!autoUpload" type="primary" :loading="subLoading" @click="submitUpload">
  54. 确认上传
  55. </el-button>
  56. </div>
  57. </template>
  58. <script setup>
  59. import { nextTick, onMounted, ref, watch } from 'vue'
  60. import { getHeader } from 'hc-vue3-ui'
  61. import wbsApi from '~api/data-fill/wbs'
  62. import { isFileSize } from 'js-fast-way'
  63. import { toPdfPage } from '~uti/btn-auth'
  64. import draggable from 'vuedraggable'
  65. const props = defineProps({
  66. fileList: {
  67. type: Array,
  68. default: () => ([]),
  69. },
  70. datas: {
  71. type: Object,
  72. default: () => ({}),
  73. },
  74. isCanupload:{
  75. type:Boolean,
  76. default:false,
  77. },
  78. action:{
  79. type:String,
  80. default:'#',
  81. },
  82. accept:{
  83. type:String,
  84. default:'application/vnd.openxmlformats-officedocument.spreadsheetml.sheet,application/vnd.ms-excel,application/pdf,.doc,.docx,application/msword',
  85. },
  86. acceptTip:{
  87. type:String,
  88. default:'允许格式:pdf/excel/word, 文件大小 小于 60MB',
  89. },
  90. autoUpload:{
  91. type:Boolean,
  92. default:true,
  93. },
  94. typevalue:{
  95. type:[String, Number],
  96. default:'',
  97. }, //附件类型
  98. isListFile:{
  99. type:Boolean,
  100. default:false,
  101. }, //是否列表文件
  102. })
  103. //事件
  104. const emit = defineEmits(['change', 'close'])
  105. //变量
  106. const uploadData = ref(props.datas)
  107. const fileListData = ref(props.fileList)
  108. const action = ref(props.action)
  109. const accept = ref(props.accept)
  110. const acceptTip = ref(props.acceptTip)
  111. const uploadDisabled = ref(false)
  112. const isCanuploadVal = ref(props.isCanupload)
  113. const autoUpload = ref(props.autoUpload)
  114. const typevalue = ref(props.typevalue)
  115. const isListFile = ref(props.isListFile)
  116. //监听
  117. watch(() => [
  118. props.fileList,
  119. props.datas,
  120. props.isCanupload,
  121. props.action,
  122. props.accept,
  123. props.acceptTip,
  124. props.autoUpload,
  125. props.typevalue,
  126. props.isListFile,
  127. ], ([fileList, datas, isCanupload, Action, Accept, Tip, auto, type, list]) => {
  128. uploadData.value = datas
  129. fileListData.value = fileList
  130. isCanuploadVal.value = isCanupload
  131. action.value = Action
  132. accept.value = Accept
  133. acceptTip.value = Tip
  134. autoUpload.value = auto
  135. typevalue.value = type
  136. isListFile.value = list
  137. })
  138. watch(() => [
  139. props.typevalue,
  140. props.autoUpload,
  141. ], ([ type, auto]) => {
  142. typevalue.value = type
  143. autoUpload.value = auto
  144. },
  145. { immediate: true },
  146. )
  147. watch(() => [
  148. props.typevalue,
  149. ], ([ type]) => {
  150. typevalue.value = type
  151. if (typevalue.value) {
  152. emit('change', { type: 'success' })
  153. }
  154. },
  155. { immediate: true },
  156. )
  157. // 在watch中添加对fileList的深度监听
  158. watch(() => props.fileList, (newVal) => {
  159. fileListData.value = [...newVal] // 使用新数组保证响应性
  160. }, { deep: true, immediate: true })
  161. //渲染完成
  162. onMounted(() => {
  163. beforeFileNum.value = 0
  164. finishFileNum.value = 0
  165. errorFileNum.value = 0
  166. })
  167. //上传前
  168. const beforeFileNum = ref(0)
  169. const beforeUpload = async (file) => {
  170. console.log(file, 'file上传前')
  171. if (isFileSize(file?.size, 60)) {
  172. beforeFileNum.value++
  173. // 获取当前文件的索引
  174. const fileIndex = fileListData.value.findIndex(f => f.raw === file.raw)
  175. // 设置uploadData中的sort参数
  176. uploadData.value.sort = fileIndex + 1
  177. return true
  178. } else {
  179. window?.$message?.warning('文件大小, 不能过60M!')
  180. console.log(fileListData.value, 'fileListData.value')
  181. return false
  182. }
  183. }
  184. //超出限制时
  185. const uploadExceed = () => {
  186. window?.$message?.warning('请上传 jpg/png/pdf/excel/word 的文件,文件大小 不超过60M')
  187. }
  188. const q = 1 // 假设q是固定偏移量,可以根据需要调整
  189. // 新增的处理方法
  190. // 文件变化处理
  191. const handleFileChange = async (file, fileList) => {
  192. if (!isFileSize(file?.size, 60)) {
  193. window?.$message?.warning('文件大小, 不能过60M!')
  194. return
  195. }
  196. fileListData.value = fileList.map((item, index) => ({
  197. ...item,
  198. sort: index + q, // 为每个文件添加sort字段
  199. url:'',
  200. }))
  201. fileListData.value = fileList.filter(file => {
  202. // 检查 size 属性是否存在且等于 60M,或者 size 属性不存在
  203. return (file.size !== undefined && isFileSize(file?.size, 60)) || file.size === undefined
  204. })
  205. }
  206. //上传文件前预览pdf
  207. const pdfLoading = ref(false)
  208. const getPdfUrl = async (arr) => {
  209. // 创建 FormData 对象
  210. const formData = new FormData()
  211. // // 1. 添加多个文件(后端接收的是 files[] 数组)
  212. arr.forEach((file) => {
  213. if (file.raw !== undefined) {
  214. formData.append('file', file.raw) // 确保 file.raw 是 File 对象
  215. }
  216. })
  217. pdfLoading.value = true
  218. const { error, code, msg, data } = await wbsApi.previewBussfile(formData) // 修改这里
  219. pdfLoading.value = false
  220. if (!error && code === 200) {
  221. console.log(data, 'data')
  222. return data
  223. } else {
  224. window?.$message?.error(msg || '操作失败')
  225. }
  226. }
  227. // 拖拽结束事件
  228. const onDragEnd = () => {
  229. // 更新排序号
  230. fileListData.value = fileListData.value.map((file, index) => ({
  231. ...file,
  232. sort: index + q,
  233. }))
  234. }
  235. // 手动删除文件
  236. const handleRemove = (file) => {
  237. console.log('手动删除文件', file)
  238. delUploadData(file)
  239. }
  240. //上传中
  241. const loadingText = ref('上传中...')
  242. const uploadprogress = () => {
  243. loadingText.value = '上传中...'
  244. uploadDisabled.value = true
  245. }
  246. //上传完成
  247. const finishFileNum = ref(0)
  248. const uploadSuccess = () => {
  249. finishFileNum.value++
  250. if (beforeFileNum.value === finishFileNum.value) {
  251. uploadDisabled.value = false
  252. emit('change', { type: 'success' })
  253. }
  254. }
  255. //上传失败
  256. const errorFileNum = ref(0)
  257. const uploadError = () => {
  258. errorFileNum.value++
  259. window?.$message?.error('上传失败')
  260. const num = finishFileNum.value + errorFileNum.value
  261. if (beforeFileNum.value === num) {
  262. uploadDisabled.value = false
  263. emit('change', { type: 'success' })
  264. }
  265. }
  266. //预览
  267. const uploadPreview = ({ url }) => {
  268. emit('close')
  269. toPdfPage(url)
  270. /*if (url) {
  271. window.open(url, '_blank')
  272. }*/
  273. }
  274. const previewUrl = async (item)=>{
  275. const pdfUrLArray = await getPdfUrl([item])
  276. item.url = pdfUrLArray[0]?.url || ''
  277. if (item.url) {
  278. toPdfPage(item.url)
  279. }
  280. }
  281. const uploadRef = ref(null)
  282. // 删除文件
  283. const delUploadData = async (file) => {
  284. const { id, status } = file
  285. console.log(file, 'file')
  286. if (!id || status === 'uploading') {
  287. // 如果id不存在或文件正在上传,直接删除文件
  288. const index = fileListData.value.findIndex(f => f.uid === file.uid)
  289. if (index !== -1) {
  290. fileListData.value.splice(index, 1)
  291. }
  292. uploadRef.value.abort()
  293. uploadDisabled.value = false
  294. } else {
  295. // 如果id存在且文件不在上传状态,调用接口删除文件
  296. loadingText.value = '删除中...'
  297. uploadDisabled.value = true
  298. // const { error, code, msg } = await (accept.value === 'application/pdf'
  299. const { error, code, msg } = await (!isListFile.value
  300. ? wbsApi.delTabById({ ids: id })
  301. : wbsApi.removeBussFile({ ids: id }))
  302. uploadDisabled.value = false
  303. if (!error && code === 200) {
  304. window?.$message?.success('删除成功')
  305. // 从fileListData中移除已删除的文件
  306. const index = fileListData.value.findIndex(f => f.uid === file.uid)
  307. if (index !== -1) {
  308. fileListData.value.splice(index, 1)
  309. }
  310. } else {
  311. window?.$message?.error(msg || '操作失败')
  312. }
  313. }
  314. }
  315. const uploadRemove = () => {
  316. if (fileListData.value.length <= 0) {
  317. emit('change', { type: 'del' })
  318. }
  319. }
  320. const beforesubmitUpload = () => {
  321. if (!typevalue.value && !autoUpload.value && !isListFile.value) {
  322. window.$message.warning('请先选择附件类型')
  323. return
  324. } else {
  325. const uploadInput = uploadRef.value.$el.querySelector('input[type=file]')
  326. if (uploadInput) {
  327. uploadInput.click()
  328. }
  329. }
  330. }
  331. const subLoading = ref(false)
  332. const submitUpload = async () => {
  333. if (fileListData.value.length === 0) {
  334. window.$message.warning('请先上传文件')
  335. return
  336. }
  337. // 确保所有文件都有 sort 参数
  338. fileListData.value = fileListData.value.map((file, index) => ({
  339. ...file,
  340. sort: index + q,
  341. }))
  342. // 创建 FormData 对象
  343. const formData = new FormData()
  344. // // 1. 添加多个文件(后端接收的是 files[] 数组)
  345. fileListData.value.forEach((file) => {
  346. if (file.raw !== undefined) {
  347. formData.append('file', file.raw) // 确保 file.raw 是 File 对象
  348. }
  349. })
  350. function hasFileFields(formData) {
  351. for (let [key, value] of formData.entries()) {
  352. if (value instanceof File || value instanceof Blob) {
  353. return true
  354. }
  355. }
  356. return false
  357. }
  358. if (hasFileFields(formData)) {
  359. console.log('formData 包含文件字段')
  360. subLoading.value = true
  361. if (isListFile.value) {
  362. // 2. 添加其他参数
  363. formData.append('classify', uploadData.value.classify)
  364. formData.append('pkeyId', uploadData.value.pkeyId)
  365. formData.append('nodeId', uploadData.value.nodeId)
  366. formData.append('type', 2)
  367. formData.append('contractId', uploadData.value.contractId)
  368. formData.append('projectId', uploadData.value.projectId)
  369. const { error, code, msg } = await wbsApi.addBussFile(formData) // 修改这里
  370. uploadDisabled.value = false
  371. subLoading.value = false
  372. if (!error && code === 200) {
  373. window?.$message?.success('上传成功')
  374. await sortFile()
  375. emit('change', { type: 'success' })
  376. } else if (code === 413) {
  377. window?.$message?.error('上传文件过大,请上传小于60M的文件')
  378. } else {
  379. window?.$message?.error(msg || '操作失败')
  380. }
  381. } else {
  382. // 2. 添加其他参数
  383. formData.append('classify', uploadData.value.classify)
  384. formData.append('nodeId', uploadData.value.nodeId)
  385. formData.append('type', uploadData.value.type)
  386. formData.append('contractId', uploadData.value.contractId)
  387. const { error, code, msg } = await wbsApi.addBussFileNode(formData) // 修改这里
  388. uploadDisabled.value = false
  389. subLoading.value = false
  390. if (!error && code === 200) {
  391. window?.$message?.success('上传成功')
  392. await sortFile()
  393. emit('change', { type: 'success' })
  394. } else if (code === 413) {
  395. window?.$message?.error('上传文件过大,请上传小于60M的文件')
  396. } else {
  397. window?.$message?.error(msg || '操作失败')
  398. }
  399. }
  400. } else {
  401. console.log('formData 不包含文件字段')
  402. // 处理没有文件字段的情况
  403. await sortFile()
  404. emit('change', { type: 'success' })
  405. }
  406. subLoading.value = false
  407. }
  408. //上传文件后排序
  409. const sortFile = async ()=>{
  410. if (fileListData.value.length <= 0) {
  411. window.$message.warning('请先上传文件')
  412. return
  413. }
  414. let list = []
  415. fileListData.value.forEach((file) => {
  416. list.push(
  417. file.name,
  418. )
  419. })
  420. let obj1 = {
  421. list:list,
  422. id:uploadData.value.nodeId,
  423. contractId:uploadData.value.contractId,
  424. classify:uploadData.value.classify,
  425. type:uploadData.value.type,
  426. }
  427. let obj2 = {
  428. list:list,
  429. id:uploadData.value.pkeyId,
  430. contractId:uploadData.value.contractId,
  431. classify:uploadData.value.classify,
  432. projectId:uploadData.value.projectId,
  433. type:2,
  434. }
  435. const { error, code, msg } = await (isListFile.value
  436. ? wbsApi.addFileSort(obj2)
  437. : wbsApi.addFileSort(obj1))
  438. uploadDisabled.value = false
  439. if (!error && code === 200) {
  440. // window?.$message?.success('排序成功')
  441. } else {
  442. window?.$message?.error(msg || '操作失败')
  443. }
  444. }
  445. </script>
  446. <style lang="scss">
  447. .hc-upload-border1 .el-upload-dragger{
  448. padding: 0px;
  449. }
  450. .hc-upload-border1 .el-upload-dragger .el-upload__text{
  451. padding: 40px;
  452. }
  453. </style>
  454. <style scoped>
  455. .file-list-container {
  456. margin-top: 16px;
  457. border: 1px solid #dcdfe6;
  458. border-radius: 4px;
  459. padding: 10px;
  460. background-color: #f9f9f9;
  461. }
  462. .file-item {
  463. display: flex;
  464. align-items: center;
  465. padding: 8px 12px;
  466. margin: 6px 0;
  467. background-color: #fff;
  468. border-radius: 4px;
  469. box-shadow: 0 1px 3px rgba(0, 0, 0, 0.1);
  470. transition: all 0.3s ease;
  471. }
  472. .file-item:hover {
  473. background-color: #f5f7fa;
  474. transform: translateY(-1px);
  475. }
  476. .drag-handle {
  477. margin-right: 10px;
  478. cursor: move;
  479. color: #999;
  480. }
  481. .drag-handle:hover {
  482. color: #409eff;
  483. }
  484. .file-icon {
  485. margin-right: 10px;
  486. color: #409eff;
  487. font-size: 18px;
  488. }
  489. .file-name {
  490. flex: 1;
  491. overflow: hidden;
  492. text-overflow: ellipsis;
  493. white-space: nowrap;
  494. font-size: 14px;
  495. }
  496. .file-status {
  497. margin-right: 10px;
  498. font-size: 12px;
  499. }
  500. .uploading-text {
  501. color: #e6a23c;
  502. }
  503. .success-text {
  504. color: #67c23a;
  505. }
  506. .fail-text {
  507. color: #f56c6c;
  508. }
  509. .cursor-move {
  510. cursor: move;
  511. }
  512. .hc-upload-border1 .el-upload-dragger{
  513. padding: 0px;
  514. }
  515. .hc-upload-border1 .el-upload-dragger .el-upload__text{
  516. padding: 40px;
  517. }
  518. </style>