开发Canvas 绘画应用(三):实现对照绘画
需求分析
在我的毕设中,提出了视图引导的概念,由两部分功能组成:
(1)可以对照着图片进行绘画,即将图片以半透明的方式呈现在绘图板上,然后用户可以对照着进行绘画;
(2)可以直接将简笔画图片直接拖拽到画布上进行检索。
那么,在这里,我们先实现第一种功能——图片对照绘画。最终想要的实现效果是:当点击图片时,图片边框会呈现蓝色,表示选中状态,同时会在画布上以半透明方式呈现该图片,再点击图片,边框和底板图片都消失,即可以切换显示状态。
a. 点击前:

b. 点击后:

c. 取消点击:

实现思路
我们来分析一下,这种半透明底板方式是如何实现的?
1)首先,需要利用分层来实现图片参照效果:即设置两个 Canvas,上层的 Canvas 仍旧用来绘画,而下层的 Canvas 则作为底板显示图片,这里需要用到 canvas 的绘制图像 API——drawImage()。
context.drawImage(iamge, srcX, srcY, srcWidth, srcHeight, desX, desY, desWidth, desHeight)
2)其次,为了出现半透明的效果,需要设置上层 canvas 的透明度,这里需要用到 canvas 的图片合成,即通过设置 globalAlpha 属性指定所有绘制的透明度。
两个应用到2D上下文中所有绘制操作的属性:
globalAlpha:属于[0, 1],用于指定所有绘制的透明度。globalCompositionOperation:表示后绘制的图形怎样与先绘制的图形结合,其属性值是字符串。
代码实现
该阶段我们需要添加2个模块:一个是视图模块(viewer.js),用于放置我们的参照图片(蜜蜂线画图图片);另一个是底板模块(basePlate.js),用于显示底板。
另外,此处增加的两个 Canvas 其实前期的画板设置是相同的,因此我们将设置单独拎到一个 cvaConfig.js 文件中,以便代码重用,然后 painter.js 和 basePlate.js 只要继承 cvaConfig.js 就行了。
☞ index.html
<img src="asset/img/bee.png" class="view">
<canvas class="painter" id="js-painter"></canvas>
<canvas class="baseplate" id="js-baseplate"></canvas>
☞ cvaConfig.js
export default class CvaConfig {
constructor() {
this.config = {
cvaW: 600,
cvaH: 500,
cvaBg: '#fff',
lineWidth: 2,
lineJoin: 'round',
strokeStyle: 'red'
};
}
// 画板宽高设置
// 注意此处不能在 css 中设置,像素会失真,会导致不能获取正确的坐标
setCvaWH() {
this.cva.setAttribute('width', this.config.cvaW);
this.cva.setAttribute('height', this.config.cvaH);
return this;
}
// 画板背景设置
setCvaBg() {
this.ctx.fillStyle = this.config.cvaBg;
this.ctx.fillRect(0, 0, this.config.cvaW, this.config.cvaH);
return this;
}
// 画笔设置
setPen() {
this.ctx.lineWidth = this.config.lineWidth;
this.ctx.lineJoin = this.config.lineJoin;
this.ctx.strokeStyle = this.config.strokeStyle;
return this;
}
}
☞ viewer.js
export default class Viewer {
constructor(options) {
this.options = options;
this.init();
}
initSelectors() {
this.view = $('.view');
return this;
}
initEvents() {
this.view.on('touchstart', $.proxy(this.touchF, this));
this.view.on('touchmove', $.proxy(this.touchF, this));
this.view.on('touchend', $.proxy(this.touchF, this));
}
touchF(e) {
e.preventDefault(); // 阻止浏览器默认行为
switch (e.type) {
case 'touchstart':
break;
case 'touchmove':
break;
case 'touchend':
this.setStyle($(e.target)); // 切换视图显示状态
this.setBasePlate(e.target); // 切换底板显示状态
break;
}
}
/**
* 切换视图显示状态,表示是否点击
* @param {[type]} $el 点击的jQuery目标对象
*/
setStyle($el) {
$el.toggleClass('blue-border');
}
/**
* 通过点击图视图调用basePlate的接口切换底板显示状态
* @param {[type]} el 点击的原生js目标对象
*/
setBasePlate(el) {
// 视图选中时传入图片
if ($(el).hasClass('blue-border')) {
this.options.setBasePlate(el);
} else { // 未选中不传图片
this.options.setBasePlate();
}
}
init() {
this.initSelectors().initEvents();
}
}
☞ basePlate.js
basePlate 中功能性的只有一个用来设置底板的函数,通过在点击 viewer 时进行调用:
import CvaConfig from './cvaConfig.js';
export default class basePlate extends CvaConfig {
constructor() {
super();
this.init();
}
initSelectors() {
this.cva = document.getElementById('js-baseplate');
this.ctx = this.cva.getContext('2d');
return this;
}
// 设置底板
setBasePlate(image) {
if (image) { // 如果有图片,在底板上显示图片
this.ctx.drawImage(image, 0, 0, this.config.cvaW, this.config.cvaH);
} else { // 设置空底板
this.setCvaBg();
}
}
init() {
this.initSelectors();
this.setCvaWH().setCvaBg();
}
}
☞ painter.js
painter 中设置其透明度:
// 设置画布背景及透明度
_setBg() {
this.ctx.globalAlpha = 0.7;
this.setCvaBg();
return this;
}
OK,参照绘画功能就实现啦,接下去要实现视图引导的第二个功能:通过直接拖拽图片进行检索。当然,我们这里不涉及跟服务端进行交涉,因此只实现前端的拖拽图片功能,这部分会用到一些坐标判断,正好趁此机会好好整理一下。利用拖拽可以实现很多效果,还是挺好玩哒~~加油加油~q(≧▽≦q)
敬请关注下一篇:开发Canvas 绘画应用(四):实现拖拽绘画
✈ Github:paintApp
✎ 参考:
开发Canvas 绘画应用(三):实现对照绘画的更多相关文章
- 开发Canvas 绘画应用(二):实现绘画
开发Canvas 绘画应用(一):搭好框架 中我们已经把基本框架及基础功能实现了,现在要实现本应用的重点:绘画功能. 首先分析一下,我们要实现绘画,需要具备的理论知识如下: (1)获取触摸点的坐标 类 ...
- 开发Canvas 绘画应用(四):实现拖拽绘画
在开发Canvas绘画应用(三):实现对照绘画中,我们实现了视图引导的第一部分,这一篇我们来完成第二部分,即将图片直接拖到画布上进行绘画. ✁ 拖放如何实现? [拖放的基本概念]:创建一个绝对定位的元 ...
- 开发Canvas 绘画应用(一):搭好框架
毕业汪今年要毕业啦,毕设做的是三维模型草图检索,年前将算法移植到移动端做了一个小应用(利用nodejs搭的服务),正好也趁此机会可以将前端的 Canvas 好好学一下~~毕设差不多做完了,现将思路和代 ...
- 一步一步开发Game服务器(三)加载脚本和服务器热更新(二)完整版
上一篇文章我介绍了如果动态加载dll文件来更新程序 一步一步开发Game服务器(三)加载脚本和服务器热更新 可是在使用过程中,也许有很多会发现,动态加载dll其实不方便,应为需要预先编译代码为dll文 ...
- [课程设计]Scrum 3.1 多鱼点餐系统开发进度(第三阶段项目构思与任务规划)
Scrum 3.1 多鱼点餐系统开发进度(第三阶段项目构思与任务规划) 1.团队名称:重案组 2.团队目标:长期经营,积累客户充分准备,伺机而行 3.团队口号:矢志不渝,追求完美 4.团队选题:餐厅到 ...
- iOS开发Swift篇—(三)字符串和数据类型
iOS开发Swift篇—(三)字符串和数据类型 一.字符串 字符串是String类型的数据,用双引号""包住文字内容 let website = "http://www ...
- Scrum 3.1 多鱼点餐系统开发进度(第三阶段项目构思与任务规划)
Scrum 3.1 多鱼点餐系统开发进度(第三阶段项目构思与任务规划) 1.团队名称:重案组 2.团队目标:长期经营,积累客户充分准备,伺机而行 3.团队口号:矢志不渝,追求完美 4.团队选题:餐厅到 ...
- ASP.NET自定义控件组件开发 第一章 第三篇
原文:ASP.NET自定义控件组件开发 第一章 第三篇 第三篇:第一章的完结篇 系列文章链接: ASP.NET自定义控件组件开发 第一章 待续 ASP.NET自定义控件组件开发 第一章 第二篇 接着待 ...
- iOS 开发问题集锦(三)
iOS 开发问题集锦(三) 介于群里大部分童鞋都是新手,为了大家能够更好的提问,并且提的问题能更好的得到回答,下面写几点提问时的注意事项: 1.认真对待你的问题,在提问题前有过认真的思考: 2.先在 ...
随机推荐
- centes7安装wdcp
CentOS7安装WDCP3 CentOS7安装WDCP3.2面板教程 到此WDCP安装完毕
- 初遇sass的两个小问题
关于sass大家都知道是一种css的开发工具,原本的css没有变量 参数一类的东西,所以比较死 效率较慢. sass就是在css里面加入了一些编程的元素如变量等,让css能够更灵活,提高效率. 刚接触 ...
- golang开源项目qor快速搭建网站qor-example运行实践
最近想找几个基于Go语言开发的简单的开源项目学习下,分享给大家,github上有心人的收集的awesome-go项目集锦:github地址 发现一个Qor项目: Qor 是基于 Golang 开发的的 ...
- 我只想要Linux的IP地址
大家都知道ifconfig 可以查看centos的ip地址,但是我如果只要ip地址该怎么办呢?首先上ifconfig [root@centos ~]# ifconfig eth0 eth0 Link ...
- hdu 1754 I Hate It 解题报告(线段树 代码+注释)
题目链接:传送门 I Hate It Time Limit: 9000/3000 MS (Java/Others) Memory Limit: 32768/32768 K (Java/Other ...
- python中下划线的特殊用法
python下划线用法总结: ① _XXX 不能用于“ from model import * ”的导入: ②__XXX__ 系统定义名字: ③__XXX 类中的私有变量名. 总结:避免随意用下划线 ...
- Python的布尔值与空值
1.Boolean值(布尔值) 一个布尔值只有Ture.False两种值 b1 =True b2 =False print (b1,b2)>>>True False 2.空值(non ...
- 后台管理系统之“图片上传” --vue
图片上传(基于vue) 相信上传图片是所有系统必备的功能吧,工作中的第一个管理系统就在上传图片的功能上卡顿了一整天. 当时用的elementUI组件,但是由于样式和设计图样式差别较大再加上原生相较好理 ...
- 对弈的C++学习笔记
2018-07-11上传 一:从C到C++ 1.C++新类型 bool 判断真假 占用一个字节 if(条件) 真1或者假 0 bool 类型的取值 true false ...
- 零基础学习JavaSE(二)——基础语法
二.Java 基础语法 2.1 Java 基础语法 java是一个面向对象的程序语言,及可把一切事物当做对象处理,而java的事物中最小的就是class (类),类中有方法,类可以创建对象,并且有一些 ...