svg = {
const svg = d3.select(DOM.svg(width, height));
const defs = svg.append("defs");
const view = svg.append("g")
.classed("view", true)
.attr("transform", `translate(${margin}, ${margin})`);
const nodes = view.selectAll("rect.node")
.data(graph.nodes)
.join("rect")
.classed("node", true)
.attr("id", d => `node-${d.index}`)
.attr("x", d => d.x0)
.attr("y", d => d.y0)
.attr("width", d => d.x1 - d.x0)
.attr("height", d => Math.max(1, d.y1 - d.y0))
.attr("fill", d => d.color)
.attr("opacity", 0.8);
const gradients = defs.selectAll("linearGradient")
.data(graph.links)
.join("linearGradient")
.attr("id", d => d.gradient.id)
gradients.append("stop").attr("offset", 0.0).attr("stop-color", d => d.source.color);
gradients.append("stop").attr("offset", 1.0).attr("stop-color", d => d.target.color);
nodes.append("title").text(d => `${d.name}\n${format(d.value)}`);
const links = view.selectAll("path.link")
.data(graph.links)
.join("path")
.classed("link", true)
.attr("d", mysankey.sankeyLinkHorizontal())
.attr("stroke", "grey")
.attr("stroke-opacity", 0.2)
.attr("stroke-width", d => Math.max(1, d.width))
.attr("fill", "none");
links.append("title").text(d => `${d.source.name} ${arrow} ${d.target.name}\n${format(d.value)}`);
function setDash(link) {
let el = view.select(`#${link.path.id}`);
let length = el.node().getTotalLength();
if(direction=="前继"){
el.attr("stroke-dasharray", `${length} ${length}`)
.attr("stroke-dashoffset", -length);
}else{
el.attr("stroke-dasharray", `${length} ${length}`)
.attr("stroke-dashoffset", length);
}
}
const gradientLinks = view.selectAll("path.gradient-link")
.data(graph.links)
.join("path")
.classed("gradient-link", true)
.attr("id", d => d.path.id)
.attr("d", mysankey.sankeyLinkHorizontal())
.attr("stroke", d => d.gradient)
.attr("stroke-opacity", 0.5)
.attr("stroke-width", d => Math.max(1, d.width))
.attr("fill", "none")
.each(setDash);
view.selectAll("text.node")
.data(graph.nodes)
.join("text")
.classed("node", true)
.attr("x", d => d.x1)
.attr("dx", 6)
.attr("y", d => (d.y1 + d.y0) / 2)
.attr("dy", "0.35em")
.attr("fill", "black")
.attr("opacity", 0.7)
.attr("text-anchor", "start")
.attr("font-size", 14)
.attr("font-family", "Arial, sans-serif")
.text(d => d.name)
.filter(d => d.x1 > width / 2)
.attr("x", d => d.x0)
.attr("dx", -6)
.attr("text-anchor", "end");
function branchAnimate(node) {
let mytextnodes = view.selectAll("text.node")
.filter((mytextnode) => {
return node.name == mytextnode.name;
});
mytextnodes.attr("font-size",14)
.transition()
.attr("fill", "black")
.attr("opacity", 0.7);
let mynodes = view.selectAll("rect.node")
.filter((mynode) => {
return node.name == mynode.name;
});
mynodes.attr("opacity", 0.8);
if(direction=="前继"){
let links = view.selectAll("path.gradient-link")
.filter((link) => {
return node.targetLinks.indexOf(link) !== -1;
});
let nextNodes = [];
links.each((link) => {
nextNodes.push(link.source);
});
links.attr("stroke-opacity", 0.5)
.transition()
.duration(duration)
.ease(d3.easeLinear)
.attr("stroke-dashoffset", 0)
.on("end", () => {
nextNodes.forEach((node) => {
branchAnimate(node);
});
});
}else{
let links = view.selectAll("path.gradient-link")
.filter((link) => {
return node.sourceLinks.indexOf(link) !== -1;
});
let nextNodes = [];
links.each((link) => {
nextNodes.push(link.target);
});
links.attr("stroke-opacity", 0.5)
.transition()
.duration(duration)
.ease(d3.easeLinear)
.attr("stroke-dashoffset", 0)
.on("end", () => {
nextNodes.forEach((node) => {
branchAnimate(node);
});
});
}
}
function ohternodes_opactiy(node) {
let allnodetexts = view.selectAll("text.node")
allnodetexts.attr("font-size",14)
.transition()
.attr("fill", "black")
.attr("opacity", 0.1);
let alllinks = view.selectAll("path.link")
alllinks.attr("stroke", "grey")
.attr("stroke-opacity", 0.1);
let allnodes = view.selectAll("rect.node")
allnodes.attr("opacity", 0.1);
branchAnimate(node);
}
function branchClear() {
gradientLinks.transition();
gradientLinks.attr("stroke-opactiy", 0.5)
.each(setDash);
let mytextnodes = view.selectAll("text.node");
mytextnodes.attr("font-size",14)
.transition()
.attr("fill", "black")
.attr("opacity", 0.7);
let alllinks = view.selectAll("path.link")
alllinks.attr("stroke", "grey")
.attr("stroke-opacity", 0.2);
let mynodes = view.selectAll("rect.node")
mynodes.attr("opacity", 0.8);
}
nodes.on("mouseover", ohternodes_opactiy)
.on("mouseout", branchClear);
return svg.node();
}