uniapp内开发的小程序如何实现 9宫格图片长按后进行拖拽排序

178 阅读4分钟

`

`


<template>
	<view class="con" :style="{ height: videos ? '300rpx' : '' }">
		<template v-if="viewWidth">
			<movable-area class="area" :style="{ height: areaHeight, }" @mouseenter="mouseenter"
				@mouseleave="mouseleave">
				<movable-view v-if="!videos" v-for="(item, index) in imageList" :key="item.id" class="view"
					direction="all" :y="item.y" :x="item.x" :damping="40" :disabled="isLongpress"
					@change="onChange($event, item)" @touchstart="touchstart(item)" @mousedown="touchstart(item)"
					@touchend="touchend(item)" @mouseup="touchend(item)" :style="{
					width: viewWidth + 'px',
					height: viewWidth + 'px',
					'z-index': item.zIndex,
					opacity: item.opacity
				}">
								<view class="area-con" :style="{
					width: childWidth,
					height: childWidth,
					borderRadius: borderRadius + 'rpx',
					transform: 'scale(' + item.scale + ')'
				}">
						<image class="pre-image" :src="item.src" mode="aspectFill"></image>
						<view class="del-con" v-if="showEdit == false" @click="delImages(item, index)"
							@touchstart.stop="delImageMp(item, index)" @touchend.stop="nothing()"
							@mousedown.stop="nothing()" @mouseup.stop="nothing()">
							<view class="del-wrap">
								<image class="del-image"
									src="">
								</image>
							</view>
						</view>
					</view>
				</movable-view>
				<view class="add" v-if="imageList.length < number && !videos"
					:style="{ top: add.y, left: add.x, width: viewWidth + 'px', height: viewWidth + 'px' }"
					@click="addImages">
					<view class="add-wrap"
						:style="{ width: childWidth, height: childWidth, borderRadius: borderRadius + 'rpx' }">
						<image src="../../static/petimg/j.png" mode=""></image>
					</view>
				</view>
				<view class="" style="position: relative" v-if="videos">
					<view class="delimg" v-if="showEdit == false" @click="deleervoider">x</view>
					<video id="myVideo" :src="videos" type="video/mp4" class="upLoadvideo" page-gesture
						show-mute-btn></video>
					<view class="numBig">
						<text>{{ numBigs }}</text>
					</view>
				</view>
			</movable-area>

		</template>
	</view>
</template>

<script>
import {
	createPoster,
	upLoad,
	createPetPoster,
	getPetsers,
	getpolicyz,
	listTopics,
	updatePost
} from '@/api/ceder.js';
export default {
	emits: ['input', 'update:modelValue', 'changeSort'],
	props: {
		// 排序图片
		value: {
			type: Array,
			default: function () {
				return []
			}
		},
		// 排序图片
		modelValue: {
			type: Array,
			default: function () {
				return []
			}
		},
		// 从 list 元素对象中读取的键名
		keyName: {
			type: String,
			default: null
		},
		// 选择图片数量限制
		number: {
			type: Number,
			default: 9
		},
		// 图片父容器宽度(实际显示的图片宽度为 imageWidth / 1.1 ),单位 rpx
		// imageWidth > 0 则 cols 无效
		imageWidth: {
			type: Number,
			default: 0
		},
		// 图片列数
		cols: {
			type: Number,
			default: 3
		},
		// 图片圆角,单位 rpx
		borderRadius: {
			type: Number,
			default: 10
		},
		// 图片周围空白填充,单位 rpx
		padding: {
			type: Number,
			default: 10
		},
		// 拖动图片时放大倍数 [0, ∞)
		scale: {
			type: Number,
			default: 1.2
		},
		// 拖动图片时不透明度
		opacity: {
			type: Number,
			default: 0.7
		},
		// 自定义添加
		addImage: {
			type: Function,
			default: null
		},
		// 删除确认
		delImage: {
			type: Function,
			default: null
		},
		// 是否是编辑
		showEdit: {
			type: Boolean,
			default: false
		},
		// 编辑视频
		editVideos: {
			type: String,
			default: ''
		},
		// 是否只能展示视频
		editchooseIndex: {
			type: Number || String,
			default: 0
		}
	},
	data() {
		return {
			imageList: [],
			width: 0,
			add: {
				x: 0,
				y: 0
			},
			checkNumber: 9,
			colsValue: 0,
			viewWidth: 0,
			tempItem: null,
			timer: null,
			changeStatus: true,
			preStatus: true,
			first: true,
			// start
			limitsathon: false,
			limitsathoncenmer: false,
			chooseIndex: 0,
			videos: '', //视频地址
			fileList: [],
			fileListOld: [], // 展示的图片地址
			fileListOnelist: [],
			deleimage: false,
			delevideo: false,
			uploadImgUrl: '', // 上传图片地址
			videosUnpload: "",
			numBigs: '',
			isLongpress:true,
			// end
		}
	},
	computed: {
		areaHeight() {
			let height = ''
			// return '355px'
			if (this.opacity == 0.7) {
				console.log('9')
				if (this.imageList.length < this.number) {
					height = (Math.ceil((this.imageList.length + 1) / this.colsValue) * this.viewWidth).toFixed() + 'px'
				} else {
					height = (Math.ceil(this.imageList.length / this.colsValue) * this.viewWidth).toFixed() + 'px'
				}
			}
			return height
		},
		childWidth() {
			return this.viewWidth - this.rpx2px(this.padding) * 2 + 'px'
		},
	},
	watch: {
		numBigs(newLength) {
			if (newLength == '上传中...100%') {
				this.numBigs = this.$t('state.shangchuanwancheng');
			}
		},
		editVideos: {
			handler(n) {
				console.log('editVideos----watch', this.editVideos);
				console.log('editVideos----n---n', n);
				if (n) {
					this.videos = n;
					this.videosUnpload = n;
				}
			},
			deep: true
		},
		value: {
			handler(n) {
				if (!this.first && this.changeStatus) {
					console.log('watch', n)
					let flag = false
					for (let i = 0; i < n.length; i++) {
						if (flag) {
							this.addProperties(this.getSrc(n[i]))
							continue
						}
						if (this.imageList.length === i || this.imageList[i].src !== this.getSrc(n[i])) {
							flag = true
							this.imageList.splice(i)
							this.addProperties(this.getSrc(n[i]))
						}
					}
				}
			},
			deep: true
		},
		modelValue: {
			handler(n) {
				if (!this.first && this.changeStatus) {
					console.log('watch', n)
					let flag = false
					for (let i = 0; i < n.length; i++) {
						if (flag) {
							this.addProperties(this.getSrc(n[i]))
							continue
						}
						if (this.imageList.length === i || this.imageList[i].src !== this.getSrc(n[i])) {
							flag = true
							this.imageList.splice(i)
							this.addProperties(this.getSrc(n[i]))
						}
					}
				}
			},
			deep: true
		},
	},
	created() {
		console.log('---重新执行了--')
		// 获取设备宽度
		this.width = uni.getSystemInfoSync().windowWidth
	},
	mounted() {
		// 获取当前的存放移动区域的属性
		const query = uni.createSelectorQuery().in(this)
		query.select('.con').boundingClientRect(data => {
			// 设置的三列 进行传值
			this.colsValue = this.cols
			// 元素宽度除以三进行均分
			this.viewWidth = data.width / this.cols
			if (this.imageWidth > 0) {
				this.viewWidth = this.rpx2px(this.imageWidth)
				this.colsValue = Math.floor(data.width / this.viewWidth)
			}
			let list = this.value
			// #ifdef VUE3
			list = this.modelValue
			// #endif
			for (let item of list) {
				this.addProperties(this.getSrc(item))
			}
			this.first = false

		})
		query.exec()
	},
	onReady: function (res) {
		this.videoContext = uni.createVideoContext('myVideo');
	},
	methods: {
		playVideo() {
			this.videoContext.pause();
		},
		// 传的如果有键值就采取这个方法。没有就不需要
		getSrc(item) {
			if (this.keyName !== null) {
				return item[this.keyName]
			}
			return item
		},
		onChange(e, item) {
			if (this.opacity == 0.7) {
				if (!item) return
				item.oldX = e.detail.x
				item.oldY = e.detail.y
				// 如果是拖动状态中
				if (e.detail.source === 'touch') {
					if (item.moveEnd) {
						item.offset = Math.sqrt(Math.pow(item.oldX - item.absX * this.viewWidth, 2) + Math.pow(item.oldY -
							item
								.absY * this.viewWidth, 2))
					}
					// x为 移动时候的坐标点加上划定的一半的区域的和 除以划定区域 去判断有没有超过自己定义的最大列数
					let x = Math.floor((e.detail.x + this.viewWidth / 2) / this.viewWidth)
					if (x >= this.colsValue) return
					// y同理  也是判断超过高度没有
					let y = Math.floor((e.detail.y + this.viewWidth / 2) / this.viewWidth)
					// index则是 如果是第一张图片右移动了半张图片的位置 index就会加一
					let index = this.colsValue * y + x
					if (item.index != index && index < this.imageList.length) {
						this.changeStatus = false
						for (let obj of this.imageList) {
							// 判断图片是左右上下移动  因为这个函数是一直触发
							if (item.index > index && obj.index >= index && obj.index < item.index) {
								this.change(obj, 1)
							} else if (item.index < index && obj.index <= index && obj.index > item.index) {
								this.change(obj, -1)
							} else if (obj.id != item.id) {
								obj.offset = 0
								obj.x = obj.oldX
								obj.y = obj.oldY
								setTimeout(() => {
									this.$nextTick(() => {
										obj.x = obj.absX * this.viewWidth
										obj.y = obj.absY * this.viewWidth
									})
								}, 0)
							}
						}
						item.index = index
						item.absX = x
						item.absY = y
						if (!item.moveEnd) {
							setTimeout(() => {
								this.$nextTick(() => {
									item.x = item.absX * this.viewWidth
									item.y = item.absY * this.viewWidth
								})
							}, 0)
						}
						// console.log('bbb', JSON.parse(JSON.stringify(item)));
						// 移动完成后重新排序
						this.sortList()
					}
				}
			}
		},
		// change事件会随着移动函数一直触发,index随之变化修改图片的位置xy
		change(obj, i) {
			obj.index += i
			obj.offset = 0
			obj.x = obj.oldX
			obj.y = obj.oldY
			obj.absX = obj.index % this.colsValue
			obj.absY = Math.floor(obj.index / this.colsValue)
			setTimeout(() => {
				this.$nextTick(() => {
					obj.x = obj.absX * this.viewWidth
					obj.y = obj.absY * this.viewWidth
				})
			}, 0)
		},
		// 长按图片时候进行所有的层级进行加大 和放大
		touchstart(item) {
				this.imageList.forEach(v => {
					v.zIndex = v.index + 9
				})
				item.zIndex = 99
				item.moveEnd = true
				this.tempItem = item
				this.timer = setTimeout(() => {
					item.scale = this.scale
					item.opacity = this.opacity
					clearTimeout(this.timer)
					this.timer = null
					this.isLongpress = false;
				}, 800)
		},
		// 点击一次没有触发四个变量的更改就会触发previewImage 变成预览
		// 拖拽过程中几个变量会变,就不会触发预览 拖拽结束后就会缩小并且改变位置 
		touchend(item) {
			if(this.opacity == 0.7){
				console.log('5')
				this.previewImage(item)
				item.scale = 1
				item.opacity = 1
				item.x = item.oldX
				item.y = item.oldY
				item.offset = 0
				item.moveEnd = false
				setTimeout(() => {
					this.$nextTick(() => {
						item.x = item.absX * this.viewWidth
						item.y = item.absY * this.viewWidth
						this.tempItem = null
						this.changeStatus = true
						this.isLongpress = true;
					})
					// console.log('ccc', JSON.parse(JSON.stringify(item)));
				}, 0)
			}
			// console.log('ddd', JSON.parse(JSON.stringify(item)));
		},
		previewImage(item) {
			// timer是定时器 changeStatus是不是在移动状态中 只要点击移动了就是false  offset也是为0
			if (this.timer && this.preStatus && this.changeStatus && item.offset < 28.28&&this.opacity == 0.7) {
				console.log('6')
				clearTimeout(this.timer)
				this.timer = null
				const list = this.value || this.modelValue
				let srcList = list.map(v => this.getSrc(v))
				console.log(list, srcList);
				uni.previewImage({
					urls: srcList,
					current: item.src,
					success: () => {
						this.preStatus = false
						setTimeout(() => {
							this.preStatus = true
						}, 600)
					},
					fail: (e) => {
						console.log(e);
					}
				})
			} else if (this.timer) {
				clearTimeout(this.timer)
				this.timer = null
			}
		},
		mouseenter() {
			//#ifdef H5
			if (this.opacity == 0.7) {
				console.log('7')
				this.imageList.forEach(v => {
					v.disable = false
				})
			}
			//#endif

		},
		mouseleave() {
			//#ifdef H5
			if (this.tempItem && this.opacity == 0.7) {
				this.imageList.forEach(v => {
					v.disable = true
					v.zIndex = v.index + 9
					v.offset = 0
					v.moveEnd = false
					if (v.id == this.tempItem.id) {
						if (this.timer) {
							clearTimeout(this.timer)
							this.timer = null
						}
						v.scale = 1
						v.opacity = 1
						v.x = v.oldX
						v.y = v.oldY
						this.$nextTick(() => {
							v.x = v.absX * this.viewWidth
							v.y = v.absY * this.viewWidth
							this.tempItem = null
						})
					}
				})
				this.changeStatus = true
			}
			//#endif
		},
		// 上传 start
		deleervoider() {
			this.videos = '';
			this.videosUnpload = ''
		},
		selevolos() {
			const that = this;
			const num = Math.floor(Math.random() * (100 - 1)) + 1;
			uni.chooseVideo({
				count: 1,
				sourceType: ['camera', 'album'],
				compressed: false,
				success: (res) => {
					if (res.size > 314572800) {
						uni.showToast({
							title: '视频体积过大',
							icon: 'none',
							duration: 1000
						});
						return
					}
					that.fileListOld = []
					that.videos = res.tempFilePath;
					if (that.videos) {
						that.delevideo = true;
						that.deleimage = false;
					}
					that.uplaodFile();
					// 清空图片切重新排序
					this.imageList = [];
					this.sortList();
				}
			});
		},
		chaoss() {
			let listItem = ['相册', '视频']
			this.limitsathon = false;
			if (this.chooseIndex == 0) {
				listItem = ['相册', '视频']
			} else if (this.chooseIndex == 1) {
				listItem = ['相册']
			} else if (this.chooseIndex == 2) {
				listItem = ['视频']
			}
			if (this.editchooseIndex == 1) {
				listItem = ['相册']
			}
			uni.showActionSheet({
				itemList: listItem,
				success: (res) => {
					const tapIndex = res.tapIndex;
					if (this.chooseIndex == 0) {
						if (tapIndex === 0) {
							//相册
							this.videos = ''
							this.selectPics();
							this.chooseIndex = 1
						} else {
							//视频
							this.fileList = []
							this.uploadImgUrl = ''
							this.selevolos();
							this.chooseIndex = 2
						}
					} else if (this.chooseIndex == 1) {
						//相册
						this.videos = ''
						this.selectPics();
						this.chooseIndex = 1
					} else if (this.chooseIndex == 2) {
						//视频
						this.fileList = []
						this.uploadImgUrl = ''
						this.selevolos();
						this.chooseIndex = 2
					}
				},
				fail() { }
			});
		},
		selectPics() {
			this.checkNumber = this.number - this.imageList.length

			uni.chooseImage({
				count: this.checkNumber,
				sourceType: ['camera', 'album'],
				success: (res) => {

					const files = res.tempFiles;
					this.fileList = []  // 
					for (let i = 0; i < files.length; i++) {
						let obj = new Object();
						obj.name = 'photo' + i;
						obj.uri = files[i].path;
						this.fileList.push(obj);
						if (this.fileList.length > 0) {
							this.deleimage = true;
							this.delevideo = false;
						}
					}
					this.uplaodFile();
				},
				fail(res) { }
			});
		},
		uplaodFile() {
			const value = uni.getStorageSync('token');
			const that = this;
			let host = '';
			let signature = '';
			let ossAccessKeyId = '';
			let policy = '';
			let key = '';
			const date = new Date();
			const nowtime = date.getTime();
			getpolicyz().then((ret) => {
				host = ret.data.host;
				signature = ret.data.signature;
				ossAccessKeyId = ret.data.OSSAccessKeyId;
				policy = ret.data.policy;
				if (that.videos) {
					// 视频
					const parts = that.videos.split('/');
					const result = parts[parts.length - 1];
					key = ret.data.dir + result;
					const uploadTask = wx.uploadFile({
						url: host,
						filePath: that.videos,
						name: 'file',
						formData: {
							key,
							policy,
							OSSAccessKeyId: ossAccessKeyId,
							signature
						},
						success: (res) => {
							console.log("res---->uploadFile", res);
							if (res.statusCode === 204) {
								that.videosUnpload = host + '/' + key;
								console.log("res---->host", res);
								console.log("that.videos---->", that.videosUnpload);
								this.$emit("changeSort", [], that.videosUnpload);
							}
						},
						fail: (err) => {
							console.log("res---->err", err);
							uni.showToast({
								title: '上传失败',
								icon: 'none',
								duration: 800
							});
						}
					});
					uploadTask.onProgressUpdate((v) => {
						that.numBigs = '上传中...' + v.progress + '%';
					});
				} else {
					//图片
					function uploadFileToOSS(file) {
						return new Promise((resolve, reject) => {
							const parts = file.uri.split('/');
							const result = parts[parts.length - 1];
							const key = ret.data.dir + result;
							wx.uploadFile({
								url: host,
								filePath: file.uri,
								name: 'file',
								formData: {
									key,
									policy,
									OSSAccessKeyId: ossAccessKeyId,
									signature
								},
								success: (res) => {
									if (res.data.includes('EntityTooLarge')) {
										reject(new Error('文件体积过大'));
									} else if (res.statusCode === 204) {
										const dat = host + '/' + key;
										resolve(dat);
									} else { }
								},
								fail: (err) => {
									reject(err);
								}
							});
						});
					}
					const uploadPromises = that.fileList.map((file) => uploadFileToOSS(file));
					Promise.all(uploadPromises).then((uploadedUrls) => {
						console.log("res---->uploadedUrls", uploadedUrls);
						if (that.uploadImgUrl == '') {
							that.uploadImgUrl = uploadedUrls;
						} else {
							that.uploadImgUrl = that.uploadImgUrl + ',' + uploadedUrls;
						}
						if (that.uploadImgUrl.indexOf(',') !== -1) {
							that.fileListOld = that.uploadImgUrl.split(',')
						} else {
							that.fileListOld = that.uploadImgUrl
						}
						if (Array.isArray(that.uploadImgUrl)) {
							that.uploadImgUrl = that.uploadImgUrl.join(',')
						}
						// 图片上传成功后在这里需要展示出来 start
						const resImeg = uploadedUrls;
						let count = this.checkNumber <= resImeg.length ? this.checkNumber : resImeg.length
						for (let i = 0; i < count; i++) {
							this.addProperties(resImeg[i])
						}
						// console.log('res.tempFilePaths',res.tempFilePaths)
						// console.log('res.tempFiles',res.tempFiles)
						this.sortList()
						// end
						console.log("that.fileListOld---->", that.fileListOld);
					}).catch((error) => {
						uni.showToast({
							title: error.message,
							icon: 'none',
							duration: 800
						});
					});
				}
			});
		},
		// 上传 end
		addImages() {


			this.chaoss();
			return;
			if (typeof this.addImage === 'function') {
				this.addImage.bind(this.$parent)()
			} else {
				let checkNumber = this.number - this.imageList.length
				uni.chooseImage({
					count: checkNumber,
					sourceType: ['album', 'camera'],
					success: res => {
						let count = checkNumber <= res.tempFilePaths.length ? checkNumber : res
							.tempFilePaths.length
						for (let i = 0; i < count; i++) {
							this.addProperties(res.tempFilePaths[i])
						}
						this.sortList()
					}
				})
			}
		},
		delImages(item, index) {
			if (typeof this.delImage === 'function') {
				this.delImage.bind(this.$parent)(() => {
					this.delImageHandle(item, index)
				})
			} else {
				this.delImageHandle(item, index)
			}
		},
		delImageHandle(item, index) {
			this.imageList.splice(index, 1)
			for (let obj of this.imageList) {
				if (obj.index > item.index) {
					obj.index -= 1
					obj.x = obj.oldX
					obj.y = obj.oldY
					obj.absX = obj.index % this.colsValue
					obj.absY = Math.floor(obj.index / this.colsValue)
					this.$nextTick(() => {
						obj.x = obj.absX * this.viewWidth
						obj.y = obj.absY * this.viewWidth
					})
				}
			}
			this.add.x = (this.imageList.length % this.colsValue) * this.viewWidth + 'px'
			this.add.y = Math.floor(this.imageList.length / this.colsValue) * this.viewWidth + 'px'
			this.sortList()
		},
		delImageMp(item, index) {
			//#ifdef MP
			this.delImages(item, index)
			//#endif
		},
		sortList() {
			console.log('sortList');
			const result = []
			let source = this.value
			// #ifdef VUE3
			source = this.modelValue
			// #endif
			// 使用slice进行深拷贝
			let list = this.imageList.slice()
			// 小到大排序
			list.sort((a, b) => {
				return a.index - b.index
			})
			for (let s of list) {
				let item = source.find(d => this.getSrc(d) == s.src)
				if (item) {
					result.push(item)
				} else {
					if (this.keyName !== null) {
						result.push({
							[this.keyName]: s.src
						})
					} else {
						result.push(s.src)
					}
				}
			}

			this.$emit("input", result);
			this.$emit("update:modelValue", result);
			this.$emit("changeSort", result);
		},
		addProperties(item) {
			console.log(item);
			// 这里的数组长度还没有。计算后push进去才从0开始计算
			// 数组长度取列数的余数 1就取1 2就取2 3就是0 4就是1
			let absX = this.imageList.length % this.colsValue
			// 向下取整数组长度除以列数  1/3取0  4/3取1
			let absY = Math.floor(this.imageList.length / this.colsValue)
			let x = absX * this.viewWidth
			let y = absY * this.viewWidth
			this.imageList.push({
				src: item,
				x,
				y,
				oldX: x,
				oldY: y,
				absX,
				absY,
				scale: 1,
				zIndex: 9,
				opacity: 1,
				index: this.imageList.length,
				id: this.guid(16),
				disable: false,
				offset: 0,
				moveEnd: false
			})
			this.add.x = (this.imageList.length % this.colsValue) * this.viewWidth + 'px'
			this.add.y = Math.floor(this.imageList.length / this.colsValue) * this.viewWidth + 'px'
		},
		nothing() { },
		rpx2px(v) {
			return this.width * v / 750
		},
		guid(len = 32) {
			const chars = '0123456789ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz'.split('')
			const uuid = []
			const radix = chars.length
			for (let i = 0; i < len; i++) uuid[i] = chars[0 | Math.random() * radix]
			uuid.shift()
			return `u${uuid.join('')}`
		}
	}
}
</script>

<style lang="scss" scoped>
.con {
	// padding: 30rpx;

	.area {
		width: 100%;

		.upLoadvideo {
			width: 80%;
			height: 300rpx !important;
			border-radius: 0.5rem;
		}

		.numBig {
			position: absolute;
			bottom: 15rpx;
			left: 0;
			font-size: 25rpx;
			color: #ff882d;
		}

		.delimg {
			display: inline-block;
			position: absolute;
			top: 0;
			left: 0;
			width: 50rpx;
			line-height: 50rpx;
			border-radius: 50%;
			text-align: center;
			background-color: #51c2b3;
			z-index: 888;
		}

		.view {
			display: flex;
			justify-content: center;
			align-items: center;

			.area-con {
				position: relative;
				overflow: hidden;

				.pre-image {
					width: 100%;
					height: 100%;
				}

				.del-con {
					position: absolute;
					top: 0rpx;
					right: 0rpx;
					padding: 0 0 20rpx 20rpx;

					.del-wrap {
						width: 40rpx;
						height: 40rpx;
						// background-color: rgba(0, 0, 0, 0.4);
						border-radius: 0 0 0 10rpx;
						display: flex;
						justify-content: center;
						align-items: center;

						.del-image {
							width: 40rpx;
							height: 40rpx;
						}
					}
				}
			}
		}

		.add {
			position: absolute;
			display: flex;
			justify-content: center;
			align-items: center;

			.add-wrap {
				display: flex;
				justify-content: center;
				align-items: center;
				background-color: #f0f0f0;
				border-radius: 0.5rem;

				image {
					width: 44rpx;
					height: 44rpx;
				}

				// background-color: #eeeeee;
			}
		}
	}
}
</style>