CFiles.vue 7.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257
  1. <template>
  2. <!-- NOTE: 附件和图片组件 -->
  3. <div class="custom-files-container">
  4. <div :class="[
  5. 'custom-files__header',
  6. 'flex', 'flex-row', 'flex-row-aic', 'flex-row-jcsp',
  7. filelist.length ? 'pb6' : '',
  8. ]">
  9. <span class="title">{{ headerTitle }}</span>
  10. <van-icon color="#a2a3a4" :size="20" name="add-o" @click="handleLaunchUploadBox" />
  11. </div>
  12. <!-- files show style -->
  13. <div class="source-listbox" v-if="isFs">
  14. <ul class="files-container">
  15. <li class="files-row flex flex-row flex-row-aic" v-for="(file, index) in filelist" :key="index">
  16. <div class="icon">
  17. <template v-if="['png', 'jpg', 'jpeg'].includes(file.type)">
  18. <span>pic</span>
  19. </template>
  20. <span v-else-if="file.type === 'word'">word</span>
  21. <span v-else-if="file.type === 'pdf'">pdf</span>
  22. <span v-else-if="['xsl'].includes(file.type)">表格</span>
  23. </div>
  24. <div class="files-row__info">
  25. <div class="files-name ellipsis">{{ file.name }}</div>
  26. <div class="files-other-info flex flex-row flex-row-aic">
  27. <div class="files-other-info__div flex flex-row">
  28. <div class="size">{{ file.size }}</div>
  29. <div class="review" @click="handleReviewFiles(file)">预览</div>
  30. </div>
  31. <van-icon :size="20" color="#A2A3A4" name="clear" @click="handleRemoveFile(file, index)" />
  32. </div>
  33. </div>
  34. </li>
  35. </ul>
  36. </div>
  37. <van-uploader v-show="filelist.length && !isFs" ref="uploadFileRef" v-model="filelist" preview-size="48px"
  38. :max-count="$attrs.maxCount" :max-size="maxSizeComp" :show-upload="false" :accept="acceptHandle"
  39. :preview-image="!isFs" :before-read="beforeRead" :after-read="afterRead" @oversize="onOversize">
  40. </van-uploader>
  41. </div>
  42. </template>
  43. <style lang="less" scoped>
  44. @import url("@/styles/variables.less");
  45. .custom-files {
  46. &-container {
  47. padding: 9px 12px;
  48. background: #fff;
  49. }
  50. &__header {
  51. &.pb6 {
  52. padding-bottom: 6px;
  53. }
  54. .title {
  55. font-size: @font-size-secondery;
  56. font-weight: 400;
  57. color: #191A1E;
  58. line-height: 20px;
  59. }
  60. }
  61. }
  62. .source-listbox {
  63. // 附件样式
  64. .files {
  65. &-row {
  66. margin-bottom: 5px;
  67. &:last-child {
  68. margin-bottom: initial;
  69. }
  70. .icon {
  71. width: 35px;
  72. height: 42px;
  73. background: #FFCF95;
  74. margin-right: 12px;
  75. }
  76. &__info {
  77. flex: 1;
  78. width: 0;
  79. }
  80. }
  81. &-name {
  82. width: 88%;
  83. font-size: @font-size-common;
  84. font-weight: 400;
  85. color: #191A1E;
  86. line-height: 18px;
  87. }
  88. &-other-info {
  89. justify-content: space-between;
  90. margin-top: 4px;
  91. .size {
  92. font-size: @font-size-third;
  93. font-weight: 400;
  94. color: #9A9A9A;
  95. line-height: 18px;
  96. margin-right: 10px;
  97. }
  98. .review {
  99. font-size: @font-size-third;
  100. font-weight: 400;
  101. color: #3290C4;
  102. line-height: 18px;
  103. }
  104. }
  105. }
  106. }
  107. </style>
  108. <script>
  109. import { ImagePreview } from 'vant';
  110. import uploadFile from '@/utils/upload'
  111. import {
  112. getStaticLinkInfo,
  113. getByteShowSize
  114. } from '@/utils/util'
  115. import {
  116. FileSize,
  117. PicSize
  118. } from "@/utils/constant"
  119. export default {
  120. name: 'CFiles',
  121. props: {
  122. value: {
  123. type: Array
  124. },
  125. ctype: {
  126. validator: val => ['files', 'images'].includes(val),
  127. default: 'files'
  128. }
  129. },
  130. computed: {
  131. isFs() { // 是否是文件类型上传
  132. return this.ctype === 'files'
  133. },
  134. acceptHandle() { // 图片上传文件限制
  135. return this.isFs ? "*" : "image/*"
  136. },
  137. headerTitle() {
  138. return this.isFs ? '附件' : '图片'
  139. },
  140. // 文件大小
  141. maxSizeComp() {
  142. let mb = 1 * 1024 * 1024;
  143. return this.isFs ? FileSize * mb : PicSize * mb
  144. }
  145. },
  146. data() {
  147. return {
  148. toastInstance: null,
  149. filelist: []
  150. }
  151. },
  152. mounted() {
  153. this.handleUpdateValues()
  154. },
  155. methods: {
  156. handleUpdateValues() {
  157. const arrs = this.value
  158. if (Array.isArray(arrs) && arrs.length) this.filelist = [...arrs]
  159. },
  160. // 启动上传组件
  161. handleLaunchUploadBox() {
  162. this.$refs.uploadFileRef.chooseFile()
  163. },
  164. // @returns Boolean {true/false}
  165. beforeRead() {
  166. this.toastInstance = this.$toast.loading({
  167. duration: 0,
  168. message: '上传中'
  169. })
  170. // NOTE: 上传前的控制
  171. return true
  172. },
  173. // 自行上传文件
  174. async afterRead(file, detail) {
  175. console.log('after read', file, detail)
  176. try {
  177. const { fullurl: url } = await uploadFile(file.file)
  178. const { suffix } = getStaticLinkInfo(url)
  179. file.name = file.file.name
  180. file.type = suffix
  181. file.url = url
  182. file.size = getByteShowSize(file.file.size)
  183. file.status = 'success'
  184. this.$forceUpdate() // 强制更新渲染视图
  185. } catch (error) {
  186. file.status = 'failed'
  187. file.message = error.message
  188. } finally {
  189. this.toastInstance.clear()
  190. }
  191. },
  192. // 超出文件大小
  193. onOversize() {
  194. let desc = this.isFs ? `文件超过${FileSize}MB` : `图片超过${PicSize}MB`
  195. this.$toast(desc)
  196. },
  197. // 预览文件
  198. handleReviewFiles(file) {
  199. // TODO: 如果是图片直接预览。 非图片其他方式预览(下载、或者插件)
  200. // NOTE: 可进行图片预览类型
  201. if (['png', 'jpg', 'jpeg'].includes(file.type)) {
  202. ImagePreview([file.url])
  203. } else {
  204. this.$toast(file.type + '预览开发中...')
  205. }
  206. },
  207. handleRemoveFile(file, index) {
  208. this.filelist.splice(index, 1)
  209. }
  210. },
  211. watch: {
  212. // NOTE: 判断是否有新添加。 新添加赋值
  213. filelist: {
  214. handler(arrs) {
  215. let hasAdd = arrs.some(image => (image.content))
  216. if (hasAdd) {
  217. this.$listeners['input'] && this.$emit('input', arrs)
  218. }
  219. }
  220. },
  221. // value: {
  222. // handler(arrs) {
  223. // if (Array.isArray(arrs) && arrs.length) {
  224. // this.filelist = [...arrs]
  225. // }
  226. // }
  227. // }
  228. }
  229. }
  230. </script>