| <!DOCTYPE html> |
| <html lang="en"> |
| <head> |
| <meta charset="UTF-8" /> |
| <title>Updated Sankey Diagram</title> |
| <script src="https://d3js.org/d3.v7.min.js"></script> |
| <script src="https://cdn.jsdelivr.net/npm/d3-sankey@0.12.3/dist/d3-sankey.min.js"></script> |
| <style> |
| body { font-family: sans-serif; margin: 0; padding: 0; } |
| svg { font: bold 14px sans-serif; } |
| </style> |
| </head> |
| <body> |
| <svg id="sankey"></svg> |
| <button onclick="saveSVG()" style="margin: 10px; padding: 8px 16px;">Download SVG</button> |
| <svg id="sankey"></svg> |
|
|
| <script> |
| const data = [ |
| { source: "Total", target: "Checkpoint", value: 4520 }, |
| { source: "Total", target: "TextualInversion", value: 1590 }, |
| { source: "Total", target: "LoRA", value: 30687 }, |
| { source: "Total", target: "Other", value: 1327 }, |
| { source: "Total", target: "LOCON", value: 1876 }, |
| { source: "LoRA", target: "Adapters", value: 30687 }, |
| { source: "LOCON", target: "Adapters", value: 1876 }, |
| { source: "TextualInversion", target: "Adapters", value: 1590 }, |
| { source: "Adapters", target: "has_no_tags", value: 23002 }, |
| { source: "Adapters", target: "has_tags", value: 11151 }, |
| { source: "has_tags", target: "has_tags + POI True", value: 2327 }, |
| { source: "has_tags + POI True", target: "Danbooru", value: 991 }, |
| { source: "Danbooru", target: "explicit", value: 233 }, |
| { source: "Danbooru", target: "non-explicit", value: 758 }, |
| { source: "has_tags + POI True", target: "CLIP Interrogator", value: 772 }, |
| { source: "CLIP Interrogator", target: "explicit", value: 5 }, |
| { source: "CLIP Interrogator", target: "non-explicit", value: 767 }, |
| { source: "has_tags + POI True", target: "unknown", value: 564 }, |
| { source: "unknown", target: "non-explicit", value: 564 }, |
| { source: "has_tags", target: "has_tags + POI False", value: 8824 }, |
| { source: "has_tags + POI False", target: "Danbooru", value: 6839 }, |
| { source: "Danbooru", target: "explicit", value: 4692 }, |
| { source: "Danbooru", target: "non-explicit", value: 2147 }, |
| { source: "has_tags + POI False", target: "CLIP Interrogator", value: 800 }, |
| { source: "CLIP Interrogator", target: "explicit", value: 37 }, |
| { source: "CLIP Interrogator", target: "non-explicit", value: 763 }, |
| { source: "has_tags + POI False", target: "unknown", value: 1185 }, |
| { source: "unknown", target: "explicit", value: 3 }, |
| { source: "unknown", target: "non-explicit", value: 1182 }, |
| { source: "explicit", target: "explicit_keyword_1", value: 558 }, |
| { source: "explicit", target: "explicit_keyword_2", value: 69 }, |
| { source: "explicit", target: "explicit_keyword_3", value: 189 } |
| ]; |
| |
| const width = 1600; |
| const height = 800; |
| |
| const svg = d3.select("#sankey") |
| .attr("width", width) |
| .attr("height", height); |
| |
| |
| const nodes = Array.from( |
| new Set(data.flatMap(d => [d.source, d.target])), |
| name => ({ name }) |
| ); |
| |
| const sankeyData = { |
| nodes, |
| links: data.map(d => Object.assign({}, d)) |
| }; |
| |
| |
| const sankey = d3.sankey() |
| .nodeId(d => d.name) |
| .nodeAlign(d3.sankeyLeft) |
| .nodeWidth(10) |
| .nodePadding(60) |
| .extent([[1, 1], [width - 1, height - 6]]); |
| |
| const { nodes: layoutNodes, links: layoutLinks } = sankey(sankeyData); |
| |
| |
| const color = name => { |
| const map = { |
| "Total": "#C0C0C0", |
| "Adapters": "rosybrown", |
| "Checkpoint": "#C0C0C0", |
| "LoRA": "rosybrown", |
| "LOCON": "rosybrown", |
| "Danbooru": "crimson", |
| "Other": "#C0C0C0", |
| "Textual Training Data": "coral", |
| "No Textual Training Data": "#ccc", |
| "has_tags + POI False": "silver", |
| "has_tags + POI True": "coral", |
| "Real person": "FF7F50", |
| "POI False": "#BC8F8F", |
| "explicit": "maroon", |
| "non-explicit": "#C0C0C0", |
| "has_tags": "coral", |
| "explicit_keyword_1": "#8A2BE2", |
| "explicit_keyword_2": "#8A2BE2", |
| "explicit_keyword_3": "#8A2BE2" |
| }; |
| return map[name] || "#ccc"; |
| }; |
| |
| |
| const labelMap = { |
| "has_tags + POI True": "Real person", |
| "has_tags + POI False": "Not real person", |
| "has_no_tags": "No text training data", |
| "explicit_keyword_1": "Loli", |
| "explicit_keyword_2": "Shota", |
| "explicit_keyword_3": "Rape", |
| "has_tags": "Textual training data", |
| "explicit": "Explicit", |
| "non-explicit": "Not explicit", |
| |
| |
| }; |
| |
| const defs = svg.append("defs"); |
| |
| layoutLinks.forEach((d, i) => { |
| const grad = defs.append("linearGradient") |
| .attr("id", d.uid = `link-gradient-${i}`) |
| .attr("gradientUnits", "userSpaceOnUse") |
| .attr("x1", d.source.x1) |
| .attr("x2", d.target.x0); |
| |
| grad.append("stop") |
| .attr("offset", "0%") |
| .attr("stop-color", color(d.source.name)); |
| |
| grad.append("stop") |
| .attr("offset", "100%") |
| .attr("stop-color", color(d.target.name)); |
| }); |
| |
| |
| |
| svg.append("g") |
| .attr("fill", "none") |
| .attr("stroke-opacity", 0.5) |
| .selectAll("path") |
| .data(layoutLinks) |
| .join("path") |
| .attr("d", d3.sankeyLinkHorizontal()) |
| .attr("stroke", d => `url(#${d.uid})`) |
| .attr("stroke-width", d => Math.max(1, d.width)); |
| |
| |
| svg.append("g") |
| |
| .selectAll("rect") |
| .data(layoutNodes) |
| .join("rect") |
| .attr("x", d => d.x0) |
| .attr("y", d => d.y0) |
| .attr("height", d => d.y1 - d.y0) |
| .attr("width", d => d.x1 - d.x0) |
| .attr("fill", d => color(d.name)) |
| .append("title") |
| .text(d => `${d.name}\n${d.value}`); |
| |
| |
| svg.append("g") |
| .selectAll("text") |
| .data(layoutNodes) |
| .join("text") |
| .attr("x", d => d.x0 < width / 2 ? d.x1 + 10 : d.x0 - 10) |
| .attr("y", d => (d.y1 + d.y0) / 2) |
| .attr("dy", "0.35em") |
| .attr("text-anchor", d => d.x0 < width / 2 ? "start" : "end") |
| .style("font-size", "22px") |
| .each(function(d) { |
| const text = d3.select(this); |
| const label = labelMap[d.name] || d.name; |
| const lines = label.split(/(?<=\w)\s+(?=\w)/); |
| |
| lines.forEach((line, i) => { |
| text.append("tspan") |
| .attr("x", d.x0 < width / 2 ? d.x1 + 10 : d.x0 - 10) |
| .attr("dy", i === 0 ? "0.35em" : "1.1em") |
| .text(line); |
| }); |
| |
| const format = d3.format(","); |
| text.append("tspan") |
| .attr("x", d.x0 < width / 2 ? d.x1 + 10 : d.x0 - 10) |
| .attr("dy", "1.5em") |
| .style("font-size", "20px") |
| .style("fill", "#555") |
| .text(`${format(d.value)}`); |
| }); |
| |
| function inlineStyles(svgElement) { |
| const styles = ` |
| text { font-family: sans-serif; fill: black; font-weight: bold; } |
| rect { stroke: none; } |
| path { stroke-opacity: 0.5; } |
| `; |
| const styleElem = document.createElementNS("http://www.w3.org/2000/svg", "style"); |
| styleElem.textContent = styles; |
| svgElement.insertBefore(styleElem, svgElement.firstChild); |
| } |
| |
| function saveSVG() { |
| const svgElement = document.querySelector("svg"); |
| inlineStyles(svgElement); |
| const serializer = new XMLSerializer(); |
| const source = serializer.serializeToString(svgElement); |
| const svgBlob = new Blob([source], { type: "image/svg+xml;charset=utf-8" }); |
| const svgUrl = URL.createObjectURL(svgBlob); |
| const downloadLink = document.createElement("a"); |
| downloadLink.href = svgUrl; |
| downloadLink.download = "sankey-diagram.svg"; |
| document.body.appendChild(downloadLink); |
| downloadLink.click(); |
| document.body.removeChild(downloadLink); |
| } |
| |
| </script> |
| </body> |
| </html> |
|
|