Drag-and-drop 是一种易学流行的交互手势:将指针指向目标对象,按下并且拖动它到一个新的位置,然后释放。D3
的 drag behavior 提供了方便灵活并且抽象的拖拽交互。例如可以使用 D3
的拖拽与
force-directed graph 进行交互:
你也可以使用 d3-drag
来实现自定义的用户交互,比如滑块。但是拖拽交互不仅仅是用来改变元素的位置的。还有许多手势可以通过拖拽实现,比如套索一些算下,或者在 canvas
上画线:
拖拽交互可以与其他的交互结合使用,比如 `d3-zoom:
拖拽行为与 DOM
无关,因此你可以在 SVG
、HTML
甚至 Canvas
中使用它。并且你可以通过高级选择技术来扩展它比如 Voronoi
或者邻近搜索:
最重要的是,拖拽交互自动统一鼠标和触摸输入,并且屏蔽了不同浏览器的不同特性。当 Pointer Events 可用时,拖拽事件也同样可用。
NPM: npm install d3-drag
. 还可以下载 latest release,可以作为单独的 standalone library,也可以作为 D3 v4 的一部分直接引入. 支持 AMD
, CommonJS
以及 vanilla
环境。如果使用 vanilla
则会暴露全局 d3
变量:
<script src="https://d3js.org/d3-dispatch.v1.min.js"></script>
<script src="https://d3js.org/d3-selection.v1.min.js"></script>
<script src="https://d3js.org/d3-drag.v1.min.js"></script>
<script>
var drag = d3.drag();
</script>
下表描述了拖拽行为如何解析原生事件:
Event(事件) | Listening Element(监听元素) | Drag Event(拖拽事件) | Default Prevented?(阻止默认行为) |
---|---|---|---|
mousedown⁵ | selection | start | no¹ |
mousemove² | window¹ | drag | yes |
mouseup² | window¹ | end | yes |
dragstart² | window | - | yes |
selectstart² | window | - | yes |
click³ | window | - | yes |
touchstart | selection | start | no⁴ |
touchmove | selection | drag | yes |
touchend | selection | end | no⁴ |
touchcancel | selection | end | no⁴ |
所有消耗型事件的传播都是被 immediately stopped(立即停止) 的,如果你想阻止某些事件触发拖拽手势,请使用 drag.filter.
¹ 捕获 iframe
之外的事件是必须的; 参考 #9.
² 仅仅适用于基于鼠标的活动的手势; 参考 #9.
³ 仅适用于一些特定鼠标事件之后才能应用; 参考 drag.clickDistance.
⁴ 触摸输入时必须允许 click emulation ; 参考 #9.
⁵ 忽略触摸手势结束后 500ms
内的 click emulation.
创建一个新的拖拽行为并返回自身。drag 既是一个对象,也是一个函数,通常通过 selection.call 被应用在选中的元素上。
将拖拽应用到指定的selection。通常不适用这个方法应用拖拽,而是通过 selection.call。例如,将拖拽实例应用到一个选择集上:
d3.selectAll(".node").call(d3.drag().on("start", started));
在内部拖拽行为使用 selection.on将拖拽必需的事件绑定到元素上,事件名称都带有 .drag
,因此可以使用这个特殊的事件名来解绑拖拽事件:
selection.on(".drag", null);
应用拖拽行为时会将 -webkit-tap-highlight-color 样式设置为透明,禁止在 IOS
上的标签高亮。如果你想要不一样的样式,请在应用完拖拽行为之后移除或者重新设置这个属性。
# drag.container([container]) <源码>
如果指定了 container,则将拖拽行为的容器访问器设置为指定的对象或方法。如果没有指定 container ,则返回当前的容器访问器,默认为:
function container() {
return this.parentNode;
}
拖拽手势的 container 定义了随后 drag events 的坐标系统。影响 event.x 和 event.y。容器访问器返回的元素随后被传递给 d3.mouse 或 d3.touch,因此需要的时候要定义好容器访问器。
默认的容器访问器返回接收到初始事件的元素的父节点(参考 drag),在拖动 SVG
或者 HTML
元素时,通常是合理的,因为这些元素通常通过父元素定位。但是拖动 Canvas
中的元素时,你可能需要将容器访问器设置为 Canvas
自身:
function container() {
return this;
}
此外,设置容器访问器时,还可以直接将元素设置为参数: drag.container(canvas)
.
如果指定了 filter,则将 filter 设置为拖拽行为的过滤器。如果没有指定 filter 则返回当前的过滤器,默认为:
function filter() {
return !d3.event.button;
}
如果过滤器返回假,则初始事件会被忽略并且不会启动拖拽手势。也就是说过滤器可以定义哪些事件被忽略,默认的过滤器会忽略辅助按钮上的鼠标按下事件,因为这些按钮通常用作其他的作用,比如上下文菜单。
# drag.touchable([touchable]) <源码>
如果指定了 touchable,则设置触摸支持检测器为执行的函数。如果没有指定,则返回当前的触摸支持检测方法,默认为:
function touchable() {
return "ontouchstart" in this;
}
触摸事件监听器仅仅在触摸支持检测器返回真的时候才会被注册。默认的检测器在绝大多数浏览器中都能正常工作,但不是全部。例如Chrome的模拟移动仿真不会正常工作。
# drag.subject([subject]) <源码>
subject 此文译为 主体
, 如果 subject 指定,则为拖拽行为指定主体访问器。如果没有指定,则返回当前的主体访问器。默认为:
function subject(d) {
return d == null ? {x: d3.event.x, y: d3.event.y} : d;
}
主体代表的是 the thing being dragged(当前被拖拽)的东西。当接收到启动输入事件时被计算,比如mousedown
或touchstart
. 在拖拽启动时立即被计算。主体通过随后的 drag events 中的event.subject 来暴露。
默认情况下,主体为接收原始事件元素上的 datum. 如果 datum
没有定义则主体为输入事件的坐标(参考上述默认主体访问器)。当在 SVG
中拖拽时。默认的主体为当前元素绑定的数据,但在 Canvas
中默认的主体为 Canvas
本身绑定的数据(不关心拖拽事件触发的位置),此时自定义的主体访问器就显得有用了,比如可以将自定义主体访问器设置为鼠标事件坐标周围一定 半径
范围内的圆:
function subject() {
var n = circles.length,
i,
dx,
dy,
d2,
s2 = radius * radius,
circle,
subject;
for (i = 0; i < n; ++i) {
circle = circles[i];
dx = d3.event.x - circle.x;
dy = d3.event.y - circle.y;
d2 = dx * dx + dy * dy;
if (d2 < s2) subject = circle, s2 = d2;
}
return subject;
}
(如果需要的话可以使用 quadtree.find加速.)
返回的主体应该是一个暴露 x
和 y
属性的对象,以便在拖动手势期间保留主体和指针的相对位置。如果主体为 null
或 undefined
则不会有拖拽手势触发,然而其他的输入事件仍然可以触发,参考 drag.filter。
拖拽的主体在拖拽手势触发后不应该再改变。主体访问器与 selection.on 的回调有相同的上下文和参数: 当前绑定的数据 d
、索引 i
, this
上下文为当前 DOM
元素。在主体访问器中 d3.event 是一个 beforestart
的drag event。使用event.sourceEvent 来访问初始输入事件和 event.identifier 来访问触摸标识。event.x 和 event.y 是相对于 container,并且使用 d3.mouse 或 d3.touch计算。
# drag.clickDistance([distance]) <源码>
如果指定了 distance 则将click
事件的触发条件: mousedown
和 mouseup
之间鼠标移动的距离设置为指定的距离。如果鼠标按下时的坐标与鼠标抬起时的坐标之间的距离大于或等于 distance 则不会触发随后的 click
事件。如果没有指定 distance 则返回当前的默认值,默认为 0。距离阈值通过坐标系统 (event.clientX 和 event.clientY) 测量得到。
# drag.on(typenames, [listener]) <源码>
如果 listener 指定,则将其设置为对应的 typenames 的回调。如果对应 typenames 已经存在事件监听器则将其替换掉。如果 listener 为 null
则表示移除对应的 typenames 上的事件监听器。如果没有指定 listener,则返回第一个对应 typenames 的事件监听器,事件监听器的调用上下文以及参数与 selection.on类似:当前元素绑定的数据 d
, 索引 i
, this
指向当前 DOM
元素。
typenames 是一个或者由空格分割的多个 typename。每个 typename 都是一个 type,可选的 name 可以由 .
和 type 分割,比如 drag.foo
和 drag.bar
。使用 name 可以为同一种 type 指定多个事件监听器。type 必须为以下几种:
start
- 拖拽开始(mousedown or touchstart).drag
- 拖拽中 (mousemove or touchmove).end
- 拖拽结束 (mouseup, touchend or touchcancel).
参考 dispatch.on.
在拖拽过程中通过 drag.on 改变事件监听器不会影响当前的拖拽手势,相反,必须使用 event.on,event.on 也允许为当前的拖拽注册一个临时的事件监听器。在拖拽期间 为每个活动的指针分发一个单独的事件。例如,如果拖拽是由多个手指触发的话,start
事件会被派发给每个手指触摸点,即使两个手指同时开始触摸。参考 Drag Events
阻止指定的 window 下原生的拖拽以及文本选中事件。作为防止 mousedown
事件默认行为的替代方法(参考#9),这个方法可以防止 mousedown
事件发生后不需要的默认行为。在支持的浏览器中,这个方法捕捉 dragstart
和 selectstart
事件,阻止相关联的默认行为并且阻止冒泡。在不支持选择事件的浏览器中,将元素的 user-select CSS
属性设置为 none
。这个方法在 mousedown
时会特意触发,在 mouseup
的时候会调用 d3.dragEnable。
# d3.dragEnable(window[, noclick]) <源码>
允许指定 window上原生的拖拽和文本选中。取消 d3.dragDisable 的影响。这个方法在 mouseup
时会调用,前序的 mousedown
会调用 d3.dragDisable。如果 noclick 为 true
,这个方法会临时抑制 click
事件,抑制 click
事件在零毫秒超时后终止,这样它只会抑制紧跟在当前 mouseup
事件之后的 click
事件(如果有的话)。
当 drag event listener 被调用时, d3.event会被设置为当前的拖拽事件,event 对象暴露以下属性:
target
- 相关联的drag behavior.type
- 字符串 “start”, “drag” 或 “end”; 参考 drag.on.subject
- 通过 drag.subject定义的subject.x
- subject 的 x-坐标; 参考 drag.container.y
- subject 的 y-坐标; 参考 drag.container.dx
- 与上一次拖拽相比 x-坐标 的变化.dy
- 与上一次拖拽相比 y-坐标 的变化.identifier
- 字符串 “mouse”, 或者表示 touch identifier的数字.active
- 当前活动的拖拽手势的数量(在start和end, 不包含这个).sourceEvent
- 底层原始事件比如 mousemove 或 touchmove.
event.active 属性对判断并发的拖拽手势序列中的 start
事件和 end
事件: 在拖拽手势开始时为0,在拖拽结束最后一个手势事件时为0。
event 对象也暴露了 event.on 方法.
# event.on(typenames, [listener]) <源码>
与 drag.on 等价, 但是仅仅应用在当前的拖拽手势。在拖拽手势开始时会创建一个当前拖拽 event listeners 的 copy . 这个副本会被绑定到当前拖拽手势并且可以被 event.on 修改. 这对于仅接收当前手势的临时监听器很有用。例如下面事件监听器将临时拖拽事件以及结束事件注册为闭包:
function started() {
var circle = d3.select(this).classed("dragging", true);
d3.event.on("drag", dragged).on("end", ended);
function dragged(d) {
circle.raise().attr("cx", d.x = d3.event.x).attr("cy", d.y = d3.event.y);
}
function ended() {
circle.classed("dragging", false);
}
}