var SankeyChart = (
function ($) {
"use strict";
var target = {
"$targetId": "", // jQuery :: $ + targetId
"targetId" : "" // targetId :: 차트를 그릴 element id
};
var setTarget = function (targetElementId) {
target.$targetId = "#" + targetElementId;
target.targetId = targetElementId;
};
var initSvg = function () {
var margin = { top: 10, right: 10, bottom: 10, left: 10 };
var width = document.getElementById(target.targetId).offsetWidth;
var height = document.getElementById(target.targetId).offsetHeight - margin.top - margin.bottom;
var vx = width + margin.left + margin.right;
var vy = height + margin.top + margin.bottom;
return d3
.select(target.$targetId) // 그려지는 위치
.append("svg")
.attr("viewBox", "0 0 " + vx + " " + vy)
.attr("width", width)
.attr("height", height)
.append("g")
.attr("transform", "translate(" + margin.left + "," + margin.top + ")");
};
var drawEmptyChart = function () {
var margin = { top: 10, right: 10, bottom: 10, left: 10 };
var width = document.getElementById(target.targetId).offsetWidth - margin.left - margin.right;
var height = 500 - margin.top - margin.bottom;
initSvg()
.append("text")
.style("font-size", "12px")
.style("fill", "white")
.style("font-weight", 5)
.text("선택된 어플리케이션이 없습니다.")
.attr("x", width / 2)
.attr("y", height / 2);
};
var loadChart = function (data, targetElementId) {
if(targetElementId) {
setTarget(targetElementId);
}
var margin = { top: 10, right: 10, bottom: 10, left: 10 };
var width = document.getElementById(target.targetId).offsetWidth - margin.left - margin.right;
var height = 500 - margin.top - margin.bottom;
var formatNumber = d3.format(",.0f");
var format = function (d) {
return formatNumber(d);
};
var iconXs = [10, 12, 11.5, 12];
var nodeIcons = ['', '', ''];
var colors = ColorPalette.d3Chart.sankeyChart;
var svg = initSvg();
if (isEmpty(data.nodes)) {
svg
.append("text")
.style("font-size", "12px")
.style("fill", "white")
.style("font-weight", 5)
.text("해당 프로젝트에 매핑된 버전이 없습니다.")
.attr("x", width / 2)
.attr("y", height / 2);
return;
}
var sankey = d3.sankey()
.nodeWidth(36)
.nodePadding(40)
.size([width, height]);
var graph = {
nodes: [],
links: []
};
var nodeMap = { };
var color = d3.scaleOrdinal(colors);
var iconX = d3.scaleOrdinal(iconXs);
var nodeIcon = d3.scaleOrdinal(nodeIcons);
data.nodes.forEach(function (nodeInfos) {
graph.nodes.push(nodeInfos);
});
data.links.forEach(function (nodeLinks) {
graph.links.push(nodeLinks);
});
graph.nodes.forEach(function (node) {
nodeMap[node.id] = node;
});
graph.links = graph.links.map(function (link) {
return {
source: nodeMap[link.source],
target: nodeMap[link.target],
value: 1
};
});
graph = sankey(graph);
var link = svg
.append("g")
.selectAll(".link")
.data(graph.links)
.enter()
.append("path")
.attr("class", "link")
.attr("d", d3.sankeyLinkHorizontal())
.attr("stroke-width", function (d) {
return d.width;
});
link.append("title").text(function (d) {
return d.source.name + " → " + d.target.name + "\n" + format(d.value);
});
var node = svg.append("g").selectAll(".node").data(graph.nodes).enter().append("g").attr("class", "node");
node
.append("rect")
.attr("x", function (d) {
return d.x0;
})
.attr("y", function (d) {
return d.y0;
})
.attr("height", function (d) {
return d.y1 - d.y0;
})
.attr("width", sankey.nodeWidth())
.style("fill", function (d) {
return (d.color = color(d.type));
})
.style("cursor", "default")
.style("stroke", function (d) {
return d3.rgb(d.color).darker(2);
})
.append("title")
.text(function (d) {
return d.name + "\n" + format(d.value);
});
node
.append("svg:foreignObject")
.attr("x", function (d) {
return d.x0 + iconX(d.type);
})
.attr("y", function (d) {
return (d.y1 + d.y0 - 16) / 2;
})
.attr("height", "16px")
.attr("width", "16px")
.style("cursor", "default")
.html((d) => nodeIcon(d.type));
node
.append("text")
.attr("x", function (d) {
return d.x0 > 0 ? d.x0 - 6 : d.x1 - d.x0 + 6;
})
.attr("y", function (d) {
return (d.y1 + d.y0 - 18) / 2;
})
.attr("dy", "0.35em")
.style("fill", function (d) {
return (d.color = color(d.type));
})
.style("text-shadow", function (d) {
return `0 1px 0 ${(d.color = color(d.type))}`;
})
.attr("text-anchor", function (d) {
return d.x0 > 0 ? "end" : "start";
})
.text(function (d) {
return `[${d.type}]`;
})
.filter(function (d) {
return d.x < width / 2;
})
.attr("x", 6 + sankey.nodeWidth())
.attr("text-anchor", "start");
node
.append("text")
.attr("x", function (d) {
return d.x0 > 0 ? d.x0 - 6 : d.x1 - d.x0 + 6;
})
.attr("y", function (d) {
return (d.y1 + d.y0 + 18) / 2;
})
.attr("dy", ".35em")
.attr("text-anchor", function (d) {
return d.x0 > 0 ? "end" : "start";
})
.attr("transform", null)
.text(function (d) {
return d.name;
})
.filter(function (d) {
return d.x < width / 2;
})
.attr("x", 6 + sankey.nodeWidth())
.attr("text-anchor", "start");
};
return { loadChart, drawEmptyChart };
})(jQuery);
////////////////////////////////////////////////////////////////////////////////////////
// 제품-버전-투입인력 차트 생성
////////////////////////////////////////////////////////////////////////////////////////
function drawProductToManSankeyChart(pdServiceLink, pdServiceVersionLinks, targetId, 하위_크기) {
let $target = "#"+targetId;
function removeSankeyChart() {
const svgElement = d3.select($target).select("svg");
if (!svgElement.empty()) {
svgElement.remove();
}
}
const url = new UrlBuilder()
.setBaseUrl('/auth-user/api/arms/dashboard/version-assignees')
.addQueryParam('pdServiceLink', pdServiceLink)
.addQueryParam('pdServiceVersionLinks', pdServiceVersionLinks)
.addQueryParam('메인_그룹_필드', "pdServiceVersions")
.addQueryParam('하위_그룹_필드들', "assignee.assignee_accountId.keyword,assignee.assignee_displayName.keyword")
.addQueryParam('크기', pdServiceVersionLinks.split(",").length)
.addQueryParam('하위_크기', 하위_크기)
.addQueryParam("isReqType", "ISSUE")
.addQueryParam('컨텐츠_보기_여부', true)
.build();
$.ajax({
url: url,
type: "GET",
contentType: "application/json;charset=UTF-8",
dataType: "json",
progress: true,
statusCode: {
200: function (apiResponse) {
removeSankeyChart();
const data = apiResponse.response;
SankeyChart.loadChart(data, targetId);
}
}
});
}