需求
- 在canvas插入图片当做背景图
- 在图片上进行涂鸦、图画
- 可以更换画笔粗细、颜色
- 可以实现撤销上一步操作
- 最后可以同图片一起保存至本地相册或者上传至服务器
功能注意点
- 使用
uni.getSystemInfo({})
获取设备的宽高作为canvas区域及绘图区域 - 绘图时,不可以直接使用网络图片,使用
wx.getImageInfo({})
获取图片信息做为绘图路径
撤销操作逻辑
- 绘图步骤start->move->end
- 在每次start的时候调用保存图片的方法,将当前画布的图片保存在本地的数组中
- 点击撤销的时候,删除保存数组中的最后一个图片地址,并重画这个地址的图片(drawImage)
以下为全部代码
HTML
<template>
<view class="container">
<!--画布区域-->
<cover-view class="topBox">
<button class="backStep" type="default" @click="backStep" size="mini">撤销</button>
<button class="saveBtn" type="warn" @click="saveimg" size="mini">保存</button>
</cover-view>
<view class="canvas_area">
<canvas
canvas-id="myCanvas"
class="myCanvas"
disable-scroll="false"
@touchstart="touchStart"
@touchmove="touchMove"
@touchend="touchEnd"
:style="{ width: w + 'px', height: h + 'px' }"
></canvas>
</view>
<!--画布工具区域-->
<view class="canvas_tools">
<view class="box box1" @click="penSelect(3)"></view>
<view class="box box2" @click="penSelect(15)"></view>
<view class="box box3" @click="colorSelect('#cc0033')"></view>
<view class="box box4" @click="colorSelect('#ff9900')"></view>
<view class="box box5" @click="colorSelect('#00CD00')"></view>
</view>
</view>
</template>
JS
<script>
export default {
data() {
return {
imgurl: '',
pen: 3, //画笔粗细默认值
color: '#cc0033', //画笔颜色默认值
startX: 0, //保存X坐标轴变量
startY: 0, //保存X坐标轴变量
w: '', //canvas宽高区域
h: '',
ctx: wx.createContext('myCanvas', this),
allDrawWorksPath: [] //图片路径 用于撤销
};
},
onLoad(option) {
//真实环境为后端返回图片路径
this.imgurl = 'https://ss3.baidu.com/9fo3dSag_xI4khGko9WTAnF6hhy/zhidao/pic/item/72f082025aafa40f2982756baa64034f78f0193b.jpg';
this.getSystemInfo(); //获取设备信息
},
onReady(option) {
var that = this;
console.log('imgurl', that.imgurl);
wx.getImageInfo({
src: that.imgurl,
success: function(ress) {
console.log('图片信息', ress);
that.ctx.drawImage(ress.path, 0, 0, that.w, that.h);
that.ctx.stroke();
wx.drawCanvas({
canvasId: 'myCanvas',
reserve: true,
actions: that.ctx.getActions() // 获取绘图动作数组
});
},
fail(err) {
console.log('err', err);
}
});
},
methods: {
// 保存图片
saveimg() {
var _this = this;
this.ctx.draw();
setTimeout(function() {
_this.drawAfter();
}, 500);
},
// 保存
drawAfter() {
let that = this;
wx.canvasToTempFilePath(
{
width: that.w, //686
height: that.h,
canvasId: 'myCanvas',
success: function success(res) {
console.log('保存图片', res);
var tempFilePath = res.tempFilePath;
console.log(tempFilePath);
//把图片保存到相册
wx.saveImageToPhotosAlbum({
filePath: tempFilePath
});
//把图片保存到相册
//进行文件的拷贝
//上传
}
},
this
);
},
// 开始
touchStart(e) {
console.log('开始绘制');
//得到触摸点的坐标
this.startX = e.changedTouches[0].x;
this.startY = e.changedTouches[0].y;
console.log('画笔颜色', this.color, this.pen);
this.ctx.setStrokeStyle(this.color); //画笔颜色
this.ctx.setLineWidth(this.pen); //线条宽度
this.ctx.setLineCap('round'); // 让线条圆润
this.ctx.beginPath();
this.saveCurrentDrawWorks(); //记录每一步步骤-撤销
},
//手指触摸后移动
touchMove(e) {
console.log('开始移动中');
var startX1 = e.changedTouches[0].x;
var startY1 = e.changedTouches[0].y;
this.ctx.moveTo(this.startX, this.startY);
this.ctx.lineTo(startX1, startY1);
this.ctx.stroke();
this.startX = startX1;
this.startY = startY1;
//只是一个记录方法调用的容器,用于生成记录绘制行为的actions数组。
// context跟<canvas/>不存在对应关系,一个context生成画布的绘制动作数组可以应用于多个<canvas/>
wx.drawCanvas({
canvasId: 'myCanvas',
reserve: true,
actions: this.ctx.getActions() // 获取绘图动作数组
});
},
//手指触摸动作结束
touchEnd() {
console.log('停止');
},
//更改画笔大小的方法
penSelect(e) {
console.log(e);
this.pen = e;
},
//更改画笔颜色的方法
colorSelect(e) {
console.log(e);
this.color = e;
},
// 保存每一步操作
saveCurrentDrawWorks() {
let that = this;
wx.canvasToTempFilePath({
x: 0,
y: 0,
width: 0,
height: 0,
canvasId: 'myCanvas',
success: function(res) {
var imgPath = res.tempFilePath;
that.allDrawWorksPath = [...that.allDrawWorksPath, imgPath];
console.log('步骤', that.allDrawWorksPath);
},
fail: res => {
console.log('获取画布图片失败', res);
}
});
},
// 撤销一步
backStep() {
var allDrawWorksPath = this.allDrawWorksPath;
if (allDrawWorksPath == null || allDrawWorksPath.length == 0 || allDrawWorksPath == undefined) {
uni.showToast({
icon: 'none',
title: '已经撤销到起始位置'
});
this.allDrawWorksPath = [];
return;
}
var privWorksPath = allDrawWorksPath.pop();
this.allDrawWorksPath = allDrawWorksPath;
this.ctx.drawImage(privWorksPath, 0, 0, this.w, this.h);
this.ctx.stroke();
let that = this;
wx.drawCanvas({
canvasId: 'myCanvas',
reserve: true,
actions: that.ctx.getActions() // 获取绘图动作数组
});
},
// 获取设备信息
getSystemInfo() {
let that = this;
uni.getSystemInfo({
success(res) {
console.log('设备信息', res);
that.h = res.windowHeight; //若加底部操作区域 需-60
that.w = res.windowWidth;
}
});
}
}
};
</script>
CSS
<style lang="scss">
.container {
position: relative;
width: 100%;
height: 100%;
.topBox {
position: fixed;
z-index: 99999;
width: 100%;
.backStep {
width: 50px;
top: 2px;
left: 2px;
}
.saveBtn {
width: 50px;
position: absolute;
top: 2px;
right: 2px;
}
}
.canvas_tools {
position: fixed;
left: 0;
bottom: 0;
width: 100%;
height: 60px;
display: flex;
flex-direction: row;
justify-content: space-around;
background-color: #eee;
align-items: center;
}
.box {
width: 100rpx;
height: 100rpx;
line-height: 100rpx;
border-radius: 50%;
background-color: rebeccapurple;
}
.box1 {
background-color: #99cccc;
background-image: url();
background-repeat: no-repeat;
background-position: center;
}
.box2 {
background-color: #0099cc;
background-image: url();
background-repeat: no-repeat;
background-position: center;
}
.box3 {
background-color: #cc0033;
}
.box4 {
background-color: #ff9900;
}
.box5 {
background-color: #00cd00;
}
.box6 {
text-align: center;
color: #fff;
}
}
</style>