ホームページ > ウェブフロントエンド > jsチュートリアル > React を使用して d3 フォースディレクテッド グラフを構築する方法 (詳細なチュートリアル)

React を使用して d3 フォースディレクテッド グラフを構築する方法 (詳細なチュートリアル)

亚连
リリース: 2018-06-12 12:04:50
オリジナル
4523 人が閲覧しました

この記事では主に React で d3 フォースディレクテッド グラフを構築する方法を紹介します。

D3js 強制指示グラフ構築

d3js は、データに基づいてドキュメントを操作できる JavaScript ライブラリです。データはHTML、CSS、SVG、Canvasを使用して表示できます。力指向グラフを使用して、ノード間の多対多の関係を表すことができます。

成果効果: 接続線に矢印があり、ノードをクリックすると、ノードの色と接続線の太さを変更したり、ズームしたりドラッグしたりできます。

バージョン: 4. 2. コードを逆アセンブルします

1. コンポーネント

import React, { Component } from 'react';
import PropTypes from 'prop-types';
import { connect } from 'react-redux';
import { push } from 'react-router-redux';
import * as d3 from 'd3';
import { Row, Form } from 'antd';

import { chartReq} from './actionCreator';
import './Chart.less';

const WIDTH = 1900;
const HEIGHT = 580;
const R = 30;

let simulation;

class Chart extends Component {
 constructor(props, context) {
  super(props, context);
  this.print = this.print.bind(this);
  this.forceChart = this.forceChart.bind(this);
  this.state = {

  };
 }

 componentWillMount() {
  this.props.dispatch(push('/Chart'));
 }

 componentDidMount() {
  this.print();
 }

 print() {
  let callback = (res) => { // callback获取后台返回的数据,并存入state
   let nodeData = res.data.nodes;
   let relationData = res.data.rels;
   this.setState({
    nodeData: res.data.nodes,
    relationData: res.data.rels,
   });
   let nodes = [];
   for (let i = 0; i < nodeData.length; i++) {
    nodes.push({
     id: (nodeData[i] && nodeData[i].id) || &#39;&#39;,
     name: (nodeData[i] && nodeData[i].name) || &#39;&#39;,
     type: (nodeData[i] && nodeData[i].type) || &#39;&#39;,
     definition: (nodeData[i] && nodeData[i].definition) || &#39;&#39;,
    });
   }
   let edges = [];
   for (let i = 0; i < relationData.length; i++) {
    edges.push({
     id: (relationData[i] && (relationData[i].id)) || &#39;&#39;,
     source: (relationData[i] && relationData[i].start.id) || &#39;&#39;,
     target: (relationData[i] && relationData[i].end.id) || &#39;&#39;,
     tag: (relationData[i] && relationData[i].name) || &#39;&#39;,
    });
   }
   this.forceChart(nodes, edges); // d3力导向图内容
  };
  this.props.dispatch(chartReq({ param: param }, callback));
 }

 // func
 forceChart(nodes, edges) {
  this.refs[&#39;theChart&#39;].innerHTML = &#39;&#39;;

  // 函数内其余代码请看拆解代码
  }

   render() {
  
    return (
     <Row style={{ minWidth: 900 }}>
      <p className="outerp">
       <p className="theChart" id="theChart" ref="theChart">
  
       </p>
      </p>
     </Row>
    );
   }
  }

  Chart.propTypes = {
   dispatch: PropTypes.func.isRequired,
  };
  
  function mapStateToProps(state) {
   return {
  
   };
  }
  
  const WrappedChart = Form.create({})(Chart);
  export default connect(mapStateToProps)(WrappedChart);
ログイン後にコピー
全体の絵はp.10に描かれます。

2. ノードと接続を構築します

<p className="theChart" id="theChart" ref="theChart">
</p>
ログイン後にコピー

具体的な構築はプロジェクト データに基づいています。

3. 力モデルを定義します

let nodes = []; // 节点
for (let i = 0; i < nodeData.length; i++) {
  nodes.push({
    id: (nodeData[i] && nodeData[i].id) || &#39;&#39;,
    name: (nodeData[i] && nodeData[i].name) || &#39;&#39;, // 节点名称
  });
}
let edges = []; // 连线
for (let i = 0; i < relationData.length; i++) {
  edges.push({
    id: (relationData[i] && (relationData[i].id)) || &#39;&#39;,
    source: (relationData[i] && relationData[i].start.id) || &#39;&#39;, // 开始节点
    target: (relationData[i] && relationData[i].end.id) || &#39;&#39;, // 结束节点
    tag: (relationData[i] && relationData[i].name) || &#39;&#39;, // 连线名称
  });
}
ログイン後にコピー

simulation.force() を通じて力を設定できます:

センタリング: 力の中心、グラフの中心点の位置を設定します。

Collision: ノード衝突力、.strength パラメーター範囲は [0, 1]。

リンク: 接続の力; . distance は、接続の両端のノード間の距離を設定します。

Many-Body: .strength パラメータが正の場合は重力をシミュレートし、負の場合は充電力をシミュレートします。. distanceMax パラメータは最大距離を設定します。

  1. 位置決め: 特定の方向に力が与えられます。

    simulation.on を通じて聴力図要素の位置の変化を監視します。
  2. 4. svgを描く

  3. const simulation = d3.forceSimulation(nodes) // 指定被引用的nodes数组
      .force(&#39;link&#39;, d3.forceLink(edges).id(d => d.id).distance(150))
      .force(&#39;collision&#39;, d3.forceCollide(1).strength(0.1))
      .force(&#39;center&#39;, d3.forceCenter(WIDTH / 2, HEIGHT / 2))
      .force(&#39;charge&#39;, d3.forceManyBody().strength(-1000).distanceMax(800));
    ログイン後にコピー
  4. svgを作成し、svg内にgを作成し、g内にノード接続やその他のコンテンツを配置します。

  5. select: 最初の対応する要素を選択します

selectAll: 対応するすべての要素を選択します

append: 要素を作成します

5. 接続を描画します

    const svg = d3.select(&#39;#theChart&#39;).append(&#39;svg&#39;) // 在id为‘theChart&#39;的标签内创建svg
       .style(&#39;width&#39;, WIDTH)
       .style(&#39;height&#39;, HEIGHT * 0.9)
       .on(&#39;click&#39;, () => {
        console.log(&#39;click&#39;, d3.event.target.tagName);
       })
       .call(zoom); // 缩放
    const g = svg.append(&#39;g&#39;); // 则svg中创建g
    ログイン後にコピー
  1. シェルを使用して線を接続しますアール曲線描画: (M 開始点 スケール フィル ビューポート

  2. 7. ノードを描画する

  3. const edgesLine = svg.select(&#39;g&#39;)
      .selectAll(&#39;line&#39;)
      .data(edges) // 绑定数据
      .enter() // 添加数据到选择集edgepath
      .append(&#39;path&#39;) // 生成折线
      .attr(&#39;d&#39;, (d) => { return d && &#39;M &#39; + d.source.x + &#39; &#39; + d.source.y + &#39; L &#39; + d.target.x + &#39; &#39; + d.target.y; }) // 遍历所有数据,d表示当前遍历到的数据,返回绘制的贝塞尔曲线
      .attr(&#39;id&#39;, (d, i) => { return i && &#39;edgepath&#39; + i; }) // 设置id,用于连线文字
      .attr(&#39;marker-end&#39;, &#39;url(#arrow)&#39;) // 根据箭头标记的id号标记箭头
      .style(&#39;stroke&#39;, &#39;#000&#39;) // 颜色
      .style(&#39;stroke-width&#39;, 1); // 粗细
    ログイン後にコピー
  4. ノードとして円を作成します。

    .call() はドラッグ関数を呼び出します。

8. ノード名

const defs = g.append(&#39;defs&#39;); // defs定义可重复使用的元素
const arrowheads = defs.append(&#39;marker&#39;) // 创建箭头
  .attr(&#39;id&#39;, &#39;arrow&#39;)
  // .attr(&#39;markerUnits&#39;, &#39;strokeWidth&#39;) // 设置为strokeWidth箭头会随着线的粗细进行缩放
  .attr(&#39;markerUnits&#39;, &#39;userSpaceOnUse&#39;) // 设置为userSpaceOnUse箭头不受连接元素的影响
  .attr(&#39;class&#39;, &#39;arrowhead&#39;)
  .attr(&#39;markerWidth&#39;, 20) // viewport
  .attr(&#39;markerHeight&#39;, 20) // viewport
  .attr(&#39;viewBox&#39;, &#39;0 0 20 20&#39;) // viewBox
  .attr(&#39;refX&#39;, 9.3 + R) // 偏离圆心距离
  .attr(&#39;refY&#39;, 5) // 偏离圆心距离
  .attr(&#39;orient&#39;, &#39;auto&#39;); // 绘制方向,可设定为:auto(自动确认方向)和 角度值
arrowheads.append(&#39;path&#39;)
  .attr(&#39;d&#39;, &#39;M0,0 L0,10 L10,5 z&#39;) // d: 路径描述,贝塞尔曲线
  .attr(&#39;fill&#39;, &#39;#000&#39;); // 填充颜色
ログイン後にコピー

テキストはノードの上層にあるため、マウスイベントが無効になっていない場合、テキストをクリックしてもノードをクリックしても反応せず、ノードを開くことはできません。引きずられた。

9. 接続名

const nodesCircle = svg.select(&#39;g&#39;)
  .selectAll(&#39;circle&#39;)
  .data(nodes)
  .enter()
  .append(&#39;circle&#39;) // 创建圆
  .attr(&#39;r&#39;, 30) // 半径
  .style(&#39;fill&#39;, &#39;#9FF&#39;) // 填充颜色
  .style(&#39;stroke&#39;, &#39;#0CF&#39;) // 边框颜色
  .style(&#39;stroke-width&#39;, 2) // 边框粗细
  .on(&#39;click&#39;, (node) => { // 点击事件
    console.log(&#39;click&#39;);
  })
  .call(drag); // 拖拽单个节点带动整个图
ログイン後にコピー

    10. マウスをノード
  1. const nodesTexts = svg.select(&#39;g&#39;)
      .selectAll(&#39;text&#39;)
      .data(nodes)
      .enter()
      .append(&#39;text&#39;)
      .attr(&#39;dy&#39;, &#39;.3em&#39;) // 偏移量
      .attr(&#39;text-anchor&#39;, &#39;middle&#39;) // 节点名称放在圆圈中间位置
      .style(&#39;fill&#39;, &#39;black&#39;) // 颜色
      .style(&#39;pointer-events&#39;, &#39;none&#39;) // 禁止鼠标事件
      .text((d) => { // 文字内容
        return d && d.name; // 遍历nodes每一项,获取对应的name
      });
    ログイン後にコピー

  2. に移動すると、バブルプロンプトが表示されます
  3. const edgesText = svg.select(&#39;g&#39;).selectAll(&#39;.edgelabel&#39;)
      .data(edges)
      .enter()
      .append(&#39;text&#39;) // 为每一条连线创建文字区域
      .attr(&#39;class&#39;, &#39;edgelabel&#39;)
      .attr(&#39;dx&#39;, 80)
      .attr(&#39;dy&#39;, 0);
    edgesText.append(&#39;textPath&#39;)// 设置文字内容
      .attr(&#39;xlink:href&#39;, (d, i) => { return i && &#39;#edgepath&#39; + i; }) // 文字布置在对应id的连线上
      .style(&#39;pointer-events&#39;, &#39;none&#39;)
      .text((d) => { return d && d.tag; });
    ログイン後にコピー

  4. 12。ドラッグ

nodesCircle.append(&#39;title&#39;)
  .text((node) => { // .text设置气泡提示内容
    return node.definition;
  });
ログイン後にコピー
13. ズーム

simulation.on(&#39;tick&#39;, () => {
  // 更新节点坐标
  nodesCircle.attr(&#39;transform&#39;, (d) => {
    return d && &#39;translate(&#39; + d.x + &#39;,&#39; + d.y + &#39;)&#39;;
  });
  // 更新节点文字坐标
  nodesTexts.attr(&#39;transform&#39;, (d) => {
    return &#39;translate(&#39; + (d.x) + &#39;,&#39; + d.y + &#39;)&#39;;
  });
  // 更新连线位置
  edgesLine.attr(&#39;d&#39;, (d) => {
    const path = &#39;M &#39; + d.source.x + &#39; &#39; + d.source.y + &#39; L &#39; + d.target.x + &#39; &#39; + d.target.y;
    return path;
  });
  // 更新连线文字位置
  edgesText.attr(&#39;transform&#39;, (d, i) => {
    return &#39;rotate(0)&#39;;
  });
});
ログイン後にコピー

3. その他の効果

1. ノードをクリックすると接続線が太くなります

function onDragStart(d) {
  // console.log(&#39;start&#39;);
  // console.log(d3.event.active);
  if (!d3.event.active) {
  simulation.alphaTarget(1) // 设置衰减系数,对节点位置移动过程的模拟,数值越高移动越快,数值范围[0,1]
   .restart(); // 拖拽节点后,重新启动模拟
  }
  d.fx = d.x;  // d.x是当前位置,d.fx是静止时位置
  d.fy = d.y;
}
function dragging(d) {
  d.fx = d3.event.x;
  d.fy = d3.event.y;
}
function onDragEnd(d) {
  if (!d3.event.active) simulation.alphaTarget(0);
  d.fx = null;    // 解除dragged中固定的坐标
  d.fy = null;
}
const drag = d3.drag()
  .on(&#39;start&#39;, onDragStart)
  .on(&#39;drag&#39;, dragging) // 拖拽过程
  .on(&#39;end&#39;, onDragEnd);
ログイン後にコピー
2. クリックされたノードの色が変わります
function onZoomStart(d) {
  // console.log(&#39;start zoom&#39;);
}
function zooming(d) {
  // 缩放和拖拽整个g
  // console.log(&#39;zoom ing&#39;, d3.event.transform, d3.zoomTransform(this));
  g.attr(&#39;transform&#39;, d3.event.transform); // 获取g的缩放系数和平移的坐标值。
}
function onZoomEnd() {
  // console.log(&#39;zoom end&#39;);
}
const zoom = d3.zoom()
  // .translateExtent([[0, 0], [WIDTH, HEIGHT]]) // 设置或获取平移区间, 默认为[[-∞, -∞], [+∞, +∞]]
  .scaleExtent([1 / 10, 10]) // 设置最大缩放比例
  .on(&#39;start&#39;, onZoomStart)
  .on(&#39;zoom&#39;, zooming)
  .on(&#39;end&#39;, onZoomEnd);
ログイン後にコピー

4. 使用上の注意反応中

nodesCircle.on(&#39;click, (node) => {
  edges_line.style("stroke-width",function(line){
    if(line.source.name==node.name || line.target.name==node.name){
      return 4;
    }else{
      return 0.5;
    }
  });
})
ログイン後にコピー
グラフを構築する場所 グラフは動的であるため、複数回レンダリングされる場合 (レンダリングが複数回実行され、複数回レンダリングされる場合)、以前にレンダリングされたグラフは上書きされません。複数のレンダリングと複数のグラフが発生します。構成図の関数print()をcomponentDidMount()に入れて実行すると一度だけ描画されます。

ノードと接続データを追加、削除、または変更した後、print() 関数を再度呼び出してグラフを再構築する必要があります。

データの取得先

データはreduxから取得されるのではなく、リクエスト送信後のコールバックによって直接取得されます。 5. 役立つ情報: d3 プロジェクトの検索 URL

D3js のすべてのプロジェクトの検索。 。 関連記事:

vue で前方リフレッシュ効果と後方非リフレッシュ効果を実現する方法

Vue2.5 の Table コンポーネントと Pagination コンポーネントを使用してページング機能を実装する方法

Bootstrap を統合する方法Laravelで4位?

jqueryのselectタグのオプション値を取得する方法


jsを使用して選択するオプションを動的に追加する方法(詳細なチュートリアル)

vue.jsを使用してシームレスなスクロール効果を実現する方法

以上がReact を使用して d3 フォースディレクテッド グラフを構築する方法 (詳細なチュートリアル)の詳細内容です。詳細については、PHP 中国語 Web サイトの他の関連記事を参照してください。

ソース:php.cn
このウェブサイトの声明
この記事の内容はネチズンが自主的に寄稿したものであり、著作権は原著者に帰属します。このサイトは、それに相当する法的責任を負いません。盗作または侵害の疑いのあるコンテンツを見つけた場合は、admin@php.cn までご連絡ください。
人気のチュートリアル
詳細>
最新のダウンロード
詳細>
ウェブエフェクト
公式サイト
サイト素材
フロントエンドテンプレート