首页 > 解决方案 > 如何在 d3.zoom (d3v4/v5) 中捕捉捏合和拉伸手势事件

问题描述

背景

我正在我的 d3v5 可视化中构建多个交互。单击和拖动应该让用户平移 svg 容器。滚轮应垂直平移。这两个都有效。

svg.call(d3.zoom().on("zoom", zoomed)).on("wheel.zoom", wheeled);

function zoomed() {
    current_transform = d3.zoomTransform(g);
    current_transform.x += d3.event.sourceEvent.movementX;
    current_transform.y += d3.event.sourceEvent.movementY;
    g.attr("transform", current_transform);
}

function wheeled() {
    current_transform = d3.zoomTransform(g);
    current_transform.y = current_transform.y - d3.event.deltaY;
    g.attr("transform", current_transform);
}

问题

我想允许用户使用捏合和拉伸触摸手势来缩放经典意义上的 svg。"wheel.zoom"触摸手势的类比是什么?如何注册用户在触控板或手机上捏和拉伸的回调?

笔记

更新

看起来可能是捏合手势被捏合和拉伸捕捉到"wheel.zoom"d3.event.type == "wheel"。此外,d3.event.deltaZ == 0对于所有这些事件。

标签: javascriptd3.js

解决方案


原来触控板捏是作为wheel.zoom设置d3.event.ctrlKey为True的。这就是浏览器所同意的:

https://medium.com/@auchenberg/detecting-multi-touch-trackpad-gestures-in-javascript-a2505babb10e

很不直观,但你有它。

这是我的最终代码:

svg.call(d3.zoom().on("zoom", zoomed)).on("wheel.zoom", wheeled);

transform = d3.zoomTransform(g);
transform.x += margin.left;
transform.y += margin.top;
g.attr("transform", transform);

function zoomed() {
    current_transform = d3.zoomTransform(g);
    current_transform.x += d3.event.sourceEvent.movementX;
    current_transform.y += d3.event.sourceEvent.movementY;
    g.attr("transform", current_transform);
}

function wheeled() {
    current_transform = d3.zoomTransform(g);
    if (d3.event.ctrlKey) {
        current_transform.k = current_transform.k - d3.event.deltaY * 0.01;
    } else {
        current_transform.y = current_transform.y - d3.event.deltaY;
    }
    g.attr("transform", current_transform);
}

推荐阅读