PIXI.v4实现的d3.v4力引导图(融合d3.zoom和d3.drag)

关于PIXI方面的资料比较少,在具体实现方面遇到的很多问题并不能在网上找到解答或现成的方案。经过努力看官方文档和不停的调试终于顺利解决所有bug。
这里先将源码po在这里,以后再对细节进行完善。

import React, { Component, PropTypes } from 'react';
import * as d3 from 'd3';
import * as PIXI from 'pixi.js'
import { Tooltip, Switch, Button, Icon } from 'antd';
const ButtonGroup = Button.Group
import data from '../../../static/data/data.json';

export default class D3Force extends Component {

    constructor( props ) {
        super(props);
        this.state = {
        }
    }

    resetZoom(){
        let {position, scale} = this.stage
        position.x = 0
        position.y = 0
        scale.x = 1
        scale.y = 1
        this.reload = true
        this.renderer.render(this.stage)
    }

    allUnlock(){
        data.nodes.forEach((node) => {
            node.isLocked = false
            this.updateNode(node, 0xFFFFFF)
        })
    }

    componentDidMount() {

        const width = 920, height = 650, r = 5, minScale = 1/3, maxScale = 10
        this.renderer = PIXI.autoDetectRenderer(width, height,
            {antialias:true, transparent: false, resolution: 1})
        this.stage = new PIXI.Container()
        this.transform = d3.zoomIdentity
        this.reload = false
        this.ix = 0
        this.iy = 0
        let renderer = this.renderer,
            stage = this.stage,
            transform = this.transform,
            links = new PIXI.Graphics(),
            node_x=[], node_y=[]//save the absolute coordinate of nodes
        this.canvas.appendChild(renderer.view)
        stage.addChild(links)

        renderer.view.style.position = "absolute"
        renderer.view.style.width = this.canvas.innerWidth + "px"
        renderer.view.style.height = this.canvas.innerHeight + "px"
        renderer.view.style.display = "block"
        renderer.view.style.border = "2px solid white"
        renderer.backgroundColor = 0xf3f6fa

        let simulation = d3.forceSimulation()
            .force("link", d3.forceLink().id((d)=> d.id))
            .force("charge", d3.forceManyBody())
            .force("center", d3.forceCenter(width / 2, height / 2))

        this.setState({tips:data.nodes[0]})

        let nodeColor = (isLocked) => isLocked ? 0x7EC1F3 : 0xC6C7C7

        this.tfRender = ()=>{
            let {position, scale} = stage
            position.x = transform.x
            position.y = transform.y
            scale.x = transform.k
            scale.y = transform.k
            renderer.render(stage)
        }
        let tfRender = this.tfRender

        let dragsubject = function() {
            let {position, scale} = stage
            let {event} = d3
            let x = transform.invertX(event.x),
                y = transform.invertY(event.y),
                dx, dy
            for(let i = data.nodes.length-1; i>=0; i--){
                let nx = node_x[i], ny = node_y[i]
                dx = x - nx
                dy = y - ny
                if(dx*dx + dy*dy < r*r){
                    this.ix = nx
                    this.iy = ny
                    return simulation.find((event.x-position.x)/scale.x, (event.y-position.y)/scale.y)
                }

            }
        }

        let dragstarted = function() {
            if(d3.event.subject.isLocked){
                return
            }
            if (!d3.event.active) simulation.alphaTarget(0.3).restart();
            let {subject} = d3.event
            subject.fx = subject.x
            subject.fy = subject.y
            tfRender()

        }

        let dragged = function() {
            if(d3.event.subject.isLocked){
                return
            }
            let {subject} = d3.event,
                {k} = transform
            subject.fx = (d3.event.x + this.ix * (k - 1)) / k
            subject.fy = (d3.event.y + this.iy * (k - 1)) / k
            tfRender()
        }

        let dragended = function() {
            if(d3.event.subject.isLocked){
                return
            }
            if (!d3.event.active) simulation.alphaTarget(0)
            let {subject} = d3.event
            subject.fx = null
            subject.fy = null
            tfRender()
        }

        let zoomed = function() {
            if(this.reload){
                this.reload = false
                let {transform} = d3.event
                transform.x = 0
                transform.y = 0
                transform.k = 1
            }
            transform = d3.event.transform
            tfRender()
        }

        let drawNode = function(node, stkClr){
            let {gfx} = node
            gfx.lineStyle(1.5, stkClr)
                .beginFill(nodeColor(node.isLocked))
                .drawCircle(0, 0, r)
                .endFill()
            stage.addChild(gfx)
            renderer.render(stage)
        }

        this.updateNode = function(node, stkClr){
            let { x, y, gfx } = node
            gfx.clear()
            gfx.position = new PIXI.Point(x, y)
            drawNode(node, stkClr)
        }
        let updateNode = this.updateNode

        data.nodes.forEach((node) => {

            node.gfx = new PIXI.Graphics()
            let {gfx} = node
            gfx.interactive = true
            gfx.buttonMode = true
            drawNode(node, 0xFFFFFF)
            let tipStyle = new PIXI.TextStyle({
                align: "left",
                fontSize: 10,
                fill: 0xFFFFFF,
                wordWrap: true,
                wordWrapWidth: 10,
            });
            gfx.on("rightclick", ()=>{
                    let stkclr = (node.isLocked = !node.isLocked) ? 0x000000 : 0xFFFFFF
                    updateNode(node, stkclr)
                 })
                .on("mouseover",()=>{
                    if(!node.isLocked) {
                        updateNode(node, 0x000000)
                    }
                    this.rect = new PIXI.Graphics()
                    this.rect.lineStyle(2, 0x000000, 0.4)
                        .beginFill(0x000000, 0.8)
                        .drawRoundedRect(node.x + 2*r, node.y - r - 12, 9*node.id.length, 20, 1)
                        .endFill()
                    this.tooltip = new PIXI.Text(node.id, tipStyle)
                    this.tooltip.x = node.x + 2.5*r
                    this.tooltip.y = node.y - r - 10
                    stage.addChild(this.rect)
                    stage.addChild(this.tooltip)
                    renderer.render(stage)
                })
                .on("mouseout",()=>{
                    if(!node.isLocked) {
                        updateNode(node, 0xFFFFFF)
                    }
                    stage.removeChild(this.rect)
                    stage.removeChild(this.tooltip)
                    renderer.render(stage)
                })
        })

        simulation
            .nodes(data.nodes)
            .on('tick', ticked)

        simulation.force('link')
            .links(data.links)

        d3.select(this.canvas)
            .call(d3.drag()
                .container(this.canvas)
                .subject(dragsubject)
                .on("start", dragstarted)
                .on("drag", dragged)
                .on("end", dragended)
            )
            .call(d3.zoom()
                .scaleExtent([minScale, maxScale])
                .on("zoom", zoomed.bind(this))
            )

        function ticked () {
            data.nodes.forEach((node,i) => {
                let { x, y, gfx } = node
                if(!node.isLocked){//Only update the unlocked nodes
                    gfx.position = new PIXI.Point(x, y)
                    node_x[i]=x
                    node_y[i]=y
                }
            })

            links.alpha = 0.6
            links.clear()
            data.links.forEach((link,k) => {
                let { source, target } = link,
                    {nodes} = data,
                    {length} = nodes,
                    lock_s = false,
                    lock_t = false
                let i,j
                for(i = 0; i < length; i++){
                    if(source.isLocked === true && source.id === nodes[i].id){
                        lock_s = true;
                        break;
                    }
                }

                for(j = 0; j < length; j++){
                    if(target.isLocked === true && target.id === nodes[j].id){
                        lock_t = true;
                        break;
                    }
                }
                links.lineStyle(1, 0x999999)
                let node_s = {"x":node_x[i],"y":node_y[i]},
                    node_t = {"x":node_x[j],"y":node_y[j]}
                if(lock_s){
                    links.moveTo(node_s.x, node_s.y)
                }
                else{
                    links.moveTo(source.x, source.y)
                }

                if(lock_t){
                    links.lineTo(node_t.x, node_t.y)
                }
                else{
                    links.lineTo(target.x, target.y)
                }
            })

            links.endFill()
            renderer.render(stage)
        }
    }

    render() {
        return (
            <div>
                <svg width="150px" height="60px">
                    <circle cx="12" cy="10" r="5" fill="#7EC1F3" />
                    <text x="25" y="12">Locked Node</text>
                    <circle cx="12" cy="30" r="6" fill="#118EEA" />
                    <text x="25" y="32">Hightlight Node</text>
                    <line x1="7" y1="50" x2="17" y2="50" stroke="#000"  />
                    <text x="25" y="52">New Link</text>
                </svg>

                <div ref={(canvas)=> {this.canvas=canvas}}
                     width = {window.innerWidth*0.45}
                     height = {window.innerHeight*0.9}
                >
                </div>
                <ButtonGroup>
                    <Tooltip placement="bottom" title="Center">
                        <Button type="primary" icon="scan" onClick={this.resetZoom.bind(this)} />
                    </Tooltip>
                    <Tooltip placement="bottom" title="Unlock">
                        <Button type="primary" icon="unlock" onClick={this.allUnlock.bind(this)} />
                    </Tooltip>
                </ButtonGroup>
            </div>
        )
    }
}
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 203,547评论 6 477
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 85,399评论 2 381
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 150,428评论 0 337
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 54,599评论 1 274
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 63,612评论 5 365
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,577评论 1 281
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 37,941评论 3 395
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,603评论 0 258
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 40,852评论 1 297
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,605评论 2 321
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,693评论 1 329
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,375评论 4 318
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 38,955评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,936评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,172评论 1 259
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 43,970评论 2 349
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,414评论 2 342