upload.vue 18 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526
  1. <template>
  2. <div class="cus-form-column-upload-com"
  3. :class="{
  4. 'cus-form-column-upload-com_view': isViewCpt,
  5. 'cus-form-column-upload-com_list': layout === 'list',
  6. 'cus-form-column-upload-com_card': layout === 'card',
  7. 'cus-form-column-upload-com_limit-no-upload': isLimitCpt && limitNoUpload !== false
  8. }"
  9. v-loading="loading"
  10. :element-loading-background="elementLoadingBackground">
  11. <el-upload
  12. class="el-upload-com"
  13. ref="ref_upload"
  14. v-bind="$attrs"
  15. :file-list="paramVal"
  16. action="#"
  17. :list-type="layout === 'card' ? 'picture-card' : 'text'"
  18. :http-request="handleRequest"
  19. :before-upload="handleBeforeUpload"
  20. :accept="acceptTypeCpt"
  21. :limit="limit > 0 ? limit : null"
  22. :on-remove="handleRemove"
  23. :disabled="isLimitCpt"
  24. >
  25. <el-tooltip :content="acceptTooltipCpt" placement="top" popper-class="upload-tooltip-popper">
  26. <template v-if="layout === 'card'">
  27. <div class="upload-layout-card __hover" :class="{'limit-disabled': isLimitCpt}">
  28. <SvgIcon name="add" color="#0062E9" size="24"/>
  29. <div>选择{{acceptTypeFormatCpt}}</div>
  30. <div v-if="limit > 0">(最多{{ limit }}个)</div>
  31. </div>
  32. </template>
  33. <template v-else>
  34. <div class="upload-layout-list_button" :class="{'limit-disabled': isLimitCpt}">
  35. <SvgIcon name="add" color="#ffffff" size="14"/>
  36. 选择{{acceptTypeFormatCpt}}
  37. <template v-if="limit > 0">(最多{{ limit }}个)</template>
  38. </div>
  39. </template>
  40. </el-tooltip>
  41. <template #file="{file}">
  42. <template v-if="layout === 'card'">
  43. <el-tooltip placement="top" :content="file[nameKey] ?? file[urlKey]">
  44. <div class="upload-layout-card_item" :class="{'item-view': isViewCpt || !delRule(file)}" @mouseenter="file.hover = true" @mouseleave="file.hover = false">
  45. <template v-if="validImgByUrl(file[urlKey])">
  46. <img class="img __hover" :src="$util.proxyNginxUrl(file[urlKey])" @click="viewImg(file[urlKey])"/>
  47. </template>
  48. <template v-else-if="validVideoByUrl(file[urlKey])">
  49. <video class="video __hover" controls :src="$util.proxyNginxUrl(file[urlKey])"/>
  50. </template>
  51. <template v-else>
  52. <img class="file __hover" :src="getFileImgByUrl(file[urlKey])" @click="downloadFileByUrl(file[urlKey], file[nameKey])"/>
  53. </template>
  54. <img class="del __hover" src="@/assets/images/cus/file-del.png" v-if="file.hover && !isViewCpt && delRule(file)" @click.stop="ref_upload.handleRemove(file)"/>
  55. </div>
  56. </el-tooltip>
  57. </template>
  58. <template v-else>
  59. <div class="upload-layout-list_item __hover" :class="{'item-view': isViewCpt || !delRule(file)}" @click="validImgByUrl(file[urlKey]) ? viewImg(file[urlKey]) : downloadFileByUrl(file[urlKey], file[nameKey])">
  60. <img class="file-type-img" :src="getFileImgByUrl(file[urlKey])"/>
  61. <CusEllipsis v-if="file[nameKey] ?? file[urlKey]" class="label" :value="file[nameKey] ?? file[urlKey]"/>
  62. <SvgIcon v-if="!isViewCpt && delRule(file)" class="close" name="close_2" size="12" @click.stop="ref_upload.handleRemove(file)"/>
  63. </div>
  64. </template>
  65. </template>
  66. </el-upload>
  67. <el-image-viewer
  68. v-if="currentImg.show"
  69. :zoom-rate="2"
  70. :max-scale="10"
  71. :min-scale="0.2"
  72. :hide-on-click-modal="true"
  73. :url-list="[currentImg.url]"
  74. :initial-index="0"
  75. @close="imageClose"
  76. />
  77. </div>
  78. </template>
  79. <script lang="ts">
  80. import {
  81. defineComponent,
  82. computed,
  83. onMounted,
  84. ref,
  85. reactive,
  86. watch,
  87. getCurrentInstance,
  88. ComponentInternalInstance,
  89. toRefs,
  90. nextTick, onBeforeMount, inject
  91. } from 'vue'
  92. import {useStore} from 'vuex'
  93. import {useRouter, useRoute} from 'vue-router'
  94. import {ElMessage} from "element-plus";
  95. import FileDefaultImg from '@/assets/images/file-type/file-type_default.png'
  96. import FilePDFImg from '@/assets/images/file-type/file-type_pdf.png'
  97. import FilePPTImg from '@/assets/images/file-type/file-type_ppt.png'
  98. import FileRARImg from '@/assets/images/file-type/file-type_rar.png'
  99. import FileTXTImg from '@/assets/images/file-type/file-type_txt.png'
  100. import FileWORDImg from '@/assets/images/file-type/file-type_word.png'
  101. import FileEXCELImg from '@/assets/images/file-type/file-type_excel.png'
  102. export default defineComponent({
  103. name: '',
  104. components: {},
  105. props: {
  106. param: {},
  107. label: {},
  108. limit: {
  109. default: 0
  110. },
  111. type: {default: 'all', validator(val: string) {
  112. return ['all', 'img', 'file'].includes(val)
  113. }},
  114. layout: {default: 'list', validator(val: string) {
  115. return ['list', 'card'].includes(val)
  116. }},
  117. acceptType: {default: '', type: String},
  118. acceptMax: {default: 0, type: Number},
  119. acceptFunc: {default: () => (options) => {}},
  120. view: {default: null},
  121. delRule: {default: () => () => false},
  122. cardWidth: {default: '147px'},
  123. cardHeight: {default: '128px'},
  124. urlKey: {default: 'url'},
  125. nameKey: {default: 'name'},
  126. limitNoUpload: {default: false}
  127. },
  128. setup(props, { emit }) {
  129. const store = useStore();
  130. const router = useRouter();
  131. const route = useRoute();
  132. const that = (getCurrentInstance() as ComponentInternalInstance).appContext.config.globalProperties
  133. const state = reactive({
  134. paramVal: <any>props.param,
  135. loading: false,
  136. elementLoadingBackground: inject('element-loading-background', null),
  137. currentImg: {
  138. show: false,
  139. url: ''
  140. },
  141. formView: inject('form-view', false),
  142. })
  143. watch(() => state.paramVal, (n) => {
  144. emit('emitParam', n)
  145. })
  146. watch(() => props.param, (n) => {
  147. state.paramVal = n
  148. })
  149. const isViewCpt = computed(() => {
  150. return that.$util.isValue(props.view) ? props.view : state.formView
  151. })
  152. const ref_upload = ref()
  153. const validImgByUrl = (url) => {
  154. if (url) {
  155. const imgList = ['bmp', 'jpg','jpeg', 'png', 'tif', 'gif', 'pcx', 'tga', 'exif', 'fpx', 'svg', 'psd', 'cdr', 'pcd', 'dxf', 'ufo', 'eps', 'ai', 'raw', 'WMF', 'webp', 'avif', 'apng', 'jfif'];
  156. //进行图片匹配
  157. let result = [...imgList.map(v => v.toLowerCase()), ...imgList.map(v => v.toUpperCase())].some((t) => url.includes('.' + t)) || url.includes('/ngx/proxy') || url.includes('/pic?');
  158. return result
  159. }
  160. return false
  161. }
  162. const validVideoByUrl = (url) => {
  163. if (url) {
  164. const videoList = ['avi', 'mp4', 'mov', 'wmv', 'flv', 'mkv', 'mpeg', '3gp'];
  165. //进行图片匹配
  166. let result = [...videoList.map(v => v.toLowerCase()), ...videoList.map(v => v.toUpperCase())].some((t) => url.includes('.' + t));
  167. return result
  168. }
  169. return false
  170. }
  171. const acceptTypeCpt = computed(() => {
  172. let str = ''
  173. if (that.$util.isValue(props.acceptType)) {
  174. str += props.acceptType
  175. } else {
  176. if (props.type === 'all') {
  177. str += store.state.app.globalConfig.imgType
  178. str += ',' + store.state.app.globalConfig.fileType
  179. } else if (props.type === 'img') {
  180. str += store.state.app.globalConfig.imgType
  181. } else if (props.type === 'file') {
  182. str += store.state.app.globalConfig.fileType
  183. }
  184. }
  185. return str
  186. })
  187. const acceptTypeFormatCpt = computed(() => {
  188. let str = ''
  189. if (that.$util.isValue(props.acceptType)) {
  190. str += `自定义文件`
  191. } else {
  192. if (props.type === 'all') {
  193. str += `图片、文件`
  194. } else if (props.type === 'img') {
  195. str += `图片`
  196. } else if (props.type === 'file') {
  197. str += `文件`
  198. }
  199. }
  200. return str
  201. })
  202. const acceptTooltipCpt = computed(() => {
  203. let str = ''
  204. if (that.$util.isValue(props.acceptType)) {
  205. str += `只支持大小在0~${props.acceptMax}M内,格式为${props.acceptType}的自定义文件`
  206. } else {
  207. if (props.type === 'all') {
  208. str += `只支持大小在0~${store.state.app.globalConfig.imgMax}M内,格式为${store.state.app.globalConfig.imgType}的图片,`
  209. str += `或大小在0~${store.state.app.globalConfig.fileMax}M内,格式为${store.state.app.globalConfig.fileType}的文件`
  210. } else if (props.type === 'img') {
  211. str += `只支持大小在0~${store.state.app.globalConfig.imgMax}M内,格式为${store.state.app.globalConfig.imgType}的图片`
  212. } else if (props.type === 'file') {
  213. str += `只支持大小在0~${store.state.app.globalConfig.fileMax}M内,格式为${store.state.app.globalConfig.fileType}的文件`
  214. }
  215. }
  216. if (isLimitCpt.value) {
  217. str = `最多上传${props.limit}个${acceptTypeFormatCpt.value}`
  218. }
  219. return str
  220. })
  221. const isLimitCpt = computed(() => {
  222. return props.limit > 0 && props.limit === state.paramVal.length
  223. })
  224. const handleBeforeUpload = (file) => {
  225. if (that.$util.isValue(props.acceptType)) {
  226. if (file.size / (1024 * 1024) > props.acceptMax) {
  227. ElMessage.warning(`自定义文件大小不可超过${props.acceptMax}M`)
  228. return false
  229. } else if (!props.acceptType.split(',').some(v => file.name.includes(v))) {
  230. ElMessage.warning(`自定义文件类型仅支持${props.acceptType}`)
  231. return false
  232. }
  233. } else {
  234. if (props.type === 'img') {
  235. if (file.size / (1024 * 1024) > store.state.app.globalConfig.imgMax) {
  236. ElMessage.warning(`图片大小不可超过${store.state.app.globalConfig.imgMax}M`)
  237. return false
  238. } else if (!store.state.app.globalConfig.imgType.split(',').some(v => file.name.includes(v))) {
  239. ElMessage.warning(`图片类型仅支持${store.state.app.globalConfig.imgType}`)
  240. return false
  241. }
  242. } else if (props.type === 'file') {
  243. if (file.size / (1024 * 1024) > store.state.app.globalConfig.fileMax) {
  244. ElMessage.warning(`文件大小不可超过${store.state.app.globalConfig.fileMax}M`)
  245. return false
  246. } else if (!store.state.app.globalConfig.fileType.split(',').some(v => file.name.includes(v))) {
  247. ElMessage.warning(`文件类型仅支持${store.state.app.globalConfig.fileType}`)
  248. return false
  249. }
  250. } else if (props.type === 'all') {
  251. if (!acceptTypeCpt.value.split(',').some(v => file.name.includes(v))) {
  252. ElMessage.warning(`文件或图片类型仅支持${acceptTypeCpt.value}`)
  253. return false
  254. } else {
  255. if (store.state.app.globalConfig.imgType.split(',').some(v => file.name.includes(v))) {
  256. if (file.size / (1024 * 1024) > store.state.app.globalConfig.imgMax) {
  257. ElMessage.warning(`图片大小不可超过${store.state.app.globalConfig.imgMax}M`)
  258. return false
  259. }
  260. } else if (store.state.app.globalConfig.fileType.split(',').some(v => file.name.includes(v))) {
  261. if (file.size / (1024 * 1024) > store.state.app.globalConfig.fileMax) {
  262. ElMessage.warning(`文件大小不可超过${store.state.app.globalConfig.fileMax}M`)
  263. return false
  264. }
  265. }
  266. }
  267. }
  268. }
  269. return file
  270. }
  271. const handleRequest = async (options) => {
  272. state.loading = true
  273. if (that.$util.isValue(props.acceptType)) {
  274. const result: any = await props.acceptFunc(options)
  275. if (result?.[props.urlKey] && result?.[props.nameKey]) {
  276. state.paramVal = [...state.paramVal, result]
  277. } else {
  278. state.paramVal = [...state.paramVal]
  279. }
  280. state.loading = false
  281. } else {
  282. if (store.state.app.globalConfig.imgType.split(',').some(v => options.file.name.includes(v))) {
  283. const formData = new FormData();
  284. formData.append("file", options.file);
  285. that.$api.uploadPicOnly(formData).then(res => {
  286. if (res?.code === 200) {
  287. state.paramVal = [...state.paramVal, {
  288. [props.urlKey]: res.imgPath,
  289. [props.nameKey]: res.fileName
  290. }]
  291. }
  292. state.loading = false
  293. }).catch(() => {
  294. state.loading = false
  295. })
  296. } else if (store.state.app.globalConfig.fileType.split(',').some(v => options.file.name.includes(v))) {
  297. const formData = new FormData();
  298. formData.append("file", options.file);
  299. that.$api.uploadFile(formData).then(res => {
  300. if (res?.code === 200) {
  301. state.paramVal = [...state.paramVal, {
  302. [props.urlKey]: res.url,
  303. [props.nameKey]: res.fileName
  304. }]
  305. }
  306. state.loading = false
  307. }).catch(() => {
  308. state.loading = false
  309. })
  310. }
  311. }
  312. }
  313. const fileTypeMapper = new Map([
  314. ['doc', FileWORDImg],
  315. ['docx', FileWORDImg],
  316. ['ppt', FilePPTImg],
  317. ['pptx', FilePPTImg],
  318. ['pdf', FilePDFImg],
  319. ['xls', FileEXCELImg],
  320. ['xlsx', FileEXCELImg],
  321. ['txt', FileTXTImg],
  322. ['rar', FileRARImg],
  323. ['zip', FileRARImg],
  324. ])
  325. const getFileImgByUrl = (url) => {
  326. if (url) {
  327. const regex = /\.([^.]*)$/;
  328. const match = url.match(regex);
  329. const t = match ? match[1] : '';
  330. if (t && fileTypeMapper.has(t)) {
  331. return fileTypeMapper.get(t)
  332. }
  333. }
  334. return FileDefaultImg
  335. }
  336. const handleRemove = (file, files) => {
  337. state.paramVal = files
  338. }
  339. const downloadFileByUrl = (url, name) => {
  340. ElMessage.info('开始下载!')
  341. const xhr = new XMLHttpRequest();
  342. xhr.open('GET', url, true);
  343. xhr.responseType = 'blob';
  344. xhr.onload = function () {
  345. if (xhr.status === 200) {
  346. const blob = xhr.response;
  347. const url = window.URL.createObjectURL(blob);
  348. const a = document.createElement('a');
  349. a.href = url;
  350. a.download = name || 'default';
  351. a.click();
  352. ElMessage.success('下载成功!')
  353. window.URL.revokeObjectURL(url);
  354. }
  355. };
  356. xhr.send();
  357. }
  358. const viewImg = (url) => {
  359. state.currentImg.url = url
  360. state.currentImg.show = true
  361. }
  362. const imageClose = () => {
  363. state.currentImg.show = false
  364. state.currentImg.url = ''
  365. }
  366. onMounted(() => {
  367. state.loading = true
  368. store.dispatch('app/LOAD_GLOBAL_CONFIG').then(() => {
  369. state.loading = false
  370. })
  371. })
  372. return {
  373. ...toRefs(state),
  374. acceptTypeCpt,
  375. handleBeforeUpload,
  376. handleRequest,
  377. handleRemove,
  378. validImgByUrl,
  379. validVideoByUrl,
  380. acceptTooltipCpt,
  381. acceptTypeFormatCpt,
  382. getFileImgByUrl,
  383. ref_upload,
  384. downloadFileByUrl,
  385. isLimitCpt,
  386. viewImg,
  387. imageClose,
  388. isViewCpt
  389. }
  390. },
  391. })
  392. </script>
  393. <style scoped lang="scss">
  394. .cus-form-column-upload-com {
  395. $uploadWidth: v-bind(cardWidth);
  396. $uploadHeight: v-bind(cardHeight);
  397. width: 100%;
  398. position: relative;
  399. .el-upload-com {
  400. :deep(.el-upload) {
  401. .upload-layout-list_button {
  402. height: 32px;
  403. display: flex;
  404. align-items: center;
  405. justify-content: center;
  406. padding: 0 18px;
  407. background-color: #0062E9;
  408. border-radius: 4px;
  409. font-size: 14px;
  410. font-family: Microsoft YaHei;
  411. font-weight: 400;
  412. color: #FFFFFF;
  413. .svg-icon {
  414. margin-right: 7px;
  415. }
  416. &.limit-disabled {
  417. cursor: no-drop;
  418. opacity: 0.5;
  419. }
  420. }
  421. .upload-layout-card {
  422. width: 100%;
  423. height: 100%;
  424. display: flex;
  425. flex-direction: column;
  426. line-height: 1;
  427. align-items: center;
  428. justify-content: center;
  429. background-color: rgba(46, 129, 255, 0.04);
  430. .svg-icon {
  431. margin-bottom: 10px;
  432. opacity: 0.5;
  433. }
  434. > div {
  435. font-size: 14px;
  436. font-family: PingFang SC-Regular, PingFang SC;
  437. font-weight: 400;
  438. color: rgba(0,98,233,0.5);
  439. }
  440. &.limit-disabled {
  441. cursor: no-drop;
  442. opacity: 0.5;
  443. }
  444. }
  445. }
  446. :deep(.el-upload-list) {
  447. margin: 0;
  448. .el-upload-list__item {
  449. transition: none;
  450. .upload-layout-list_item {
  451. display: flex;
  452. align-items: center;
  453. padding: 0 6px;
  454. .file-type-img {
  455. width: 14px;
  456. margin: 0 6px 0 6px;
  457. }
  458. .label {
  459. width: calc(100% - 6px - 6px - 14px - 30px);
  460. }
  461. .close {
  462. margin-left: auto;
  463. }
  464. &.item-view {
  465. .label {
  466. width: calc(100% - 6px - 6px - 14px);
  467. }
  468. }
  469. }
  470. .upload-layout-card_item {
  471. width: 100%;
  472. height: 100%;
  473. position: relative;
  474. .img {
  475. width: 100%;
  476. height: 100%;
  477. }
  478. .video {
  479. width: 100%;
  480. height: 100%;
  481. object-fit: fill;
  482. }
  483. .file {
  484. width: 100%;
  485. height: 100%;
  486. }
  487. .del {
  488. position: absolute;
  489. top: 0;
  490. right: 0;
  491. z-index: 2;
  492. width: 20px;
  493. height: 20px;
  494. }
  495. }
  496. }
  497. }
  498. }
  499. &.cus-form-column-upload-com_view, &.cus-form-column-upload-com_limit-no-upload {
  500. .el-upload-com {
  501. :deep(.el-upload) {
  502. display: none;
  503. }
  504. }
  505. }
  506. &.cus-form-column-upload-com_card {
  507. .el-upload-com {
  508. :deep(.el-upload) {
  509. width: $uploadWidth;
  510. height: $uploadHeight;
  511. }
  512. :deep(.el-upload-list) {
  513. .el-upload-list__item {
  514. width: $uploadWidth;
  515. height: $uploadHeight;
  516. }
  517. }
  518. }
  519. }
  520. }
  521. </style>