1

Mobile Safari にドラッグ アンド ドロップを実装しようとしています。

基本的に、ループする CSS アニメーションを再生しながら、タッチで div をドラッグしたいと考えています。また、いつドロップ ターゲットを超えたかを判断し、何か特別なことをしたいと考えています。

elementFromPointを使用して、div がtouchmoveイベントでドラッグされている要素を特定しています。ただし、ドラッグされた div は常に最上位の要素です。したがって、クエリを実行する前に、display: noneに設定しました。ただし、これには、タッチが移動するフレームごとにアニメーションをリセットする効果があります。

クエリを実行するたびに CSS アニメーションをリセットせずに、上にドラッグしているものを特定するにはどうすればよいですか?

WebKit でタッチ イベントを使用するJSFiddle 。

HTMLすべての touchmove で、ドラッグdiv を移動します。

<div id='drop'>Drop here</div>
<div id='drag'>Drag me</div>

JavaScript

var drag = document.getElementById('drag');
var drop = document.getElementById('drop');

drag.addEventListener('touchmove', move, true);

function move(e) {
    var touch = e.changedTouches[0];
    drag.style.left = touch.pageX - 25 + 'px';
    drag.style.top = touch.pageY - 25 + 'px';

    drag.style.display = 'none';
    if (drop === document.elementFromPoint(touch.pageX, touch.pageY)) {
        drag.classList.add('in-drop-zone');
    } else {
        drag.classList.remove('in-drop-zone');    
    }
    drag.style.display = 'inline';
}

CSS

div {
    position: absolute;
}

div#drag {
    width: 50px;
    height: 50px;
    background-color: blue;
    -webkit-transform-origin: 50% 50%;
    -webkit-animation: sway 1s infinite alternate;
}

div#drop {
    width: 100px;
    height: 100px;
    background-color: green;
    left: 200px;
    top: 200px;
}

div#drag.in-drop-zone {
    background-color: yellow;
}

@-webkit-keyframes sway {
    from {
        -webkit-transform: rotate(-30deg);
    }
    to {
        -webkit-transform: rotate(30deg);
    }
}
4

1 に答える 1

0

おそらく設定pointer-events: noneしてから を設定できますがpointer-events: auto、それはかなりハックなようです。より良いオプションは、交差関数を持つことです:

var drag = document.getElementById('drag');
var drop = document.getElementById('drop');
var drop_bbox = drop.getBoundingClientRect();

window.addEventListener('touchmove', move, true);
window.addEventListener('mousemove', move, true);

function move(e) {
    drag.style.left = e.clientX - 25 + 'px';
    drag.style.top = e.clientY - 25 + 'px';
    var drag_bbox = e.target.getBoundingClientRect();
    drag.className = rectsIntersect(drag_bbox, drop_bbox) ? "in-drop-zone" : null;
}

 function rectsIntersect(r1, r2) {
  if (!r1 || !r2) return false;
    return r2.left < (r1.left+r1.width) && 
        (r2.left+r2.width) > r1.left &&
        r2.top < (r1.top+r1.height) &&
        (r2.top+r2.height) > r1.top;
};

ここにJSFiddleがありますhttp://jsfiddle.net/duopixel/SvPpw/

于 2013-05-20T08:24:52.253 に答える