d3.event 在 ReactJS + d3JS 组件中为 null

IT技术 d3.js reactjs system
2021-04-19 04:04:35

我正在使用ReactJS、d3JS 和 ES6创建组织结构图。我可以创建图表并查看它。但我想将缩放行为添加到图表中,所以我使用了d3.behavior.zoom我可以看到该方法缩放被调用,但d3.event为空。

在我的 ASP.NET 项目中,我确保除了在 systemJS 配置中没有任何对d3.js 的引用,其中许多 stackoverflow 答案提到与d3.event相关的问题为空。

这是我的 systemJS 配置:

<script>
    System.defaultJSExtensions = true;
    System.config({
        map: {
            'rx': "https://cdnjs.cloudflare.com/ajax/libs/rxjs/4.0.7/rx.all.min.js",
            'react': 'https://cdnjs.cloudflare.com/ajax/libs/react/15.0.1/react.js',
            'react-dom': 'https://cdnjs.cloudflare.com/ajax/libs/react/15.0.1/react-dom.js',
            'd3': 'https://cdnjs.cloudflare.com/ajax/libs/d3/3.5.16/d3.js'
        }
    });
    System.import('../app/app.js');
</script>

组织结构图组件:

import React, { Component } from 'react';
import * as Rx from 'rx';
import * as d3 from 'd3';
import Person from './person';

class OrgChart extends Component {

    constructor(props) {
        super(props);

        this.state = { persons: [] };
    }

    componentWillMount() {
        var source = Rx.Observable.fromPromise($.getJSON("/js/org-persons.json"));

        source.subscribe(
            function(chart) {
                var tree = d3.layout.tree()
                    .nodeSize([110, 50])
                    .separation(function() {
                        return 1;
                    });

                var nodes = tree.nodes(chart[0]).reverse();

                var links = tree.links(nodes);

                var p = [];

                nodes.forEach(function(node) {
                    fs.push((<Person x={node.x} y={node.y}></Person>));
                }.bind(this));

                this.setState({ person: p });

            }.bind(this)
        );
    }

    componentDidMount() {
        var rootX = document.body.clientWidth / 4;
        var rootY = 300;

        var zoom = d3.behavior.zoom()
            .scaleExtent([1, 10])
            .on("zoom", this.zoomed)
            .translate([rootX, rootY]);

        d3.select("#viewport")
            .call(zoom);
    }

    zoomed() {
        d3.select("#viewport").attr("transform", "translate(" + d3.event.translate + ")scale(" + d3.event.scale + ")");
    }

    render() {

        return (
            <svg id='fullTree' width='100%'>
                <g id='viewport'>
                    { this.state.persons }
                </g>
            </svg>
        );
    }
}

export default OrgChart;
5个回答

如果你使用的是 React、Babel 或 Webpack 和 d3.js v4,看起来你需要event从 d3-selection导入才能使用d3.event

import * as d3 from 'd3';
import {event as currentEvent} from 'd3-selection';

您现在可以currentEvent像使用d3.event. 有关更多信息,请参阅https://github.com/d3/d3/issues/2733

我有同样的问题,但使用d3v3。我很幸运有import {event as currentEvent} from 'd3'; 来源
2021-06-06 04:04:35

如果您要导入多个 d3 包。这些包有一个可访问的 d3 对象,可能会相互冲突。您必须使用不同的名称导入它们。

import d3 from 'd3-selection'
import d3Zoom from 'd3-zoom'
import d3Scale from 'd3-scale'

并为您的回调提供匿名函数

.on("zoom", () => d3.select("#viewport")
    .attr("transform", "translate(" + d3.event.translate + ")scale(" + d3.event.scale + ")");
)
你为什么要进口import d3Scale from 'd3-scale'
2021-05-27 04:04:35
这应该是公认的答案。出于某种原因,导入eventfromd3-selection对我不起作用,同时import d3 from d3-selection解决了问题!
2021-06-05 04:04:35
import d3Scale from 'd3-scale'添加只是作为示例。与上面的代码段无关。
2021-06-17 04:04:35

您是否尝试像这样导入 d3 : import d3 from 'd3';

当文档中存在两次 d3 时,似乎会发生此问题。

快速拍摄,试试这个。

    var zoom = d3.behavior.zoom()
        .scaleExtent([1, 10])
        .on("zoom", () => d3.select("#viewport").attr("transform", "translate(" + d3.event.translate + ")scale(" + d3.event.scale + ")");)
        .translate([rootX, rootY]);

如果这有效,那么以下应该可以解决您的问题,

var zoom = d3.behavior.zoom()
            .scaleExtent([1, 10])
            .on("zoom", this.zoomed.bind(this))
            .translate([rootX, rootY]);
您可以将 d3 作为参数传递给回调函数,例如 .on("zoom", (d3) => d3.select("#viewport").attr("transform", "translate(" + d3.event.translate + ")scale(" + d3.event.scale + ")");)
2021-06-10 04:04:35

我不知道究竟你在做什么,但它闻起来像你这得太多。理论上,缩放行为会改变比例范围。

所以你所要做的就是将你的可视化建立在两种尺度上——一种用于x坐标,一种用于y坐标。然后你告诉 zoom 改变这些比例,你的可视化会自动改变。

基本上,放大只是意味着xy地图更宽。zoom-level = 1一个概念坐标(0.1, 0.1)映射到,比如说,(10, 10)当缩放级别更改为 2 时,相同的映射变为(20, 20)这会产生放大的印象。

这是我的一些使用此原理创建缩放的旧代码:https : //github.com/Swizec/candidate-bucket-chart/blob/169779e110c8825f4545c71ae5845ff7bad4f1f4/src/BubbleChart.jsx

React 本身现在有点过时了,我还没有时间将缩放推断为一个更清晰的示例项目,但我希望它有所帮助。

请记住,forceUpdate因为 React 不知道您的比例发生了变化。

我将我的代码与您的代码进行了比较。它是相同的,并且在响应中在正确的 API 上进行的调用。但是我将d3.event 中的事件设为 null。如果你已经看到d3.event 中的事件不为空,这个问题可能是 ReactJS 相关的 1.15 因为那是我使用的。
2021-06-18 04:04:35