From e4249979b67127dbeaf8d78bab6500154dcf6cde Mon Sep 17 00:00:00 2001 From: "Thomas F. K. Jorna" Date: Thu, 15 Jul 2021 15:06:41 +0200 Subject: added 3d and expandable mode --- app/components/graph/graph.tsx | 65 +++++++++++++++++++++++++++++++++++++++--- 1 file changed, 61 insertions(+), 4 deletions(-) (limited to 'app/components/graph/graph.tsx') diff --git a/app/components/graph/graph.tsx b/app/components/graph/graph.tsx index 934901f..7a4843b 100644 --- a/app/components/graph/graph.tsx +++ b/app/components/graph/graph.tsx @@ -1,5 +1,5 @@ import * as React from "react" -import { useState, useEffect, useRef } from "react" +import { useState, useEffect, useRef, useMemo, useCallback } from "react" import { StyleProp, TextStyle, View, ViewStyle } from "react-native" import { observer } from "mobx-react-lite" import { color, typography } from "../../theme" @@ -8,7 +8,9 @@ import { flatten } from "ramda" //import data from "../../data/miserables.json" //import genRandomTree from "../../data/randomdata"; -import rando from "../../data/rando.json" +//import rando from "../../data/rando.json" +import rando from "../../data/randorev.json" + import { ForceGraph2D, ForceGraph3D, ForceGraphVR, ForceGraphAR } from 'react-force-graph'; import * as d3 from "d3-force"; @@ -58,17 +60,72 @@ export const Graph = observer(function Graph(props: GraphProps) { fg.d3Force('charge').strength(physics.charge); }); + const rootId = 0; + + const nodesById = useMemo(() => { + const nodesById = Object.fromEntries(rando.nodes.map(node => [node.id, node])); + + // link parent/children + rando.nodes.forEach(node => { + node.collapsed = node.id !== rootId; + node.childLinks = []; + }); + rando.links.forEach(link => nodesById[link.source].childLinks.push(link)); + + return nodesById; + }, [rando]); + + const getPrunedTree = useCallback(() => { + const visibleNodes = []; + const visibleLinks = []; + (function traverseTree(node = nodesById[rootId]) { + visibleNodes.push(node); + if (node.collapsed) return; + visibleLinks.push(...node.childLinks); + node.childLinks + .map(link => ((typeof link.target) === 'object') ? link.target : nodesById[link.target]) // get child node + .forEach(traverseTree); + })(); + + return { nodes: visibleNodes, links: visibleLinks }; + }, [nodesById]); + + const [prunedTree, setPrunedTree] = useState(getPrunedTree()); + + const handleNodeClick = useCallback(node => { + node.collapsed = !node.collapsed; // toggle collapse state + setPrunedTree(getPrunedTree()) + }, []); + return ( + {!physics.threedim ? d.id%GROUPS} + graphData={physics.collapse ? prunedTree : rando} + // nodeAutoColorBy={d => d.id%GROUPS} + linkAutoColorBy={d => rando.nodes[d.source].id%GROUPS} + linkColor={"#ffffff"} + linkWidth={2} + linkDirectionalParticles={2} + nodeColor={node => !node.childLinks.length ? 'green' : node.collapsed ? 'red' : 'yellow'} + onNodeClick={handleNodeClick} + //d3VelocityDecay={visco} + /> + : + d.id%GROUPS} linkAutoColorBy={d => rando.nodes[d.source].id%GROUPS} linkColor={"#ffffff"} linkWidth={2} + linkDirectionalParticles={2} + nodeColor={node => !node.childLinks.length ? 'green' : node.collapsed ? 'red' : 'yellow'} + onNodeClick={handleNodeClick} //d3VelocityDecay={visco} /> + } ) }) -- cgit v1.2.3