index.vue 11 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321
  1. <template>
  2. <view class="van-uploader">
  3. <view class="van-uploader__wrapper">
  4. <!-- 预览样式 -->
  5. <view v-if="previewImage" class="van-uploader__preview" :data-index="index" @tap="onClickPreview" v-for="(item, index) in lists" :key="index">
  6. <image
  7. v-if="item.isImage"
  8. :mode="imageFit"
  9. :src="item.url || item.path"
  10. :alt="item.name || '图片' + index"
  11. class="van-uploader__preview-image"
  12. :style="'width: ' + utils.addUnit(previewSize) + '; height: ' + utils.addUnit(previewSize) + ';'"
  13. :data-index="index"
  14. @tap="onPreviewImage"
  15. />
  16. <view v-else class="van-uploader__file" :style="'width: ' + utils.addUnit(previewSize) + '; height: ' + utils.addUnit(previewSize) + ';'">
  17. <van-icon name="description" class="van-uploader__file-icon" />
  18. <view class="van-uploader__file-name van-ellipsis">{{ item.name || item.url || item.path }}</view>
  19. </view>
  20. <view v-if="item.status === 'uploading' || item.status === 'failed'" class="van-uploader__mask">
  21. <van-icon v-if="item.status === 'failed'" name="warning-o" class="van-uploader__mask-icon" />
  22. <van-loading v-else class="van-uploader__loading" />
  23. <text v-if="item.message" class="van-uploader__upload-text">{{ item.message }}</text>
  24. </view>
  25. <view v-if="deletable && item.deletable" :data-index="index" class="van-uploader__preview-delete" @tap.stop.prevent="deleteItem">
  26. <van-icon name="clear" class="van-uploader__preview-delete__icon" />
  27. </view>
  28. </view>
  29. <!-- 上传样式 -->
  30. <block v-if="isInCount">
  31. <view class="van-uploader__slot" @tap="startUpload">
  32. <slot />
  33. </view>
  34. <!-- 默认上传样式 -->
  35. <view
  36. v-if="showUpload"
  37. :class="'van-uploader__upload ' + (disabled ? 'van-uploader__upload--disabled' : '')"
  38. :style="'width: ' + utils.addUnit(previewSize) + '; height: ' + utils.addUnit(previewSize) + ';'"
  39. @tap="startUpload"
  40. >
  41. <van-icon :name="uploadIcon" class="van-uploader__upload-icon" />
  42. <text v-if="uploadText" class="van-uploader__upload-text">{{ uploadText }}</text>
  43. </view>
  44. </block>
  45. </view>
  46. </view>
  47. </template>
  48. <script module="utils" lang="wxs" src="@/miniprogram_npm/@vant/weapp/wxs/utils.wxs"></script>
  49. <script>
  50. 'use strict';
  51. var __assign =
  52. (this && this.__assign) ||
  53. function () {
  54. __assign =
  55. Object.assign ||
  56. function (t) {
  57. for (var s, i = 1, n = arguments.length; i < n; i++) {
  58. s = arguments[i];
  59. for (var p in s) {
  60. if (Object.prototype.hasOwnProperty.call(s, p)) {
  61. t[p] = s[p];
  62. }
  63. }
  64. }
  65. return t;
  66. };
  67. return __assign.apply(this, arguments);
  68. };
  69. Object.defineProperty(exports, '__esModule', {
  70. value: true
  71. });
  72. var component_1 = require('../common/component');
  73. var utils_1 = require('./utils');
  74. var shared_1 = require('./shared');
  75. component_1.VantComponent({
  76. props: __assign(
  77. __assign(
  78. {
  79. disabled: Boolean,
  80. multiple: Boolean,
  81. uploadText: String,
  82. useBeforeRead: Boolean,
  83. afterRead: null,
  84. beforeRead: null,
  85. previewSize: {
  86. type: null,
  87. value: 90
  88. },
  89. name: {
  90. type: [Number, String],
  91. value: ''
  92. },
  93. accept: {
  94. type: String,
  95. value: 'image'
  96. },
  97. fileList: {
  98. type: Array,
  99. value: [],
  100. observer: 'formatFileList'
  101. },
  102. maxSize: {
  103. type: Number,
  104. value: Number.MAX_VALUE
  105. },
  106. maxCount: {
  107. type: Number,
  108. value: 100
  109. },
  110. deletable: {
  111. type: Boolean,
  112. value: true
  113. },
  114. showUpload: {
  115. type: Boolean,
  116. value: true
  117. },
  118. previewImage: {
  119. type: Boolean,
  120. value: true
  121. },
  122. previewFullImage: {
  123. type: Boolean,
  124. value: true
  125. },
  126. imageFit: {
  127. type: String,
  128. value: 'scaleToFill'
  129. },
  130. uploadIcon: {
  131. type: String,
  132. value: 'photograph'
  133. }
  134. },
  135. shared_1.chooseImageProps
  136. ),
  137. shared_1.chooseVideoProps
  138. ),
  139. data: {
  140. lists: [],
  141. isInCount: true
  142. },
  143. methods: {
  144. formatFileList: function () {
  145. var _a = this;
  146. var _b = _a.fileList;
  147. var fileList = _b === void 0 ? [] : _b;
  148. var maxCount = _a.maxCount;
  149. var lists = fileList.map(function (item) {
  150. return __assign(__assign({}, item), {
  151. isImage: typeof item.isImage === 'undefined' ? utils_1.isImageFile(item) : item.isImage,
  152. deletable: typeof item.deletable === 'undefined' ? true : item.deletable
  153. });
  154. });
  155. this.setData({
  156. lists: lists,
  157. isInCount: lists.length < maxCount
  158. });
  159. },
  160. getDetail: function (index) {
  161. return {
  162. name: this.name,
  163. index: index == null ? this.fileList.length : index
  164. };
  165. },
  166. startUpload: function () {
  167. var that = this;
  168. var _a = this;
  169. var maxCount = _a.maxCount;
  170. var multiple = _a.multiple;
  171. var accept = _a.accept;
  172. var lists = _a.lists;
  173. var disabled = _a.disabled;
  174. if (disabled) {
  175. return;
  176. }
  177. utils_1
  178. .chooseFile(
  179. __assign(__assign({}, this), {
  180. maxCount: maxCount - lists.length
  181. })
  182. )
  183. .then(function (res) {
  184. var file = null;
  185. if (utils_1.isVideo(res, accept)) {
  186. file = __assign(
  187. {
  188. path: res.tempFilePath
  189. },
  190. res
  191. );
  192. } else {
  193. file = multiple ? res.tempFiles : res.tempFiles[0];
  194. }
  195. that.onBeforeRead(file);
  196. })
  197. .catch(function (error) {
  198. that.$emit('error', error);
  199. });
  200. },
  201. onBeforeRead: function (file) {
  202. var that = this;
  203. var _a = this;
  204. var beforeRead = _a.beforeRead;
  205. var useBeforeRead = _a.useBeforeRead;
  206. var res = true;
  207. if (typeof beforeRead === 'function') {
  208. res = beforeRead(file, this.getDetail());
  209. }
  210. if (useBeforeRead) {
  211. res = new Promise(function (resolve, reject) {
  212. that.$emit(
  213. 'before-read',
  214. __assign(
  215. __assign(
  216. {
  217. file: file
  218. },
  219. that.getDetail()
  220. ),
  221. {
  222. callback: function (ok) {
  223. ok ? resolve() : reject();
  224. }
  225. }
  226. )
  227. );
  228. });
  229. }
  230. if (!res) {
  231. return;
  232. }
  233. if (utils_1.isPromise(res)) {
  234. res.then(function (data) {
  235. return that.onAfterRead(data || file);
  236. });
  237. } else {
  238. this.onAfterRead(file);
  239. }
  240. },
  241. onAfterRead: function (file) {
  242. var maxSize = this.maxSize;
  243. var oversize = Array.isArray(file)
  244. ? file.some(function (item) {
  245. return item.size > maxSize;
  246. })
  247. : file.size > maxSize;
  248. if (oversize) {
  249. this.$emit(
  250. 'oversize',
  251. __assign(
  252. {
  253. file: file
  254. },
  255. this.getDetail()
  256. )
  257. );
  258. return;
  259. }
  260. if (typeof this.afterRead === 'function') {
  261. this.afterRead(file, this.getDetail());
  262. }
  263. this.$emit(
  264. 'after-read',
  265. __assign(
  266. {
  267. file: file
  268. },
  269. this.getDetail()
  270. )
  271. );
  272. },
  273. deleteItem: function (event) {
  274. var index = event.currentTarget.dataset.index;
  275. this.$emit(
  276. 'delete',
  277. __assign(__assign({}, this.getDetail(index)), {
  278. file: this.fileList[index]
  279. })
  280. );
  281. },
  282. onPreviewImage: function (event) {
  283. if (!this.previewFullImage) {
  284. return;
  285. }
  286. var index = event.currentTarget.dataset.index;
  287. var lists = this.lists;
  288. var item = lists[index];
  289. uni.previewImage({
  290. urls: lists
  291. .filter(function (item) {
  292. return item.isImage;
  293. })
  294. .map(function (item) {
  295. return item.url || item.path;
  296. }),
  297. current: item.url || item.path,
  298. fail: function () {
  299. uni.showToast({
  300. title: '预览图片失败',
  301. icon: 'none'
  302. });
  303. }
  304. });
  305. },
  306. onClickPreview: function (event) {
  307. var index = event.currentTarget.dataset.index;
  308. var item = this.lists[index];
  309. this.$emit('click-preview', __assign(__assign({}, item), this.getDetail(index)));
  310. }
  311. }
  312. });
  313. </script>
  314. <style>
  315. @import './index.css';
  316. </style>