要在小程序中实现图片放大缩小是着实不容易呀,还要把用户选择的指定区域生成图片,简直令人发指。
不多说,上烧鸡(代码)
首先还是先来看看要实现的效果
用户可以在指定的区域中滑动,放大,缩小自己的图片,点击确定之后,把方框中的区域生成图片。这个和上传头像功能差不多。
首页我们要做的就是怎么才能让照片完美的显示在页面。图片一般分为横版和竖版图片,这两种情况要分开显示。
wx.chooseImage({
count: 1,
sizeType: [\'original\'],
success: function (res) {
wx.getImageInfo({
src: res.tempFilePaths[0],
success: function (res) {
//console.log(res.width);
//console.log(res.height);
var str = res.width / res.height;
if(str > 1){//横版图片
}else{//竖版图片
}
}
})
}
})
- 1
- 2
- 3
- 4
- 5
- 6
- 7
- 8
- 9
- 10
- 11
- 12
- 13
- 14
- 15
- 16
- 17
- 18
- 19
调用wx.getImageInfo方法,拿到用户选择的图片的宽高,用宽除以高就可以判断图片的类型了。
接下来就是处理图片的宽高了,假设方框的大小是400*400,那么:
wx.chooseImage({
count: 1,
sizeType: [\'original\'],
success: function (res) {
wx.getImageInfo({
src: res.tempFilePaths[0],
success: function (res) {
//console.log(res.width);
//console.log(res.height);
var str = res.width / res.height;//图片的宽高比
if(str > 1){//横版图片
_this.data.height = 400;//图片的显示高度为400
_this.data.width = str * _this.data.height; //图片的宽度 = 宽高比 * 图片的显示高度
}else{//竖版图片
_this.data.width = 400;//图片的显示宽度为400
_this.data.height = str * _this.data.width; //图片的高度 = 宽高比 * 图片的显示宽度
}
}
})
}
})
- 1
- 2
- 3
- 4
- 5
- 6
- 7
- 8
- 9
- 10
- 11
- 12
- 13
- 14
- 15
- 16
- 17
- 18
- 19
- 20
- 21
- 22
这样,当是横版图片的时候,用户能左右滑动自己的图片,上下不让他滑动,竖版同理。
关于怎么让图片和方框居中显示,我是用的padding填充,不是啥难事,这个就自己去摸索吧。
完成了图片的显示之后,接下来就是进入正题了,首先我们先完成图片的放大缩小功能。
在小程序中提供了touchstart,touchmove,touchend这三个方法,分别是手指触摸动作开始,手指触摸后移动,手指触摸动作结束,我们要用到这三个方法来完成图片的放大缩小功能。
wxml:
<scroll-view scroll-y="true" scroll-x="true" class = "FilePath" bindtouchmove="scroll" bindtouchstart=\'scroll\' bindtouchend=\'endTou\'>
<image src=\'{{src}}\' style="width: {{width}}px;height: {{height}}px;" bindtouchmove="touch"></image>
</scroll-view>
- 1
- 2
- 3
监听手指触摸的方法:
olddistance,//上一次两个手指的距离
newdistance:"",//本次两手指之间的距离,两个一减咱们就知道了滑动了多少,以及放大还是缩小(正负嘛)
diffdistance:\'\', //这个是新的比例,新的比例一定是建立在旧的比例上面的,给人一种连续的假象
Scale: 1,//图片放大的比例,
baseHeight:\'\', //原始高
baseWidth:\'\', //原始宽
//手指在屏幕上移动
scroll: function (e) {
var _this = this;
//当e.touches.length等于1的时候,表示是单指触摸,我们要的是双指
if (e.touches.length == 2) {//两个手指滑动的时候
var xMove = e.touches[1].clientX - e.touches[0].clientX;//手指在x轴移动距离
var yMove = e.touches[1].clientY - e.touches[0].clientY;//手指在y轴移动距离
var distance = Math.sqrt(xMove * xMove + yMove * yMove);//根据勾股定理算出两手指之间的距离
if (_this.data.olddistance == 0) {
_this.data.olddistance = distance; //要是第一次就给他弄上值,什么都不操作
// console.log("第一次");
}else {
_this.data.newdistance = distance; //第二次就可以计算它们的差值了
_this.data.diffdistance = _this.data.newdistance - _this.data.olddistance;//两次差值
_this.data.olddistance = _this.data.newdistance; //计算之后更新比例
_this.data.Scale = _this.data.oldscaleA + 0.005 * _this.data.diffdistance;//这条公式是我查阅资料后找到的,按照这条公式计算出来的比例来处理图片,能给用户比较好的体验
if (_this.data.Scale > 2.5){//放大的最大倍数
return;
} else if (_this.data.Scale < 1) {//缩小不能小于当前
return;
}
//刷新.wxml ,每次相乘,都是乘以图片的显示宽高
_this.setData({
height: _this.data.baseHeight * _this.data.Scale,
width: _this.data.baseWidth * _this.data.Scale
})
_this.data.oldscaleA = _this.data.Scale;//更新比例
}
}
},
//手指离开屏幕
endTou: function (e) {
this.data.olddistance = 0;//重置
this.getRect();
}
- 1
- 2
- 3
- 4
- 5
- 6
- 7
- 8
- 9
- 10
- 11
- 12
- 13
- 14
- 15
- 16
- 17
- 18
- 19
- 20
- 21
- 22
- 23
- 24
- 25
- 26
- 27
- 28
- 29
- 30
- 31
- 32
- 33
- 34
- 35
- 36
- 37
- 38
- 39
- 40
- 41
- 42
- 43
- 44
getRect()是我用来获取节点信息的方法,用于得到wx.canvasToTempFilePath方法的坐标点。(不懂得朋友可以点进去看看)
getRect: function () {
var _this = this;
wx.createSelectorQuery().select(\'.FilePath\').boundingClientRect(function (rect) {
_this.data.x = Math.abs(rect.left);//x坐标
_this.data.y = Math.abs(rect.top);//y坐标
}).exec()
}
- 1
- 2
- 3
- 4
- 5
- 6
- 7
wx.createSelectorQuery()可以获取到节点的信息。
估计看到这里有人就蒙了,其实rect.left和rect.top分别是节点的左边界坐标和节点的上边界坐标,这个坐标可以视为偏移量,就是说’.FilePath’这个节点也就是我放图片的标签往左边和上边偏移了多少,再取他们的绝对值,就得到了我们需要的坐标点了。
接下来就是最后一步了,我们要来截取图片,
wxml:
<canvas canvas-id="myCanvas_A" style="width: {{width}}px;height: {{height}}px;"/>
- 1
js:
generate: function () {
var _this = this;
const ctx_A = wx.createCanvasContext(\'myCanvas_A\');
var baseWidth = _this.data.baseWidth * _this.data.Scale;//图片放大之后的宽
var baseHeight = _this.data.baseHeight * _this.data.Scale;//图片放大之后的高
ctx_A.drawImage(_this.data.src, 0, 0, baseWidth, baseHeight);//我们要在canvas中画一张和放大之后的图片宽高一样的图片
ctx_A.draw();
wx.showToast({
title: \'截取中...\',
icon: \'loading\',
duration: 10000
});//
setTimeout(function(){//给延时是因为canvas画图需要时间
wx.canvasToTempFilePath({//调用方法,开始截取
x: _this.data.x,
y: _this.data.y,
width: 400,
height: 400,
destWidth: 400,
destHeight: 400,
canvasId: \'myCanvas_A\',
success: function (res) {
console.log(res.tempFilePath);
}
})
}, 2000)
},
- 1
- 2
- 3
- 4
- 5
- 6
- 7
- 8
- 9
- 10
- 11
- 12
- 13
- 14
- 15
- 16
- 17
- 18
- 19
- 20
- 21
- 22
- 23
- 24
- 25
- 26
- 27
- 28
终于,世界都安静了,完成!!!!!!!!!!
我该开始接到这个任务的时候,我的内心其实是挺崩溃的,后来静下心来,一个功能一个功能的慢慢来,最终,还是弄出来了,挺开心的。
弄出来之后,发现其实也不难,只要处理好了图片的缩放,其他的都不是啥难事,就是调试的时候特别坑爹,只能在手机上调试,在控制台打了很多输出,慢慢看着来弄的。
总结:
图片的缩放就是监听用户双指滑动的距离,在通过公式得到比例,然后拿这个比例和图片的基本宽高相乘就可以了。截取图片这个就很简单了,就是wx.canvasToTempFilePath的开始坐标这个我是废了一番脑子才想到方法取得的。世上无难事,只怕有心人啊。
再废话一句,这个页面的功能的实现,可页面的样式也是有关的哟。照片的宽高,照片父节点的宽高。。。
from https://blog.csdn.net/yang7789/article/details/78933734