Skip to content

Instantly share code, notes, and snippets.

@jedahan
Forked from mbostock/.block
Last active August 29, 2015 14:04
Show Gist options
  • Save jedahan/d1c0bbf64013cccbfc6a to your computer and use it in GitHub Desktop.
Save jedahan/d1c0bbf64013cccbfc6a to your computer and use it in GitHub Desktop.

Click to add nodes! Nodes near the cursor will be linked to the new node.

D3's force layout uses the Barnes–Hut approximation to compute repulsive charge forces between all nodes efficiently. Links are implemented as geometric constraints on top of position Verlet integration, offering greater stability. A virtual spring between each node and the center of the chart prevents nodes from drifting into space.

<!DOCTYPE html>
<meta charset="utf-8">
<style>
rect {
fill: none;
pointer-events: all;
}
.node {
fill: #000;
}
.cursor {
fill: none;
stroke: brown;
pointer-events: none;
}
.link {
stroke: #666;
stroke-width: 1.5px;
}
</style>
<body>
<script src=http://d3js.org/d3.v3.js charset=utf-8></script>
<script>
/*
var ws = new WebSocket('ws://localhost:8080/');
ws.onmessage = function(event) {
data = event.data;
console.log(data);
};
*/
var width = 960,
height = 500;
var fill = d3.scale.category20();
var force = d3.layout.force()
.size([width, height])
.nodes([{}]) // initialize with a single node
.linkDistance(30)
.charge(-60)
.on("tick", tick);
var svg = d3.select("body").append("svg")
.attr("width", width)
.attr("height", height)
.on("mousemove", mousemove)
.on("mousedown", mousedown);
svg.append("rect")
.attr("width", width)
.attr("height", height);
var nodes = force.nodes(),
links = force.links(),
node = svg.selectAll(".node"),
link = svg.selectAll(".link");
var cursor = svg.append("circle")
.attr("r", 30)
.attr("transform", "translate(-100,-100)")
.attr("class", "cursor");
restart();
function addnode(id){
for(var i=0; i<nodes.length; i++){
if(nodes[i].id===id){
console.log("node "+id+" already exists, not adding");
return;
}
}
console.log("adding node "+id);
var node = {x: 200, y: 50, id: id};
nodes.push(node);
restart();
return node;
}
function findnode(id){
return nodes.filter(function(node){
return node.id===id;
})
}
function findlink(source_id, target_id){
return links.filter(function(link){
return (link.source.id===source_id && link.target.id===target_id) ||
(link.source.id===target_id && link.target.id===source_id);
})
}
function addlink(source_id, target_id){
// find or create nodes
var source = findnode(source_id)[0] || addnode(source_id);
var target = findnode(target_id)[0] || addnode(target_id);
// find or create source and target node
if(findlink(source_id,target_id).length){
console.log("link "+source_id+" => "+target_id+" already exists, not adding");
return;
}
console.log("adding link "+source_id+" => "+target_id);
links.push({source: source, target: target})
restart();
}
function mousemove() {
cursor.attr("transform", "translate(" + d3.mouse(this) + ")");
}
function mousedown() {
var point = d3.mouse(this),
node = {x: point[0], y: point[1]},
n = nodes.push(node);
// add links to any nearby nodes
nodes.forEach(function(target) {
var x = target.x - node.x,
y = target.y - node.y;
if (Math.sqrt(x * x + y * y) < 30) {
links.push({source: node, target: target});
}
});
restart();
}
function tick() {
link.attr("x1", function(d) { return d.source.x; })
.attr("y1", function(d) { return d.source.y; })
.attr("x2", function(d) { return d.target.x; })
.attr("y2", function(d) { return d.target.y; });
node.attr("cx", function(d) { return d.x; })
.attr("cy", function(d) { return d.y; });
}
function restart() {
link = link.data(links);
link.enter().insert("line", ".node")
.attr("class", "link");
node = node.data(nodes);
node.enter().insert("circle", ".cursor")
.attr("class", "node")
.attr("r", 5)
.call(force.drag);
force.start();
}
</script>
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment