开发Canvas 绘画应用(四):实现拖拽绘画

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

✁ 拖放如何实现?

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

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

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

touchF(e) {
    e.preventDefault(); // 阻止浏览器默认行为
    const touches = e.changedTouches;
    const point = touches[0];
    let el = e.target,
        $el = $(e.target);

    switch (e.type) {
        case 'touchstart':

            // 触摸点起始坐标,不带单位
            this.p_start = {
                x: point.clientX,
                y: point.clientY
            }

            // 图片起始坐标,因为带单位,所以用parseFloat进行转换
            this.el_start = {
                x: parseFloat($el.css('left')),
                y: parseFloat($el.css('top'))
            };

            break;
        case 'touchmove':
            this.dragging = true;

            // 触摸点移动坐标差值,不带单位
            let diffX = point.clientX - this.p_start.x,
                diffY = point.clientY - this.p_start.y;

            if (this.dragging) {

                // 随触摸点坐标更改目标元素的坐标
                $el.css({
                    'left': this.el_start.x + diffX,
                    'top': this.el_start.y + diffY
                });
            }

            break;
        case 'touchend':
            if (!this.dragging) {
                this.setStyle($(e.target)); // 切换视图显示状态
                this.setBasePlate(e.target); // 切换底板显示状态
            }

            this.dragging = false;

            break;
        default:
            this.dragging = false;

            break;
    }
}

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

// 采用jquery的offset方法获取坐标,注意里面的属性不是x和y,而是left和top
this.el_start = this.$el.offset();

// 又或者采用getBoundingClientRect来获取坐标,也要注意left和top属性
this.el_start = this.el.getBoungdingClientRect();

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

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

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

position: absolute;
top: 20px;
left: 50%;
transform: translateX(-50%);

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

console.log('通过css()方式获取:');
console.log({
    left: parseFloat($el.css('left')),
    top: parseFloat($el.css('top'))
});

console.log('通过offset()方式获取:');
console.log($el.offset());

console.log('通过getBoungdingClientRect()方式获取:');
console.log(el.getBoundingClientRect());
坐标坑

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

【方式2】:求解答

margin: auto;
top: 20px;
left: 0;
right: 0;
z-index: 3;

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

坐标坑2

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

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

AnyWay,附上实现效果:

拖动效果实现

✁ 克隆对象

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

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

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

克隆对象

✁ 判断是否拖入画布

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

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

case 'touchend':

    // 获取克隆元素的宽高及坐标
    let clone_rect = (this.$clone)[0].getBoundingClientRect();

    if (!this.dragging) {
        this.setStyle($el); // 切换视图显示状态
        this.setBasePlate(el); // 切换底板显示状态

        // 如果进入画布
    } else if (this.intoPainter(clone_rect)) {
        this.setBasePlate(); // 清空底板
        this.drawResult(el); // 在painter上进行绘画

        // 否则回到初始状态
    } else {

    }

    this.dragging = false;
    this.$clone.remove(); // 移除clone对象

    break;

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

/**
 * 判断是否进入了 painter 画布
 * @param  {[object]} srcRect 进入画布对象的大小及在视口中的坐标信息
 * @return {[boolean]}   
 */
intoPainter(srcRect) {
    const rect = this.painter.getBoundingClientRect();

    // 上下左右边界判断
    let cL = srcRect.left > rect.left,
        cT = srcRect.top > rect.top,
        cR = srcRect.right < rect.right,
        cB = srcRect.bottom < rect.bottom;

    return cL && cT && cR && cB;
}

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

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

✈ Github:paintApp

✎ 参考:

javascript小实例,移动端页面中的拖拽
移动端拖拽的实现效果
HTML5 移动端div块跟随手指拖动

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 215,634评论 6 497
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,951评论 3 391
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 161,427评论 0 351
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,770评论 1 290
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,835评论 6 388
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,799评论 1 294
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,768评论 3 416
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,544评论 0 271
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,979评论 1 308
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,271评论 2 331
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,427评论 1 345
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,121评论 5 340
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,756评论 3 324
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,375评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,579评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,410评论 2 368
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,315评论 2 352

推荐阅读更多精彩内容

  • 1、通过CocoaPods安装项目名称项目信息 AFNetworking网络请求组件 FMDB本地数据库组件 SD...
    阳明先生_X自主阅读 15,979评论 3 119
  • 问答题47 /72 常见浏览器兼容性问题与解决方案? 参考答案 (1)浏览器兼容问题一:不同浏览器的标签默认的外补...
    _Yfling阅读 13,747评论 1 92
  • 我向青春献支歌 生活随想 十四行诗 也是一种浪漫的传说 迷失的乡土之思 无字碑之断想 那不管天高地厚的执着 也是一...
    阿桂爱原创阅读 283评论 14 24
  • 今夜,我有一杯茶 茶香四溢,让我不忍入口 于是,我将它放在庭院的圆桌上 轻轻地坐在摇椅上,静静地享受阳光 深吸一口...
    甜格格阅读 434评论 0 0