首先,我们需要两张画布,一个展示选取图片的压缩图,一个展示截取后的图片。

废话不多讲上代码如下:

HTML

<!DOCTYPE html>
<html>
	<head>
		<meta charset="UTF-8">
		<meta name="viewport" content="width=device-width,initial-scale=1,minimum-scale=1,maximum-scale=1,user-scalable=no" />
		<!--<meta name="viewport" content="target-densitydpi=320,width=640,user-scalable=no">--><!--1.禁止缩放-->
		<title></title>
		<!--引入vue-->
		<script src="https://cdnjs.cloudflare.com/ajax/libs/vue/2.5.17-beta.0/vue.min.js"></script>
		<!--引入css-->
		<link href="https://cdnjs.cloudflare.com/ajax/libs/twitter-bootstrap/3.3.6/css/bootstrap.min.css" rel="stylesheet">
		<link rel="stylesheet" type="text/css" href="global.css"/>
		<link rel="stylesheet" type="text/css" href="index.css" />	
	</head>
	<body>
		<div id="app" v-cloak>			
			<div class="container">
				<ul class="information creater_class">
					<li class="infor files">
						<div style="width: 100%;height: 300px;border: 1px solid #ccc;" class="center">
						    <div class="center">
						      <canvas style="border: 1px solid #ccc;border-radius: 50%;width: 300px;height: 300px;margin: 0 auto;display: block;" ref="originalCanvas" ref="originalCanvas" @touchstart.prevent="hasImgSrc?pressDown(originalCtx,$event):false" @touchmove.prevent="mouseDown?moveMouse($event):false" @touchend.prevent="stopCut">
						        您的浏览器不支持canvas,请升级最新版本
						      </canvas>
						      <p class="img-scale theme-red center">
						        图片压缩:
						        <span v-text="Math.round(originalCanvasWidth) + ' * ' + Math.round(originalCanvasHeight)"></span>
						      </p>
						      <div v-if="hasImgSrc" class="center">
						          <button @click="chooseSize(cutSize)" type="button" class="btn btn-danger">Min</button>
								  <button @click="reduceSize" type="button" class="btn btn-danger">-</button>
								  <button @click="plusSize" type="button" class="btn btn-danger">+</button>
								  <button @click="chooseSize(scaleSize)" type="button" class="btn btn-danger">Max</button>
						      </div>
						      <div class="bg-info text-center" style="line-height: 50px;height: 50px;">选择图片
						        <input type="file" @change="chooseImg($event)">
						      </div>
						    </div>
						    <div  v-show="hasImgSrc" class="center">
						      <canvas ref="cutCanvas">
						        您的浏览器不支持canvas,请升级最新版本
						      </canvas>
						      <p class="theme-red center">
						        截图长度:
						        <span v-text="Math.round(cutSize) + ' * ' + Math.round(cutSize)"></span>
						      </p>
						      <button @click="submitImg" class="btn btn-info">上传头像</button>
						    </div>
						</div>
					</li>
				</ul>				
			</div>
		</div>
		<script src="index.js" type="text/javascript" charset="utf-8"></script>
	</body>
</html>

JS

//model
var data = {
	//图片
	originalCanvas: null, // 原图 画布
	originalCtx: null, // 原图 上下文
	originalImg: null, // 原图片的对象
	cutCanvas: null, // 显示裁剪后图片的画布
	cutCtx: null, // 裁图 画布 上下文
	cutImg: null, // 裁剪后 图片的对象
	originalCanvasWidth: 0, // 原图 压缩后 宽度
	originalCanvasHeight: 0, // 原图 压缩后 高度
	scaleSize: 300, // 原图压缩的尺寸
	cutSize: 200, // 裁剪 正方形的宽高
	hasImgSrc: false, // 是否有可裁剪图片
	centerX: 0, // 原图画布中心 X坐标
    centerY: 0, // 原图画布中心 Y坐标
    mouseX: 0, // 鼠标所在点 X坐标
    mouseY: 0, // 鼠标所在点 Y坐标
    differenceX: 0, // 原点X - 鼠标X点 差值
    differenceY: 0, // 原点Y - 鼠标Y点 差值
    mouseDown: false, // 鼠标是否点击
    hasImgSrc: false, // 是否有可裁剪图片
    imageDate: null, // 截图 对象
    tempImageData: null, // 实时截图 缓存对象
    imgFormat: ['jpg', 'png'] // 图片上传格式
}
//ViewModel
var app = new Vue({
	el: "#app",
	data: data,
	mounted () {
	    // 初始化 原图的画布
	    let originalCanvas = this.originalCanvas = this.$refs.originalCanvas
	    let originalCtx = this.originalCtx = originalCanvas.getContext('2d')
	    // 初始化 裁图 画布
	    let cutCanvas = this.cutCanvas = this.$refs.cutCanvas
	    this.cutCtx = cutCanvas.getContext('2d')
	    // 设置画布宽高
	    originalCanvas.width = this.originalCanvasWidth = this.scaleSize
	    originalCanvas.height = this.originalCanvasHeight = this.scaleSize
	    cutCanvas.width = this.cutSize
	    cutCanvas.height = this.cutSize
	    // 绘制默认图片
	    if (originalCanvas.getContext) {
	      let originalImg = this.originalImg = new Image()
	      originalImg.src = '';
	      originalImg.onload = () => {
	        originalCtx.drawImage(originalImg, originalCanvas.width / 2 - this.cutSize / 2, originalCanvas.height / 2 - this.cutSize / 2, this.cutSize, this.cutSize)
	      }
	    }
 	},
 	methods:{
 		// 选择图片
		chooseImg:function (e) {
		  // 选中文件择执行
		  if (e.target.files.length !== 0) {		  	
		    // 获取文件
		    let file = e.target.files[0]
		    // 限定上传格式
		    let arr = this.imgFormat
		    let str = file.name.split('.').pop()
		    if (arr.indexOf(str) !== -1) {
		      // 转换成Base 64位
			  this.changeDataURL(file, (dataUrl) => {
			    this.originalImg.src = dataUrl
			    // 获取图片 宽高
			    let width = this.originalImg.width
			    let height = this.originalImg.height
			    
			    //在画布没发生变化时,清除画布区域,重置画布的宽高为默认的值,然后对比宽高,短的一边给默认值(300),长的一边等比计算:
			    // 清除画布
				this.clearCanvas()
				// 重置画布的宽高
				  let originalCanvas = this.originalCanvas
				  originalCanvas.width = this.originalCanvasWidth = this.scaleSize
				  originalCanvas.height = this.originalCanvasHeight = this.scaleSize
				  // 设置画布压缩后的宽高
				  if (width > height) {
				    originalCanvas.width = this.originalCanvasWidth = width * this.scaleSize / height
				  } else if (width < height) {
				    originalCanvas.height = this.originalCanvasHeight = height * this.scaleSize / width
				  }
			    
			      // 设置绘图原点
				  this.centerX = Math.round(originalCanvas.width / 2)
				  this.centerY = Math.round(originalCanvas.height / 2)
				  // 图片加载完成 绘制选中的图片
				  this.originalImg.onload = () => {
				    // 允许鼠标点击事件
				    this.hasImgSrc = true
				    // 绘制图片
				    this.drawOriginal(this.originalImg, originalCanvas.width, originalCanvas.height)
				  }
			    
			    
			  })
		    } else {
		      alert('图片格式错误')
		    }
		  }
		},
		// 转换成dataUrl
		changeDataURL:function (fileObj, callback) {
		  let file = new FileReader()
		  file.readAsDataURL(fileObj)
		  file.onload = (e) => { callback(e.target.result) }		  
		},
		// 清除画布
		clearCanvas:function () {
		  // 绘制原图片
		  this.originalCtx.clearRect(0, 0, this.originalCanvasWidth, this.originalCanvasHeight)
		},
		// 绘制 压缩后的图片
		drawOriginal:function (img, width, height) {
		  // 绘制 原图
		  this.originalCtx.drawImage(img, 0, 0, width, height)
		  // 缓存 截图对象
		  this.imageData = this.originalCtx.getImageData(this.centerX - this.cutSize / 2, this.centerY - this.cutSize / 2, this.cutSize, this.cutSize)
		  // 清除截图区域时使用
		  this.tempImageData = this.originalCtx.getImageData(this.centerX - this.cutSize / 2 - 1, this.centerY - this.cutSize / 2 - 1, this.cutSize + 2, this.cutSize + 2)
		  // 绘制 截图框
		  this.drawCuttingFrame(this.originalCtx, this.centerX, this.centerY, this.cutSize / 2, 20)
		  // 绘制 裁剪图片
		  this.drawCutImage(this.imageData)
		},
		drawCuttingFrame:function (ctx, x, y, l, d) {
		      ctx.beginPath()
		      // 右上框
		      ctx.moveTo(x - 1, y - l)
		      ctx.lineTo(x + 1, y - l)
		      ctx.moveTo(x + d, y - l)
		      ctx.lineTo(x + l, y - l)
		      ctx.lineTo(x + l, y - d)
		      // 右下框
		      ctx.moveTo(x + l, y - 1)
		      ctx.lineTo(x + l, y + 1)
		      ctx.moveTo(x + l, y + d)
		      ctx.lineTo(x + l, y + l)
		      ctx.lineTo(x + d, y + l)
		      // 左下框
		      ctx.moveTo(x - 1, y + l)
		      ctx.lineTo(x + 1, y + l)
		      ctx.moveTo(x - d, y + l)
		      ctx.lineTo(x - l, y + l)
		      ctx.lineTo(x - l, y + d)
		      // 左上框
		      ctx.moveTo(x - l, y - 1)
		      ctx.lineTo(x - l, y + 1)
		      ctx.moveTo(x - l, y - d)
		      ctx.lineTo(x - l, y - l)
		      ctx.lineTo(x - d, y - l)
		      ctx.strokeStyle = 'green'
		      ctx.stroke()
		      ctx.closePath()
		 },
		 // 绘制 裁剪后的图片
		 drawCutImage:function (imageData) {
		    this.cutCtx.putImageData(imageData, 0, 0)
		},
		// 鼠标点击事件
		pressDown:function (ctx,e) {
//			pc:@mousedown.prevent="hasImgSrc ? pressDown(originalCtx, $event) : false" @mousemove="mouseDown ? moveMouse($event) : false" @mouseup="stopCut"
//			pc:this.mouseX = e.offsetX
//			   this.mouseY = e.offsetY
			console.log(e.touches[0].clientX +'...'+e.touches[0].clientY)
		  // 鼠标 坐标
//		  this.mouseX = e.touches[0].clientX
//		  this.mouseY = e.touches[0].clientY
		  this.mouseX = e.touches[0].clientX
		  this.mouseY = e.touches[0].clientY
		  // 原点到鼠标之间的差值
		  this.differenceX = this.centerX - this.mouseX
		  this.differenceY = this.centerY - this.mouseY
		  // 截图框内才能拖动
		  if (Math.abs(this.differenceX) <= this.cutSize / 2 && Math.abs(this.differenceY) <= this.cutSize / 2) {
		  // 允许鼠标移动事件
		    this.mouseDown = true
		  }
		},
		// 鼠标移动事件
		moveMouse:function (e) {
		  if (this.mouseDown) {
		    // 清空 上个路径的截图框 X、Y坐标 - 1 宽高 + 2 彻底清除整个截图框区域
		    this.originalCtx.clearRect(this.centerX - this.cutSize / 2 - 1, this.centerY - this.cutSize / 2 - 1, this.cutSize + 2, this.cutSize + 2)
		    this.originalCtx.putImageData(this.tempImageData, this.centerX - this.cutSize / 2 - 1, this.centerY - this.cutSize / 2 - 1)
		  	// 重新赋值 截图框原点位置
		    if (e.touches[0].clientX > this.cutSize / 2 - this.differenceX && e.touches[0].clientX + this.cutSize / 2 + this.differenceX < this.originalCanvas.width) {
		      // 可移动范围内赋值
		      this.centerX = e.touches[0].clientX + this.differenceX
		    } else {
		      // 不可移动范围内赋值
		      this.centerX = e.touches[0].clientX <= this.cutSize / 2 - this.differenceX ? this.cutSize / 2 : this.originalCanvas.width - this.cutSize / 2
		    }
		    if (e.touches[0].clientY > this.cutSize / 2 - this.differenceY && e.touches[0].clientY + this.cutSize / 2 + this.differenceY < this.originalCanvas.height) {
		       this.centerY = e.touches[0].clientY + this.differenceY
		     } else {
		       this.centerY = e.touches[0].clientY <= this.cutSize / 2 - this.differenceY ? this.cutSize / 2 : this.originalCanvas.height - this.cutSize / 2
		     }
		     // 缓存 截图对象
		     this.imageData = this.originalCtx.getImageData(this.centerX - this.cutSize / 2, this.centerY - this.cutSize / 2, this.cutSize, this.cutSize)
		     // 缓存 清除截图框的 绘图模块
		     this.tempImageData = this.originalCtx.getImageData(this.centerX - this.cutSize / 2 - 1, this.centerY - this.cutSize / 2 - 1, this.cutSize + 2, this.cutSize + 2)
		     // 重绘截图框
		     this.drawCuttingFrame(this.originalCtx, this.centerX, this.centerY, this.cutSize / 2, 20)
		     // 绘制 裁剪图片
		     this.drawCutImage(this.imageData);
		     //阻止页面的滑动默认事件
//		      document.addEventListener("touchmove",function(){
//		          event.preventDefault();
//		      },false);
		  }
		},
		// 松开鼠标点击
		stopCut:function () {
		  // 停止鼠标拖动事件
		  this.mouseDown = false
		},
		// 尺寸减1 重绘
		reduceSize:function () {
		  // 不能小于裁剪大小
		  if (this.originalCanvasWidth > this.cutSize && this.originalCanvasHeight > this.cutSize) {
		    // 清除画布
		    this.clearCanvas()
		    // 固定短边,计算长边的值
		    if (this.originalCanvasWidth > this.originalCanvasHeight) {
		      this.originalCanvas.height = this.originalCanvasHeight -= 1
		      this.originalCanvas.width = this.originalCanvasWidth = this.originalCanvasHeight / (this.originalCanvasHeight + 1) * this.originalCanvasWidth
		    } else if (this.originalCanvasWidth < this.originalCanvasHeight) {
		      this.originalCanvas.width = this.originalCanvasWidth -= 1
		      this.originalCanvas.height = this.originalCanvasHeight = this.originalCanvasWidth / (this.originalCanvasWidth + 1) * this.originalCanvasHeight
		    } else {
		      this.originalCanvas.width = this.originalCanvasWidth -= 1
		      this.originalCanvas.height = this.originalCanvasHeight -= 1
		    }
		    // 设置绘图原点
		    this.centerX = Math.round(this.originalCanvas.width / 2)
		    this.centerY = Math.round(this.originalCanvas.height / 2)
		    // 绘制 原图 截图框
		    this.drawOriginal(this.originalImg, this.originalCanvasWidth, this.originalCanvasHeight)
		  }
		},
		// 尺寸加1 重绘
		plusSize:function () {
		  if (!(this.originalCanvasWidth >= 300 && this.originalCanvasHeight >= 300)) {
		    // 清除画布
		    this.clearCanvas()
		    // 计算长边的值
		    if (this.originalCanvasWidth > this.originalCanvasHeight) {
		      this.originalCanvas.height = this.originalCanvasHeight += 1
		      this.originalCanvas.width = this.originalCanvasWidth = this.originalCanvasHeight / (this.originalCanvasHeight - 1) * this.originalCanvasWidth
		    } else if (this.originalCanvasWidth < this.originalCanvasHeight) {
		      this.originalCanvas.width = this.originalCanvasWidth += 1
		      this.originalCanvas.height = this.originalCanvasHeight = this.originalCanvasWidth / (this.originalCanvasWidth - 1) * this.originalCanvasHeight
		    } else {
		      this.originalCanvas.width = this.originalCanvasWidth += 1
		      this.originalCanvas.height = this.originalCanvasHeight += 1
		    }
		    // 设置绘图原点
		    this.centerX = Math.round(this.originalCanvas.width / 2)
		    this.centerY = Math.round(this.originalCanvas.height / 2)
		    // 绘制 原图 截图框
		    this.drawOriginal(this.originalImg, this.originalCanvasWidth, this.originalCanvasHeight)
		  }
		},
		// 选择最大 最小 缩放比例 重绘
		chooseSize:function (size) {
		  // 清除画布
		  this.clearCanvas()
		  // 计算长边的值
		  let width = this.originalCanvasWidth
		  let height = this.originalCanvasHeight
		  let originalCanvas = this.originalCanvas
		  if (width > height) {
		    originalCanvas.width = this.originalCanvasWidth = size / height * width
		    originalCanvas.height = this.originalCanvasHeight = size
		  } else if (width < height) {
		    originalCanvas.height = this.originalCanvasHeight = size / width * height
		    originalCanvas.width = this.originalCanvasWidth = size
		  } else {
		    originalCanvas.width = this.originalCanvasWidth = size
		    originalCanvas.height = this.originalCanvasHeight = size
		  }
		  // 设置绘图原点,最好取整,有奇怪的bug
		    this.centerX = Math.round(originalCanvas.width / 2)
		    this.centerY = Math.round(originalCanvas.height / 2)
		  // 绘制 原图 截图框
		  this.drawOriginal(this.originalImg, originalCanvas.width, originalCanvas.height)
		},
		// 上传头像
		submitImg:function () {
		  // 生成截图
		  let dataUrl = this.cutCanvas.toDataURL('image/png');
		  console.log(dataUrl)
		}
 	}
});

完整代码:https://download.csdn.net/download/qq_42396791/10731191

Logo

前往低代码交流专区

更多推荐