开发Canvas绘画应用(三):实现对照绘画中,我们实现了视图引导的第一部分,这一篇我们来完成第二部分,即将图片直接拖到画布上进行绘画。

✁ 拖放如何实现?

【拖放的基本概念】:创建一个绝对定位的元素,使其可以用鼠标或手指移动。

注意,为了使元素能被拖放,它必须是绝对定位的。

然后,我们需要填充我们的 touchF 函数来实现拖动功能,添加了 this.dragging 用于判断是否是拖动状态,只有当 touchmove 触发的时候才为 true。另外,当拖动的时候,需要改变目标对象的位置,通过 clientXclientY 来进行更改。

  1. touchF(e) {
  2. e.preventDefault(); // 阻止浏览器默认行为
  3. const touches = e.changedTouches;
  4. const point = touches[0];
  5. let el = e.target,
  6. $el = $(e.target);
  7. switch (e.type) {
  8. case 'touchstart':
  9. // 触摸点起始坐标,不带单位
  10. this.p_start = {
  11. x: point.clientX,
  12. y: point.clientY
  13. }
  14. // 图片起始坐标,因为带单位,所以用parseFloat进行转换
  15. this.el_start = {
  16. x: parseFloat($el.css('left')),
  17. y: parseFloat($el.css('top'))
  18. };
  19. break;
  20. case 'touchmove':
  21. this.dragging = true;
  22. // 触摸点移动坐标差值,不带单位
  23. let diffX = point.clientX - this.p_start.x,
  24. diffY = point.clientY - this.p_start.y;
  25. if (this.dragging) {
  26. // 随触摸点坐标更改目标元素的坐标
  27. $el.css({
  28. 'left': this.el_start.x + diffX,
  29. 'top': this.el_start.y + diffY
  30. });
  31. }
  32. break;
  33. case 'touchend':
  34. if (!this.dragging) {
  35. this.setStyle($(e.target)); // 切换视图显示状态
  36. this.setBasePlate(e.target); // 切换底板显示状态
  37. }
  38. this.dragging = false;
  39. break;
  40. default:
  41. this.dragging = false;
  42. break;
  43. }
  44. }

▶▶▶ 在获取视图对象的坐标位置时,除了上述用到的 css() 方式,还有下面两种:

  1. // 采用jquery的offset方法获取坐标,注意里面的属性不是x和y,而是left和top
  2. this.el_start = this.$el.offset();
  3. // 又或者采用getBoundingClientRect来获取坐标,也要注意left和top属性
  4. this.el_start = this.el.getBoungdingClientRect();

但是这里有一个大坑!!!也是我们不采用后两种方式,而采用 css() 方式的原因,这跟元素在css中如何定位有关,下面来看看这个坑。

我们当前视口的大小是 980×874,有两种css方式可以将元素居中定位,但是获取位置时会有区别:

【方式1】:正确的打开方式

  1. position: absolute;
  2. top: 20px;
  3. left: 50%;
  4. transform: translateX(-50%);

通过不同的方式获取坐标:

  1. console.log('通过css()方式获取:');
  2. console.log({
  3. left: parseFloat($el.css('left')),
  4. top: parseFloat($el.css('top'))
  5. });
  6. console.log('通过offset()方式获取:');
  7. console.log($el.offset());
  8. console.log('通过getBoungdingClientRect()方式获取:');
  9. console.log(el.getBoundingClientRect());

而在实现中,通过第一种css()方法获取坐标时进行对象移动是正常的,后两种都会在移动时将图片往左偏移100像素,为什么呢?因为在css中获取的就是元素的left值,即通过 left:50% 后偏离的 490px,是不包含transform 变换的,因此才会多出来100像素。

【方式2】:求解答

  1. margin: auto;
  2. top: 20px;
  3. left: 0;
  4. right: 0;
  5. z-index: 3;

同样通过不同的方式获取坐标得到:

这里有个坑,就是左右移动的时候,当距离增大时,触摸点跟图片的距离会越来越增大,即图片移动的速度更不上触摸点移动的速度,上下移动时是好的,求大神解答。。。。

▲▲▲ 因此,为什么 css() 方式可以实现我们的正常不偏移位置的拖动,因为我们用的是绝对定位!!而在更改元素坐标时,采用的是加上坐标差的方式,即在原先 left 值的基础上加上偏移量。好吧,得承认这一块坑死我了T^T。

AnyWay,附上实现效果:

✁ 克隆对象

当前,我们移动的是视图本身,但是我们想要原本的视图不动,移动它的一个复制图片,这就需要克隆一个当前对象,在 touchstart 中进行实现:

  1. // 克隆一个对象
  2. this.$clone = $el.clone();
  3. this.$clone.insertBefore($el.siblings()[0]).css({
  4. 'z-index': 4,
  5. 'border': 'none'
  6. });

然后我们对这个克隆对象进行位置操作便可。

✁ 判断是否拖入画布

  • 接着,我们需要判断是否将图片拖入画布,这需要对画布的坐标进行判断;
  • 然后,当拖入的时候,在画布上调用 drawImage() 方法进行绘画,否则,这个克隆对象将回到原始位置,和目标图片重合;
  • 最后,无论是否进入,当 touchend 触发时,都需要销毁这个克隆对象。

① 整理逻辑,在 touchend 时进行判断:

  1. case 'touchend':
  2. // 获取克隆元素的宽高及坐标
  3. let clone_rect = (this.$clone)[0].getBoundingClientRect();
  4. if (!this.dragging) {
  5. this.setStyle($el); // 切换视图显示状态
  6. this.setBasePlate(el); // 切换底板显示状态
  7. // 如果进入画布
  8. } else if (this.intoPainter(clone_rect)) {
  9. this.setBasePlate(); // 清空底板
  10. this.drawResult(el); // 在painter上进行绘画
  11. // 否则回到初始状态
  12. } else {
  13. }
  14. this.dragging = false;
  15. this.$clone.remove(); // 移除clone对象
  16. break;

② 完善 intoPainter 函数,判断是否进入画布

  1. /**
  2. * 判断是否进入了 painter 画布
  3. * @param {[object]} srcRect 进入画布对象的大小及在视口中的坐标信息
  4. * @return {[boolean]}
  5. */
  6. intoPainter(srcRect) {
  7. const rect = this.painter.getBoundingClientRect();
  8. // 上下左右边界判断
  9. let cL = srcRect.left > rect.left,
  10. cT = srcRect.top > rect.top,
  11. cR = srcRect.right < rect.right,
  12. cB = srcRect.bottom < rect.bottom;
  13. return cL && cT && cR && cB;
  14. }

③ 在 pinter.js 中完善 drawResult() 函数:

  1. /**
  2. * 绘制拖入的图片
  3. * @param {[type]} image 视图对象,原生js <img>
  4. * @return {[type]} [description]
  5. */
  6. drawResult(image) {
  7. this.clearBg(); // 清除画布
  8. this.ctx.drawImage(image, 0, 0, this.config.cvaW, this.config.cvaH);
  9. }

✈ Github:paintApp

✎ 参考:

javascript小实例,移动端页面中的拖拽

移动端拖拽的实现效果

HTML5 移动端div块跟随手指拖动

开发Canvas 绘画应用(四):实现拖拽绘画的更多相关文章

  1. iOS开发拓展篇—xib中关于拖拽手势的潜在错误

    iOS开发拓展篇—xib中关于拖拽手势的潜在错误 一.错误说明 自定义一个用来封装工具条的类 搭建xib,并添加一个拖拽的手势. 主控制器的代码:加载工具条 封装工具条以及手势拖拽的监听事件 此时运行 ...

  2. 快速开发 HTML5 WebGL 的 3D 斜面拖拽生成模型

    前言 3D 场景中的面不只有水平面这一个,空间是由无数个面组成的,所以我们有可能会在任意一个面上放置物体,而空间中的面如何确定呢?我们知道,空间中的面可以由一个点和一条法线组成.这个 Demo 左侧为 ...

  3. 移动端H5混合开发,Touch触控,拖拽,长按, 滑屏 实现方案

    概述 近期由于产品快速原型开发的需要,不想用原声的方式开发App两端一起搞时间来不及,目前产品处于大量上feature的阶段,采用混合开发是最合适的选择,所以花了3天的时间研究怎么去实现移动端,拖拽, ...

  4. YAPI windows 二次开发 树形结构 多层级结构 拖拽 数据导入 接口自动化测试

    什么是YAPI: 高效.易用.功能强大的API管理平台 http://yapi.demo.qunar.com/ github: https://github.com/YMFE/yapi 可以去那里下载 ...

  5. javascript动画系列第四篇——拖拽改变元素大小

    × 目录 [1]原理简介 [2]范围圈定 [3]大小改变[4]代码优化 前面的话 拖拽可以让元素移动,也可以改变元素大小.本文将详细介绍拖拽改变元素大小的效果实现 原理简介 拖拽让元素移动,是改变定位 ...

  6. 从零开始学 Web 之 HTML5(四)拖拽接口,Web存储,自定义播放器

    大家好,这里是「 从零开始学 Web 系列教程 」,并在下列地址同步更新...... github:https://github.com/Daotin/Web 微信公众号:Web前端之巅 博客园:ht ...

  7. iOS开发小技巧--cell往左拖拽出现很多按钮的实现,仅仅适用于iOS8以后

    // 往左拖拽cell出现多个按钮的实现,仅仅适用于iOS_8.0以后 - (NSArray<UITableViewRowAction *> *)tableView:(UITableVie ...

  8. jsPlumb开发入门教程(实现html5拖拽连线)

    jsPlumb是一个强大的JavaScript连线库,它可以将html中的元素用箭头.曲线.直线等连接起来,适用于开发Web上的图表.建模工具等.它同时支持jQuery+jQuery UI.MooTo ...

  9. 开发Canvas 绘画应用(三):实现对照绘画

    需求分析 在我的毕设中,提出了视图引导的概念,由两部分功能组成: (1)可以对照着图片进行绘画,即将图片以半透明的方式呈现在绘图板上,然后用户可以对照着进行绘画: (2)可以直接将简笔画图片直接拖拽到 ...

随机推荐

  1. 学号 2018-2019-20175212 童皓桢《Java程序设计》第5周学习总结

    学号 2018-2019-20175212 <Java程序设计>第5周学习总结 教材学习内容总结 接口 声明接口: interface 名字: 接口体:接口体中只有常量无变量,只有抽象方法 ...

  2. 【Python】Part I 设置Python环境

    01 设置Python环境 02 破解WingIDE (1)下载专业版wingide http://wingware.com/downloads/wing-pro/6.0.11-1/binaries& ...

  3. ios打包unity应用以及配置签名

    先决条件是必须为苹果mac机.拥有公司苹果账号,并确保电脑上安装了unity,unity包 ios-support.和xcode. 1.打开了unity应用之后,选择buildSettings 然后点 ...

  4. springboot 多端口启动

    以eclipse(STS)为例, 选中项目右键Run Configurations 点击Spring Boot App,选中需要设定多端口项目,在启动参数一栏输入:-Dserver.port=7003 ...

  5. tinyproxy代理配置

    tinyproxy代理配置 应用场景: 生产机处于内网,无法直接访问外网,程序安装和漏洞修复等操作需要进行联网操作:通过在办公网(可访问外网)上设置代理服务器,生产机通过代理由办公网访问外网 代理服务 ...

  6. Hello vue.js的随笔记录

    数据双向绑定的script在组件定义位置后面才顶用. 使用它的话,引用js就好,比较简单. 声明一个vm对象,new Vue({}).这个构造里传一个对象,包含el:界面元素,data:数据,meth ...

  7. install openjdk & tomcat on the centos

    1.检查当前服务器是否已安装openjdk

  8. 学习python及Pygame的安装及运行

    Python: 注意勾上Add Python 2.7 to PATH,然后点“Install Now”即可完成安装. 或手动修改环境变量,win7:右击我的电脑->属性->高级->环 ...

  9. java代码生成json数据

    https://www.cnblogs.com/libo0125ok/p/7905665.html

  10. PAT B1023

    PAT B1023 标签(空格分隔): PAT 解决方法:贪心法 #include <cstdio> int main() { int count[10]; for (int i = 0; ...