web功能实例 - Canvas裁剪工具

嗯,手撸官方文档2天,发现没啥用,尤其是动画,那种计算出来的,根本想不到。因此学着学了抱着要做个东西的想法,去网上找相关案例,最终做出了这个裁剪工具。

PS :先说一下思路:

  1. 核心实现有3个canvas图层, 其中一个负责图片的预览。
  2. 另外2个叠加到一起,底层canvas负责图片的渲染; 上层的canvas负责蒙版的绘制,和选择框(挖空)区域的绘制。
  3. 我们将,移动选择框,记录的坐标和宽高,一个是同步到蒙版canvas里面,实现挖空。
  4. 第二个同步到图片渲染canvas,通过getImageData()方法,拿取选择框挖空的区域的像素数据,渲染到图片预览canvas里面,然后通过canvas的toBlob()方法将预览canvas,导出成blobUrl,实现图片下载。

针对部分核心功能进行思路讲解。

1.上传图片

FileReader.readAsDataURL() - Web API 接口参考 | MDN (mozilla.org)

在 web 应用程序中使用文件 - Web API 接口参考 | MDN (mozilla.org)

下面,我们点击upBut,从而触发upInpchange事件,从而触发文件上传。我们对文件上传的类型进行是不是图片判断。然后,将图片File ,通过FileReader对象的readAsDataURL(File)方法,将图片File转换为dataUrl,并封装成Img对象。进而绘制到图片渲染canvas里面。

除此以外涉及到的函数,下文都会讲到。

	<input type="file" id="up-inp" name="文件上传" style="display: none" />
    <button type="button" id="up-but">上传</button>




    let upBut = document.getElementById('up-but')
    let upInp = document.getElementById('up-inp')
	upBut.addEventListener('click', (e) => { // 给按钮绑定事件,点击 input type='file',从而弹出文件上传框
			upInp.click()
		})

		// 上传图片
		const updateFile =  (e) =>{
			let file = e.target.files[0]
			if (!file.type.startsWith('image')) {
				alert('只允许上传图片')
				return
			}
			const reader = new FileReader()
			reader.onload = (e) => {  // 利用fileReader将file文件转换成dataUrl
				let img1 = new Image() // 转换成img对象,进而绘画到canvas里面
				img1.src = e.target.result
				img = img1

				img1.onload = (e) => { 
					// 读取完毕之后
					selectCropObj = computeImage({
						imgWidth: img.width,
						imgHeight: img.height,
						width: cropCardbg.width,
						height: cropCardbg.height,
					})
					initImgObj = JSON.parse(JSON.stringify(selectCropObj))

					drawImage(initImgObj)
					drawModal()
					drawClip()
					drawClipDiv()
					imgPreview()
					cropModal.style.display = 'none'
					clip.style.display = 'block'
				}
			}
			reader.readAsDataURL(file)
		}

		upInp.addEventListener('change', updateFile)

2. 蒙版绘制

蒙版说白了,就是占据canvas画布全部半透明矩形

	// 画模版
		const drawModal =  () =>{
			ctxCardbg.clearRect(0, 0, cropCardbg.width, cropCardbg.height)
			ctxCardbg.fillStyle = 'rgba(0,0,0,0.5)'
			ctxCardbg.fillRect(0, 0, cropCardbg.width, cropCardbg.height)
		}

3.挖空

挖空:蒙版随选择框的移动要扣除的透明区域。 我们先绘制蒙版,然后用clearRect()方法清除指定区域( 随着选择框移动,对应的在蒙版canvas里面的坐标和宽高围绕的区域,清除这个区域),达到挖空的效果

        // 挖空
		const drawClip = () => {
			ctxCardbg.clearRect(selectCropObj.x, selectCropObj.y, selectCropObj.w, selectCropObj.h)
		}

 4.选择框的绘制

我们先封装一个函数,用来注册拖拽选择框(中心,和8个点)的鼠标按下、移动、抬起事件

 

// 注册选择框拖拽事件
		const registerEvents = () => {
			// 注册那8个拖拽点事件
			const register = (_class) => {
				let node = document.querySelector(`.${_class}`)
				node.addEventListener('mousedown', (e) => {
					down = true
				})
				node.addEventListener('mousemove', carMouseMove)

				node.addEventListener('mouseup', function (e) {
					down = false
				})
			}

			register('top-center')
			register('bottom-center')
			register('left-center')
			register('right-center')
			register('bottom-right')
			register('bottom-left')
			register('top-right')
			register('top-left')

			// 注册拖拽中央移动的事件
			let clip = document.getElementById('crop-clip')
			clip.addEventListener('mousedown', (e) => {
				down = true
			})

			clip.addEventListener('mousemove', carMouseMove)

			clip.addEventListener('mouseup', (e) => {
				down = false
			})
		}

这是拖拽选择框的html 结构。我们的选择框采用html绘制,由于它和另外2个叠加的canvas,由于父元素相对定位,子绝对定位叠在一起,因此选择框的left和top值,就相当于canvas里面的x、y坐标,我们同样将选择框宽高映射到,canvas的挖空区域,从而在拖拽的时候实现实时挖空的效果。

 

	<div id="crop-clip" style="display: none">
				<div class="dot top-left"></div>
				<div class="dot top-right"></div>
				<div class="dot top-center"></div>
				<div class="dot bottom-left"></div>
				<div class="dot bottom-center"></div>
				<div class="dot bottom-right"></div>
				<div class="dot left-center"></div>
				<div class="dot right-center"></div>
			</div>

下面通过selectCropObj对象,记录选择框在移动期间的x,y坐标,以及选择框变化的宽高

	// 在canvas的裁剪框尺寸和坐标
		let selectCropObj = {
			x: 0,
			y: 0,
			w: 0,
			h: 0,
		}

 这是我们处理选择框移动的函数。根据拖拽的元素(选择框中央、其余8个点)的携带的class不同,从而调用不同的移动处理方法。

	// 通过拖拽事件,调用的方法(将选择框中央拖拽 ,和点拖拽通过一个函数处理)
		const carMouseMove = (e) => {
			// 两个都是false,证明我们一个没按下
			if (img === null) {
				return
			}

			if (!down) {
				return
			}

			let ele = e.target
			let { movementX, movementY } = e

			const isExistsCls = (_cls) => {
				for (let i = 0; i < ele.classList.length; i++) {
					const val = ele.classList[i]
					if (val === _cls) {
						return true
					}
				}
				return false
			}

			// 中央拖拽
			if (ele.id == 'crop-clip') {
				selectCropObj.x += movementX
				selectCropObj.y += movementY
				// 点拖拽
			} else if (isExistsCls('top-left')) {
				// 坐标和宽高的都要变
				selectCropObj.x += movementX
				selectCropObj.y += movementY
				selectCropObj.w += -movementX
				selectCropObj.h += -movementY
			} else if (isExistsCls('top-right')) {
				selectCropObj.y += movementY
				selectCropObj.w += movementX
				selectCropObj.h += -movementY
			} else if (isExistsCls('top-center')) {
				selectCropObj.y += movementY
				selectCropObj.h += -movementY
			} else if (isExistsCls('bottom-left')) {
				selectCropObj.x += movementX
				selectCropObj.y += movementY
				selectCropObj.w += -movementX
				selectCropObj.h += movementY
			} else if (isExistsCls('bottom-right')) {
				selectCropObj.w += movementX
				selectCropObj.h += movementY
			} else if (isExistsCls('bottom-center')) {
				selectCropObj.h += movementY
			} else if (isExistsCls('left-center')) {
				selectCropObj.x += movementX
				selectCropObj.w += -movementX
			} else if (isExistsCls('right-center')) {
				selectCropObj.w += movementX
			}

			drawClipDiv()
			drawModal()
			drawClip()
			imgPreview()
		}

 对应选择框的具体绘制方法。

		const drawClipDiv = () => {
			let cropClip = document.getElementById('crop-clip')
			cropClip.style.width = `${selectCropObj.w}px`
			cropClip.style.height = `${selectCropObj.h}px`
			cropClip.style.left = `${selectCropObj.x}px`
			cropClip.style.top = `${selectCropObj.y}px`
		}

5.画图像

就是清除之前渲染到图片canvas里面的图像,并将新的图片绘制到上面。

	// 画图像
		const drawImage = ({ x, y, w, h }) => {
			ctxImg.clearRect(0, 0, cropImg.width, cropImg.height)
			ctxImg.drawImage(img, x, y, w, h)
		}

6.图片旋转

CanvasRenderingContext2D.rotate() - Web API 接口参考 | MDN (mozilla.org)

关于图片旋转这里,就是现将原点移动canvas画布中央(图像中央),然后定义旋转角度,然后又将原点移动回去,然后画的图像就是围绕中心旋转的。

主要是围绕图像的中心原点旋转。

大家可以自行去网上找关于rotate让图像围绕中心原点旋转的问题。因为作者也没能明白。

// 主要是这段代码:

            ctxImg.translate(cropImg.width / 2, cropImg.height / 2)
            ctxImg.rotate(angle)
            ctxImg.translate(-cropImg.width / 2, -cropImg.height / 2)

	// 向右、左转(每次向左30度、向右30度)图片
		const imgRotate = (e, t = 1) => {
			if (img == null) {
				alert('请先上传图片')
				return
			}
			if (t == 1) {
				// 向右转,逆时针
				angle += -30 * (Math.PI / 180)
			} else {
				// 向左转,顺时针
				angle += 30 * (Math.PI / 180)
			}
			ctxImg.clearRect(0, 0, cropImg.width, cropImg.height)
			ctxImg.translate(cropImg.width / 2, cropImg.height / 2)
			ctxImg.rotate(angle)
			ctxImg.translate(-cropImg.width / 2, -cropImg.height / 2)
			drawImage(initImgObj)
			imgPreview()
		}

7.图片扩大和缩小

通过initImgObj对象,记录图像渲染到图片渲染canvas尺寸和坐标

// 图像渲染到到canvas的坐标尺寸和坐标
		let initImgObj = {
			// 主要是为了缩放/扩大图片用
			x: 0,
			y: 0,
			w: 0,
			h: 0,
		}

 通过computeImage()函数,计算出图像要在canvas画布中央实际渲染的坐标和尺寸


		// 计算图像在canvas实际渲染的图片坐标和宽高。(居于canvas中央,尺寸小于画布尺寸)
		const computeImage = function ({ imgWidth, imgHeight, width, height, base = 1 }) {
			if (imgWidth / base < width && imgHeight / base < height) {
				return {
					x: (width - imgWidth) / 2,
					y: (height - imgHeight) / 2,
					w: imgWidth / base,
					h: imgHeight / base,
				}
			}
			return computeImage({
				imgWidth,
				imgHeight,
				width,
				height,
				base: base + 0.1,
			})
		}

 通过计算 initImgObj对象的放大、缩小后的宽高,然后通过computeImage()方法计算出canvas实际渲染的图像坐标和尺寸,并渲染。


		// 缩小/扩大图片
		const imgScale = (e, t) => {
			if (img == null) {
				alert('请先上传图片')
				return
			}
			let { w, h } = initImgObj

			if (t == 1) {
				// 缩小
				w = w / 2
				h = h / 2
			} else {
				// 扩大
				w = w * 2
				h = h * 2
			}

			// 重新求出,缩小、放大之后的图片宽高和坐标。
			initImgObj = computeImage({
				imgWidth: w,
				imgHeight: h,
				width: cropImg.width,
				height: cropImg.height,
			})

			drawImage(initImgObj)
			imgPreview()
		}

8.图片预览

ImageData - Web API 接口参考 | MDN (mozilla.org)

通过选择框拖拽,记录的 selectCropObj对象的坐标和宽高信息

通过getImageData()方法获取图片渲染canvas里面的ImageData对象,并通过putImageData()方法将这个对象渲染到图片预览canvas里面。

// 图片预览
		const imgPreview = () => {
			ctxPreview.clearRect(0, 0, cropPreview.width, cropPreview.height)
			let { x, y, w, h } = selectCropObj
			let imgData = ctxImg.getImageData(x, y, w, h)
			ctxPreview.putImageData(imgData, x, y)
		}

9.保存预览图片

HTMLCanvasElement.toBlob() - Web API 接口参考 | MDN (mozilla.org)

HTMLCanvasElement.toDataURL() - Web API 接口参考 | MDN (mozilla.org)

我们通过,canvas的toBlob()方法将canvas画布转换为blobUrl,然后就是创建一个a 元素,并将这个url封装成a的href属性,用日期作为下载的图片名字,用js模拟点击实现下载。

// 保存裁剪图片
		const saveImg = (e) => {
			if (img == null) {
				alert('请先上传图片')
				return
			}
			cropPreview.toBlob((blob) => {
                // 下载图片
				let a = document.createElement('a')
				a.href = window.URL.createObjectURL(blob)
				a.download = `${getDateStr(new Date())}.png` 
				a.dispatchEvent(new MouseEvent('click'))
			})
		}

完整代码

<!DOCTYPE html>
<html lang="en">
	<head>
		<meta charset="UTF-8" />
		<meta name="viewport" content="width=device-width, initial-scale=1.0" />
		<title>图片裁剪工具</title>
		<style>
			* {
				padding: 0;
				margin: 0;
			}

			#crop-tool {
				position: relative;
				overflow: hidden;
				width: 500px;
				height: 300px;
				margin: 100px auto;
				border: 1px solid black;
				border-radius: 15px;
			}
			#crop-img {
				z-index: -1;
				top: 0;
				left: 0;
				background: url('./img/mosaic.jpg');
			}

			#crop-cardbg {
				position: absolute;
				top: 0;
				left: 0;
			}
			#crop-modal {
				position: absolute;
				top: 0;
				left: 0;
			}
			#crop-clip {
				position: absolute;
				left: 25px;
				top: 37.1364px;
				cursor: all-scroll;
				width: 308.091px;
				height: 197.864px;
				border: 1px solid rgb(30, 158, 251);
			}

			#crop-modal {
				position: absolute;
				left: 0;
				top: 0;
				width: 500px;
				height: 300px;
				line-height: 300px;
				font-size: 30px;
				font-weight: 700;
				text-align: center;
				color: white;
				background-color: rgba(0, 0, 0, 0.5);
			}
			#crop-preview {
				border: 1px solid black;
				border-radius: 15px;
				width: 500px;
				height: 300px;
				margin: 0 700px;
			}

			#crop-but {
				width: 400px;
				margin: 0 auto;
			}
			button {
				width: 50px;
			}

			#crop-clip .dot {
				position: absolute;
				width: 20px;
				height: 20px;
				border-radius: 50%;
				background: #1e9efb;
			}

			#crop-clip .top-left {
				top: -10px;
				left: -10px;
				cursor: nwse-resize;
			}

			#crop-clip .top-right {
				top: -10px;
				right: -10px;
				cursor: nesw-resize;
			}

			#crop-clip .top-center {
				top: -10px;
				left: 50%;
				transform: translate(-50%);
				cursor: ns-resize;
			}

			#crop-clip .bottom-left {
				bottom: -10px;
				left: -10px;
				cursor: nesw-resize;
			}

			#crop-clip .bottom-center {
				bottom: -10px;
				left: 50%;
				transform: translate(-50%);
				cursor: ns-resize;
			}

			#crop-clip .bottom-right {
				bottom: -10px;
				right: -10px;
				cursor: nwse-resize;
			}

			#crop-clip .left-center {
				top: 50%;
				transform: translateY(-50%);
				left: -10px;
				cursor: ew-resize;
			}

			#crop-clip .right-center {
				top: 50%;
				transform: translateY(-50%);
				right: -10px;
				cursor: ew-resize;
			}
		</style>
		<!-- <link rel="stylesheet" href="./css/iconfont.css" /> -->
	</head>
	<body>
		<div id="crop-tool">
			<canvas id="crop-img" width="500" height="300"></canvas>
			<canvas id="crop-cardbg" width="500" height="300"></canvas>
			<div id="crop-clip" style="display: none">
				<div class="dot top-left"></div>
				<div class="dot top-right"></div>
				<div class="dot top-center"></div>
				<div class="dot bottom-left"></div>
				<div class="dot bottom-center"></div>
				<div class="dot bottom-right"></div>
				<div class="dot left-center"></div>
				<div class="dot right-center"></div>
			</div>
			<div id="crop-modal">请先上传图片</div>
		</div>
		<canvas id="crop-preview" width="500" height="300"> </canvas>
		<div id="crop-but">
			<input type="file" id="up-inp" name="文件上传" style="display: none" />
			<button type="button" id="up-but">上传</button>
			<button type="button" id="add-but">+</button>
			<button type="button" id="del-but">-</button>
			<button type="button" id="xzz-but">
				left
				<!-- <span class="iconfont icon-xiangzuoxuanzhuan"></span> -->
			</button>
			<button type="button" id="xyz-but">
				right
				<!-- <span class="iconfont icon-xiangyouxuanzhuan"></span> -->
			</button>

			<button type="button" id="save-but">截图</button>
		</div>
	</body>
	<script src="./js/Index.js"></script>
	<script>
		///
		 variable
		///

		// 按钮组
		let upBut = document.getElementById('up-but')
		let saveBut = document.getElementById('save-but')
		let addBut = document.getElementById('add-but')
		let delBut = document.getElementById('del-but')
		let xzzBut = document.getElementById('xzz-but')
		let xyzBut = document.getElementById('xyz-but')

		let upInp = document.getElementById('up-inp')

		let cropImg = document.querySelector('#crop-img') // 背景层
		let cropCardbg = document.querySelector('#crop-cardbg') // 裁剪层
		let cropPreview = document.querySelector('#crop-preview') // 裁剪层
		let cropModal = document.querySelector('#crop-modal') // 遮罩层
		let clip = document.getElementById('crop-clip') // 选择框

		let ctxCardbg = cropCardbg.getContext('2d')
		let ctxImg = cropImg.getContext('2d')
		let ctxPreview = cropPreview.getContext('2d')

		// 在canvas的裁剪框尺寸和坐标
		let selectCropObj = {
			x: 0,
			y: 0,
			w: 0,
			h: 0,
		}

		// 一开始存储到canvas的坐标尺寸和坐标
		let initImgObj = {
			// 主要是为了缩放/扩大图片用
			x: 0,
			y: 0,
			w: 0,
			h: 0,
		}
		let img = null
		let down = false // 中心拖拽或者按钮拖拽
		let angle = 0 // 旋转角度

		///
		 method
		///
		upBut.addEventListener('click', (e) => {
			upInp.click()
		})

		// 上传图片
		const updateFile = function (e) {
			let file = e.target.files[0]
			if (!file.type.startsWith('image')) {
				alert('只允许上传图片')
				return
			}
			const reader = new FileReader()
			reader.onload = (e) => {
				let img1 = new Image()
				img1.src = e.target.result
				img = img1

				img1.onload = (e) => {
					// 读取完毕之后
					selectCropObj = computeImage({
						imgWidth: img.width,
						imgHeight: img.height,
						width: cropCardbg.width,
						height: cropCardbg.height,
					})
					initImgObj = JSON.parse(JSON.stringify(selectCropObj))

					drawImage(initImgObj)
					drawModal()
					drawClip()
					drawClipDiv()
					imgPreview()
					cropModal.style.display = 'none'
					clip.style.display = 'block'
				}
			}
			reader.readAsDataURL(file)
		}

		upInp.addEventListener('change', updateFile)

		// 保存裁剪图片
		const saveImg = (e) => {
			if (img == null) {
				alert('请先上传图片')
				return
			}
			cropPreview.toBlob((blob) => {
				let a = document.createElement('a')
				a.href = window.URL.createObjectURL(blob)
				a.download = `${getDateStr(new Date())}.png`
				a.dispatchEvent(new MouseEvent('click'))
			})
		}

		saveBut.addEventListener('click', saveImg)

		// 图片预览
		const imgPreview = () => {
			ctxPreview.clearRect(0, 0, cropPreview.width, cropPreview.height)
			let { x, y, w, h } = selectCropObj
			let imgData = ctxImg.getImageData(x, y, w, h)
			ctxPreview.putImageData(imgData, x, y)
		}

		// 缩小/扩大图片

		const imgScale = (e, t) => {
			if (img == null) {
				alert('请先上传图片')
				return
			}
			let { w, h } = initImgObj

			if (t == 1) {
				// 缩小
				w = w / 2
				h = h / 2
			} else {
				// 扩大
				w = w * 2
				h = h * 2
			}

			// 重新求出,缩小,方法的图片坐标
			initImgObj = computeImage({
				imgWidth: w,
				imgHeight: h,
				width: cropImg.width,
				height: cropImg.height,
			})

			drawImage(initImgObj)
			imgPreview()
		}

		// 向右、左转(每次向左90度、向右90度)图片
		const imgRotate = (e, t = 1) => {
			if (img == null) {
				alert('请先上传图片')
				return
			}
			if (t == 1) {
				// 向右转,逆时针
				angle += -30 * (Math.PI / 180)
			} else {
				// 向左转,顺时针
				angle += 30 * (Math.PI / 180)
			}
			ctxImg.clearRect(0, 0, cropImg.width, cropImg.height)
			ctxImg.translate(cropImg.width / 2, cropImg.height / 2)
			ctxImg.rotate(angle)
			ctxImg.translate(-cropImg.width / 2, -cropImg.height / 2)
			drawImage(initImgObj)
			imgPreview()
		}

		delBut.addEventListener('click', (e) => {
			imgScale(e, 1)
		})

		addBut.addEventListener('click', (e) => {
			imgScale(e, 2)
		})

		xzzBut.addEventListener('click', (e) => {
			imgRotate(e, 1)
		})

		xyzBut.addEventListener('click', (e) => {
			imgRotate(e, 2)
		})

		// 画模版
		const drawModal = function () {
			ctxCardbg.clearRect(0, 0, cropCardbg.width, cropCardbg.height)
			ctxCardbg.fillStyle = 'rgba(0,0,0,0.5)'
			ctxCardbg.fillRect(0, 0, cropCardbg.width, cropCardbg.height)
		}

		// 计算图片在canvas实际坐标和尺寸
		const computeImage = function ({ imgWidth, imgHeight, width, height, base = 1 }) {
			if (imgWidth / base < width && imgHeight / base < height) {
				return {
					x: (width - imgWidth) / 2,
					y: (height - imgHeight) / 2,
					w: imgWidth / base,
					h: imgHeight / base,
				}
			}
			return computeImage({
				imgWidth,
				imgHeight,
				width,
				height,
				base: base + 0.1,
			})
		}

		// 画图像
		const drawImage = function ({ x, y, w, h }) {
			ctxImg.clearRect(0, 0, cropImg.width, cropImg.height)
			ctxImg.drawImage(img, x, y, w, h)
		}

		// 挖空
		const drawClip = function () {
			ctxCardbg.save()
			ctxCardbg.clearRect(selectCropObj.x, selectCropObj.y, selectCropObj.w, selectCropObj.h)
			ctxCardbg.restore()
		}

		// 挖空时,选择框的变化
		const drawClipDiv = () => {
			let cropClip = document.getElementById('crop-clip')
			cropClip.style.width = `${selectCropObj.w}px`
			cropClip.style.height = `${selectCropObj.h}px`
			cropClip.style.left = `${selectCropObj.x}px`
			cropClip.style.top = `${selectCropObj.y}px`
		}

		// 卡片拖拽事件,调用的方法(将选择框中央拖拽 ,和点拖拽一个方法处理)
		const carMouseMove = (e) => {
			// 两个都是false,证明我们一个没按下
			if (img === null) {
				return
			}

			if (!down) {
				return
			}

			let ele = e.target
			let { movementX, movementY } = e

			const isExistsCls = (_cls) => {
				for (let i = 0; i < ele.classList.length; i++) {
					const val = ele.classList[i]
					if (val === _cls) {
						return true
					}
				}
				return false
			}

			// 中央拖拽
			if (ele.id == 'crop-clip') {
				selectCropObj.x += movementX
				selectCropObj.y += movementY
				// 点拖拽
			} else if (isExistsCls('top-left')) {
				// 坐标和宽高的都要变
				selectCropObj.x += movementX
				selectCropObj.y += movementY
				selectCropObj.w += -movementX
				selectCropObj.h += -movementY
			} else if (isExistsCls('top-right')) {
				selectCropObj.y += movementY
				selectCropObj.w += movementX
				selectCropObj.h += -movementY
			} else if (isExistsCls('top-center')) {
				selectCropObj.y += movementY
				selectCropObj.h += -movementY
			} else if (isExistsCls('bottom-left')) {
				selectCropObj.x += movementX
				selectCropObj.y += movementY
				selectCropObj.w += -movementX
				selectCropObj.h += movementY
			} else if (isExistsCls('bottom-right')) {
				selectCropObj.w += movementX
				selectCropObj.h += movementY
			} else if (isExistsCls('bottom-center')) {
				selectCropObj.h += movementY
			} else if (isExistsCls('left-center')) {
				selectCropObj.x += movementX
				selectCropObj.w += -movementX
			} else if (isExistsCls('right-center')) {
				selectCropObj.w += movementX
			}

			drawClipDiv()
			drawModal()
			drawClip()
			imgPreview()
		}

		// 注册选择框拖拽事件
		const registerEvents = () => {
			// 注册那8个拖拽点事件
			const register = (_class) => {
				let node = document.querySelector(`.${_class}`)
				node.addEventListener('mousedown', (e) => {
					down = true
				})
				node.addEventListener('mousemove', carMouseMove)

				node.addEventListener('mouseup', function (e) {
					down = false
				})
			}

			register('top-center')
			register('bottom-center')
			register('left-center')
			register('right-center')
			register('bottom-right')
			register('bottom-left')
			register('top-right')
			register('top-left')

			// 注册拖拽中央移动的事件
			let clip = document.getElementById('crop-clip')
			clip.addEventListener('mousedown', (e) => {
				down = true
			})

			clip.addEventListener('mousemove', carMouseMove)

			clip.addEventListener('mouseup', (e) => {
				down = false
			})
		}

		registerEvents()
	</script>
</html>
const getDateStr = (time, tag1 = '-', tag2 = ':') => {
    const date = new Date(time)

    let y = date.getFullYear()
    let M = date.getMonth() + 1
    let d = date.getDate()
    let h = date.getHours()
    let m = date.getMinutes()
    let s = date.getSeconds()

    return `${y}${tag1}${M}${tag1}${d} ${h}${tag2}${m}${tag2}${s}`
}

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:/a/271221.html

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

\r\n和缓冲区/进度条小程序

一 前置知识 带有\n就会立马刷新缓冲区&#xff0c;\r不会刷新缓冲区 刷新的2个场景: 1 ~fflush 缓冲区中存在\r或\n --> \r fflush --> 不换行的\n) 2 ~ 文件关闭自动刷新缓冲区 倒计时小程序0-9 倒计时小程序0-99

信号与线性系统翻转课堂笔记11——连续LTI系统频域分析

信号与线性系统翻转课堂笔记11——连续LTI系统频域分析 The Flipped Classroom11 of Signals and Linear Systems 对应教材&#xff1a;《信号与线性系统分析&#xff08;第五版&#xff09;》高等教育出版社&#xff0c;吴大正著 一、要点 &#xff08;1&#xff0c;重点&…

Java基础知识复习

目录 一、Java语言基础知识&#xff0c;包括面向对象编程、语法特性等 0、什么是面向对象编程&#xff1f; 1、基本语法&#xff1a; 2、变量和数据类型&#xff1a; 3、运算符和表达式&#xff1a; 4、控制结构&#xff1a; 5、函数和方法&#xff1a; 6、类和对象&#xff1a…

因吹斯汀!只需上传照片,GPT-4V精准识别食物的卡路里和摄入热量

健身和减肥的朋友有福啦&#xff01; 最近一篇文章探索了GPT-4V在膳食评估领域的强大能力&#xff0c;可以根据饮食图片精准判断食物的种类与重量&#xff0c;并给出营养成分的分析&#xff0c;包括碳水化合物、蛋白质、脂肪占比。 最最重要的是&#xff0c;它还能告诉我们这…

算法与数据结构--二叉搜索树与自平衡二叉搜索树

0.字典&#xff08;即c的map&#xff09; 注&#xff1a;字典的 "member运算" 指的是检查字典中是否存在某个特定的键的操作&#xff0c;即查询操作。 如果我们使用数组来实现字典/map&#xff0c;虽然使用二分法查询也可以达到logn&#xff0c;但是的话插入和删除太…

GPT每预测一个token就要调用一次模型

问题&#xff1a;下图调用了多少次模型&#xff1f; 不久以前我以为是调用一次 通过看代码是输出多少个token就调用多少次&#xff0c;如图所示&#xff1a; 我理解为分类模型 预测下一个token可以理解为分类模型&#xff0c;类别是vocab的所有token&#xff0c;每一次调用都…

Description:An attempt was made to call a method that does not exist.

错误信息 Description: An attempt was made to call a method that does not exist. The attempt was made from the following location: okio.Segment.writeTo(Segment.kt:169) The following method did not exist: kotlin.collections.ArraysKt.copyInto([B[BIII)[B T…

多行文本(多行字符串)中,如果每行文本前都有空格,各行文本前空格数最小为n,则删除每行文本前的n个空格。textwrap.dedent(多行字符串)

【小白从小学Python、C、Java】 【计算机等考500强证书考研】 【Python-数据分析】 多行文本(多行字符串)中&#xff0c; 如果每行文本前都有空格&#xff0c; 各行文本前空格数最小为n&#xff0c; 则删除每行文本前的n个空格。 textwrap.dedent(多行字符串) 请问以下代…

ES的安装和RestClient的操作

目录 初识elasticsearch 什么是elasticsearch elasticsearch的发展 Lucene的优缺点 elasticsearch的优势 倒排索引 es与mysql的概念对比 文档 索引 概念对比 架构 安装es 安装kibana 安装ik分词器 分词器 安装ik分词器 ik分词器的拓展和停用词典 操作索引库…

c语言中数据结构

一、结构体的由来 1. 数据类型的不足 C语言中&#xff0c;基本数据类型只有整型、字符型、浮点型等少数几种&#xff0c;无法满足复杂数据类型的需要。 2. 数组的限制 虽然数组可以存储多个同类型的数据&#xff0c;但是数组中的元素个数是固定的&#xff0c;无法动态地改变…

Unity VR Pico apk安装失败:INSTALL_FAILED_UPDATE_INCOMPATIBLE

我的报错&#xff1a; PICO4企业版。安装apk&#xff0c;报错“安装失败。&#xff08;所属的Unity项目打包的apk&#xff0c;被我在同一台pico4安装了20次&#xff09; 调试方法&#xff1a; PIco4发布使用UNITY开发的Vr应用&#xff0c;格式为apk&#xff0c;安装的时候发生…

SQL手工注入漏洞测试(MySQL数据库)

一、实验平台 https://www.mozhe.cn/bug/detail/elRHc1BCd2VIckQxbjduMG9BVCtkZz09bW96aGUmozhe 二、实验目标 获取到网站的KEY&#xff0c;并提交完成靶场。 三、实验步骤 ①、启动靶机&#xff0c;进行访问查找可能存在注入的页面 ②、通过测试判断注入点的位置(id) (1)…

嵌入式-stm32-用PWM点亮LED实现呼吸灯

一&#xff1a;知识前置 1.1、LED灯怎么才能亮&#xff1f; 答&#xff1a;LED需要低电平才能亮&#xff0c;高电平是灯灭。 1.2、LED灯为什么可以越来越亮&#xff0c;越来越暗&#xff1f; 答&#xff1a;这是用到不同占空比来实现的&#xff0c;控制LED实现呼吸灯&…

matlab时间转换

采集的GNSS数据是10hz的。 data&#xff08;选取其中一部分&#xff09;如下&#xff1a; &#xff08;1&#xff09;char类型 formatOut yyyy-mm-dd HH:MM:SS; str datestr(data,formatOut); str如下&#xff1a; &#xff08;2&#xff09;double类型 DateVector dat…

STM32独立看门狗

时钟频率 40KHZ 看门狗简介 STM32F10xxx 内置两个看门狗&#xff0c;提供了更高的安全性、时间的精确性和使用的灵活性。两个看 门狗设备 ( 独立看门狗和窗口看门狗 ) 可用来检测和解决由软件错误引起的故障&#xff1b;当计数器达到给 定的超时值时&#xff0c;触发一个中…

WU反走样算法

WU反走样算法 由离散量表示连续量而引起的失真称为走样&#xff0c;用于减轻走样现象的技术成为反走样&#xff0c;游戏中称为抗锯齿。走样是连续图形离散为想想点后引起的失真&#xff0c;真实像素面积不为 零。走样是光栅扫描显示器的一种固有现象&#xff0c;只能减轻&…

Drogon Win11 编译 /MT

Drogon是一个基于C17/20的Http应用框架&#xff0c;使用Drogon可以方便的使用C构建各种类型的Web应用服务端程序。 Drogon的主要应用平台是Linux&#xff0c;也支持Mac OS、FreeBSD和Windows。 它的主要特点如下&#xff1a; 网络层使用基于epoll(macOS/FreeBSD下是kqueue)的…

nginx反向代理服务器及负载均衡服务配置

一、正向代理与反向代理 正向代理&#xff1a;是一个位于客户端和原始服务器(oricin server)之间的服务器&#xff0c;为了从原始服务器取得内容&#xff0c;客户端向代理发送一个请求并指定目标(原始服务器)&#xff0c;然后代理向原始服务器转交请求并将获得的内容返回给客户…

Matlab/Simulink的一些功能用法笔记(3)

01--引言 最近加入到一个项目组&#xff0c;有一些测试需要去支持&#xff0c;通过了解原先团队的测试方法后&#xff0c;自己作了如下改善&#xff0c;大大提高了工作效率。这也许就是软件开发的意义吧&#xff0c;能够去除一些重复的机械的人工操作并且结果还非常不可靠。 …

Discrete Time Signals and Systems

Discrete Time Signals and Systems 文章目录 Discrete Time Signals and SystemsSignal classificationbasic signalOperation on signalSystem of discrete signalLinear systems and nonlinear systemsCausal and non-causal SystemsTime-varying and time-invariant system…