///////////////////
//Page 전역 변수
///////////////////
var selectedPdServiceId;
var selectedVersionId;
var versionListData;
var globalDeadline;
// 최상단 메뉴 변수
var req_state, resource_info, issue_info, period_info, total_days_progress;
// 필요시 작성
////////////////////////////////////////////////////////////////////////////////////////
//Document Ready
////////////////////////////////////////////////////////////////////////////////////////
function execDocReady() {
var pluginGroups = [
[
"../reference/light-blue/lib/vendor/jquery.ui.widget.js",
"../reference/light-blue/lib/vendor/http_blueimp.github.io_JavaScript-Templates_js_tmpl.js",
"../reference/light-blue/lib/vendor/http_blueimp.github.io_JavaScript-Load-Image_js_load-image.js",
"../reference/light-blue/lib/vendor/http_blueimp.github.io_JavaScript-Canvas-to-Blob_js_canvas-to-blob.js",
"../reference/light-blue/lib/jquery.iframe-transport.js",
// echarts
"../reference/jquery-plugins/echarts-5.4.3/dist/echarts.min.js",
// d3(게이지 차트 사용)
"../reference/jquery-plugins/d3-5.16.0/d3.min.js",
// chart Colors
"./js/common/colorPalette.js",
// 최상단 메뉴
"js/analysis/topmenu/topMenuApi.js",
"./js/common/chart/eCharts/basicRadar.js",
// 버전 timeline js, css
"./js/analysis/time/D_analysisTime.js",
"./js/analysis/time/timeline_analysisTime.js",
"./js/dashboard/chart/infographic_custom.css",
// 히트맵 사용 js, css
"./js/analysis/time/calendar_yearview_blocks_analysisTime.js",
"../reference/jquery-plugins/github-calendar-heatmap/css/calendar_yearview_blocks.css",
],
[
"../reference/lightblue4/docs/lib/slimScroll/jquery.slimscroll.min.js",
"../reference/jquery-plugins/unityping-0.1.0/dist/jquery.unityping.min.js",
"../reference/light-blue/lib/bootstrap-datepicker.js",
"../reference/jquery-plugins/datetimepicker-2.5.20/build/jquery.datetimepicker.min.css",
"../reference/jquery-plugins/datetimepicker-2.5.20/build/jquery.datetimepicker.full.min.js",
"../reference/lightblue4/docs/lib/widgster/widgster.js"
],
[
"../reference/jquery-plugins/select2-4.0.2/dist/css/select2_lightblue4.css",
"../reference/jquery-plugins/lou-multi-select-0.9.12/css/multiselect-lightblue4.css",
"../reference/jquery-plugins/multiple-select-1.5.2/dist/multiple-select-bluelight.css",
"../reference/jquery-plugins/select2-4.0.2/dist/js/select2.min.js",
"../reference/jquery-plugins/lou-multi-select-0.9.12/js/jquery.quicksearch.js",
"../reference/jquery-plugins/lou-multi-select-0.9.12/js/jquery.multi-select.js",
"../reference/jquery-plugins/multiple-select-1.5.2/dist/multiple-select.min.js"
],
[
"../reference/jquery-plugins/dataTables-1.10.16/media/css/jquery.dataTables_lightblue4.css",
"../reference/jquery-plugins/dataTables-1.10.16/extensions/Responsive/css/responsive.dataTables_lightblue4.css",
"../reference/jquery-plugins/dataTables-1.10.16/extensions/Select/css/select.dataTables_lightblue4.css",
"../reference/jquery-plugins/dataTables-1.10.16/media/js/jquery.dataTables.min.js",
"../reference/jquery-plugins/dataTables-1.10.16/extensions/Responsive/js/dataTables.responsive.min.js",
"../reference/jquery-plugins/dataTables-1.10.16/extensions/Select/js/dataTables.select.min.js",
"../reference/jquery-plugins/dataTables-1.10.16/extensions/RowGroup/js/dataTables.rowsGroup.min.js",
"../reference/jquery-plugins/dataTables-1.10.16/extensions/Buttons/js/dataTables.buttons.min.js",
"../reference/jquery-plugins/dataTables-1.10.16/extensions/Buttons/js/buttons.html5.js",
"../reference/jquery-plugins/dataTables-1.10.16/extensions/Buttons/js/buttons.print.js",
"../reference/jquery-plugins/dataTables-1.10.16/extensions/Buttons/js/jszip.min.js"
]
// 추가적인 플러그인 그룹들을 이곳에 추가하면 됩니다.
];
loadPluginGroupsParallelAndSequential(pluginGroups)
.then(function () {
console.log("모든 플러그인 로드 완료");
//vfs_fonts 파일이 커서 defer 처리 함.
setTimeout(function () {
var script = document.createElement("script");
script.src = "../reference/jquery-plugins/dataTables-1.10.16/extensions/Buttons/js/vfs_fonts.js";
script.defer = true; // defer 속성 설정
document.head.appendChild(script);
}, 5000); // 5초 후에 실행됩니다.
//pdfmake 파일이 커서 defer 처리 함.
setTimeout(function () {
var script = document.createElement("script");
script.src = "../reference/jquery-plugins/dataTables-1.10.16/extensions/Buttons/js/pdfmake.min.js";
script.defer = true; // defer 속성 설정
document.head.appendChild(script);
}, 5000); // 5초 후에 실행됩니다.
// 사이드 메뉴 처리
$(".widget").widgster();
setSideMenu("sidebar_menu_analysis", "sidebar_menu_analysis_time");
//제품(서비스) 셀렉트 박스 이니시에이터
makePdServiceSelectBox();
//버전 멀티 셀렉트 박스 이니시에이터
makeVersionMultiSelectBox();
// 높이 조정
$('.top-menu-div').matchHeight({
target: $('.top-menu-div-scope')
});
// candleStickChart();
})
.catch(function (error) {
console.error("플러그인 로드 중 오류 발생" + error);
});
}
///////////////////////
//제품 서비스 셀렉트 박스
//////////////////////
function makePdServiceSelectBox() {
//제품 서비스 셀렉트 박스 이니시에이터
$(".chzn-select").each(function () {
$(this).select2($(this).data());
});
//제품 서비스 셀렉트 박스 데이터 바인딩
$.ajax({
url: "/auth-user/api/arms/pdService/getPdServiceMonitor.do",
type: "GET",
contentType: "application/json;charset=UTF-8",
dataType: "json",
progress: true,
statusCode: {
200: function (data) {
//////////////////////////////////////////////////////////
for (let k in data.response) {
let obj = data.response[k];
let newOption = new Option(obj.c_title, obj.c_id, false, false);
$("#selected_pdService").append(newOption).trigger("change");
}
}
}
});
$("#selected_pdService").on("select2:open", function () {
//슬림스크롤
makeSlimScroll(".select2-results__options");
});
// --- select2 ( 제품(서비스) 검색 및 선택 ) 이벤트 --- //
$("#selected_pdService").on("select2:select", function (e) {
// 제품( 서비스 ) 선택했으니까 자동으로 버전을 선택할 수 있게 유도
// 디폴트는 base version 을 선택하게 하고 ( select all )
//~> 이벤트 연계 함수 :: Version 표시 jsTree 빌드
dateTimePickerBinding();
dailyChartDataSearchEvent();
baseDateReset();
bind_VersionData_By_PdService();
let checked = $("#checkbox1").is(":checked");
let endPointUrl = "";
// if (checked) {
// endPointUrl = "/T_ARMS_REQSTATUS_" + $("#selected_pdService").val() + "/getStatusMonitor.do?disable=true";
// } else {
// endPointUrl = "/T_ARMS_REQSTATUS_" + $("#selected_pdService").val() + "/getStatusMonitor.do?disable=false";
// }
console.log("[ analysisTime :: makePdServiceSelectBox ] :: 선택된 제품(서비스) c_id = " + $("#selected_pdService").val());
});
} // end makePdServiceSelectBox()
function bind_VersionData_By_PdService() {
$(".multiple-select option").remove();
$.ajax({
url: "/auth-user/api/arms/pdService/getVersionList.do?c_id=" + $("#selected_pdService").val(),
type: "GET",
dataType: "json",
progress: true,
statusCode: {
200: function (data) {
console.log("[ analysisTime :: bind_VersionData_By_PdService ] :: 선택된 버전 데이터 = ");
console.log(data.response); // versionData
versionListData = data.response.reduce((obj, item) => {
obj[item.c_id] = item;
return obj;
}, {});
let pdServiceVersionIds = [];
for (let k in data.response) {
let obj = data.response[k];
pdServiceVersionIds.push(obj.c_id);
let newOption = new Option(obj.c_title, obj.c_id, true, false);
$(".multiple-select").append(newOption);
}
selectedPdServiceId = $("#selected_pdService").val();
selectedVersionId = pdServiceVersionIds.join(",");
if (!selectedPdServiceId || selectedPdServiceId === null || selectedPdServiceId === undefined || selectedPdServiceId === "") {
return;
}
baseDateReset();
// 최상단 메뉴 세팅
TopMenuApi.톱메뉴_초기화();
TopMenuApi.톱메뉴_세팅();
// 버전 및 게이지차트, 버전 타임라인 차트 초기화
statisticsMonitor(selectedPdServiceId, selectedVersionId);
// 히트맵 차트 초기화
calendarHeatMap(selectedPdServiceId, selectedVersionId);
// 요구사항 및 연결된 이슈 생성 누적 개수 및 업데이트 상태 현황 멀티 스택바 차트
dailyUpdatedStatusScatterChart(selectedPdServiceId, selectedVersionId);
dailyCreatedCountAndUpdatedStatusesMultiStackCombinationChart(selectedPdServiceId, selectedVersionId);
// vertical timeline chart
//verticalTimeLineChart(selectedPdServiceId, selectedVersionId, 1);
timeLineChart(selectedPdServiceId, selectedVersionId);
if (data.length > 0) {
console.log("display 재설정.");
}
//$('#multiversion').multipleSelect('refresh');
//$('#edit_multi_version').multipleSelect('refresh');
$(".multiple-select").multipleSelect("refresh");
//////////////////////////////////////////////////////////
}
}
});
}
////////////////////
//버전 멀티 셀렉트 박스
////////////////////
function makeVersionMultiSelectBox() {
//버전 선택시 셀렉트 박스 이니시에이터
$(".multiple-select").multipleSelect({
filter: true,
onClose: function () {
console.log("[ analysisTime :: makeVersionMultiSelectBox ] :: onOpen event fire!\n");
let checked = $("#checkbox1").is(":checked");
let endPointUrl = "";
let versionTag = $(".multiple-select").val();
if (!versionTag || versionTag === null || versionTag === undefined || versionTag === "" || versionTag.length === 0) {
alert("버전이 선택되지 않았습니다.");
return;
}
selectedPdServiceId = $("#selected_pdService").val();
selectedVersionId = versionTag.join(",");
if (selectedPdServiceId === null || selectedPdServiceId === undefined || selectedPdServiceId === "") {
return;
}
baseDateReset();
// 최상단 메뉴 통계
TopMenuApi.톱메뉴_초기화();
TopMenuApi.톱메뉴_세팅();
// 버전 및 게이지차트, 버전 타임라인 차트 초기화
statisticsMonitor(selectedPdServiceId, selectedVersionId);
// 히트맵 차트 초기화
calendarHeatMap(selectedPdServiceId, selectedVersionId);
// 요구사항 및 연결된 이슈 생성 누적 개수 및 업데이트 상태 현황 멀티 스택바 차트
dailyUpdatedStatusScatterChart(selectedPdServiceId, selectedVersionId);
dailyCreatedCountAndUpdatedStatusesMultiStackCombinationChart(selectedPdServiceId, selectedVersionId);
// getRelationJiraIssueByPdServiceAndVersions(selectedPdServiceId, selectedVersionId);
// timeline chart (비동기 유지 API 2개)
timeLineChart(selectedPdServiceId, selectedVersionId);
$(".ms-parent").css("z-index", 1000);
},
onOpen: function() {
console.log("open event");
$(".ms-parent").css("z-index", 9999);
}
});
}
function dateTimePickerBinding() {
let today = new Date();
$('#scatter_start_date').datetimepicker({
theme:'dark',
onShow: function(ct) {
this.setOptions({
maxDate: $('#scatter_end_date').val()?$('#scatter_end_date').datetimepicker('getValue'):false
})
},
timepicker: false,
format: 'Y-m-d',
onSelectDate: function(ct, $i) {
var startDate = $('#scatter_start_date').datetimepicker('getValue');
var endDate = $('#scatter_end_date').datetimepicker('getValue');
var dayDifference = (endDate - startDate) / (1000 * 60 * 60 * 24);
if (dayDifference > 31) {
alert('시작일과 종료일의 차이는 최대 30일입니다.');
var newDate = new Date(endDate);
newDate.setDate(endDate.getDate() - 30);
$i.val(formatDate(newDate));
}
},
});
$('#scatter_end_date').datetimepicker({
theme:'dark',
onShow: function(ct) {
this.setOptions({
// minDate: $('#scatter_start_date').val()?$('#scatter_start_date').datetimepicker('getValue'):false,
maxDate: today
})
},
timepicker: false,
format: 'Y-m-d',
onSelectDate: function(ct, $i) {
var startDate = $('#scatter_start_date').datetimepicker('getValue');
var endDate = $('#scatter_end_date').datetimepicker('getValue');
var dayDifference = (endDate - startDate) / (1000 * 60 * 60 * 24);
if (dayDifference > 31) {
alert('시작일과 종료일의 차이는 최대 30일입니다.');
var newDate = new Date(startDate);
newDate.setDate(startDate.getDate() + 30);
$i.val(formatDate(newDate));
}
},
// onClose: onScatterChartDateEndChanged
});
$('#multi_stack_start_date').datetimepicker({
theme:'dark',
onShow: function(ct) {
this.setOptions({
maxDate: $('#multi_stack_end_date').val()?$('#multi_stack_end_date').datetimepicker('getValue'):false
})
},
timepicker: false,
format: 'Y-m-d',
onSelectDate: function(ct, $i) {
var startDate = $('#multi_stack_start_date').datetimepicker('getValue');
var endDate = $('#multi_stack_end_date').datetimepicker('getValue');
var dayDifference = (endDate - startDate) / (1000 * 60 * 60 * 24);
if (dayDifference > 31) {
alert('시작일과 종료일의 차이는 최대 30일입니다.');
var newDate = new Date(endDate);
newDate.setDate(endDate.getDate() - 30);
$i.val(formatDate(newDate));
}
}
});
$('#multi_stack_end_date').datetimepicker({
theme:'dark',
onShow: function(ct) {
this.setOptions({
// minDate: $('#multi_stack_start_date').val()?$('#multi_stack_start_date').datetimepicker('getValue'):false,
maxDate: today
})
},
timepicker: false,
format: 'Y-m-d',
onSelectDate: function(ct, $i) {
var startDate = $('#multi_stack_start_date').datetimepicker('getValue');
var endDate = $('#multi_stack_end_date').datetimepicker('getValue');
var dayDifference = (endDate - startDate) / (1000 * 60 * 60 * 24);
if (dayDifference > 31) {
alert('시작일과 종료일의 차이는 최대 30일입니다.');
var newDate = new Date(startDate);
newDate.setDate(startDate.getDate() + 30);
$i.val(formatDate(newDate));
}
},
// onClose: onMultiStackChartDateEndChanged
});
$('#timeline_start_date').datetimepicker({
theme:'dark',
onShow: function(ct) {
this.setOptions({
maxDate: $('#timeline_end_date').val()?$('#timeline_end_date').datetimepicker('getValue'):false
})
},
timepicker: false,
format: 'Y-m-d',
onSelectDate: function(ct, $i) {
var startDate = $('#timeline_start_date').datetimepicker('getValue');
var endDate = $('#timeline_end_date').datetimepicker('getValue');
//var dayDifference = (endDate - startDate) / (1000 * 60 * 60 * 24);
var monthDifference = endDate.getMonth() - startDate.getMonth() +
(12 * (endDate.getFullYear() - startDate.getFullYear()));
if (monthDifference > 6) {
alert('시작일과 종료일의 차이는 최대 6개월입니다.');
var newDate = new Date(endDate);
//newDate.setDate(endDate.getDate() - 30);
newDate.setMonth(endDate.getMonth() - 6);
$i.val(formatDate(newDate));
}
}
});
$('#timeline_end_date').datetimepicker({
theme:'dark',
onShow: function(ct) {
this.setOptions({
// minDate: $('#timeline_start_date').val()?$('#timeline_start_date').datetimepicker('getValue'):false,
maxDate: today
})
},
timepicker: false,
format: 'Y-m-d',
onSelectDate: function(ct, $i) {
var startDate = $('#timeline_start_date').datetimepicker('getValue');
var endDate = $('#timeline_end_date').datetimepicker('getValue');
//var dayDifference = (endDate - startDate) / (1000 * 60 * 60 * 24);
var monthDifference = endDate.getMonth() - startDate.getMonth() +
(12 * (endDate.getFullYear() - startDate.getFullYear()));
if (monthDifference > 6) {
alert('시작일과 종료일의 차이는 최대 6개월입니다.');
var newDate = new Date(startDate);
newDate.setMonth(startDate.getMonth() + 6);
$i.val(formatDate(newDate));
}
},
// onClose: onMultiStackChartDateEndChanged
});
}
function baseDateReset() {
globalDeadline = undefined;
let today = new Date();
$("#scatter_end_date").val(formatDate(today));
$("#multi_stack_end_date").val(formatDate(today));
$("#timeline_end_date").val(formatDate(today));
let aMonthAgo = new Date();
aMonthAgo.setDate(today.getDate() - 30);
$("#scatter_start_date").val(formatDate(aMonthAgo));
$("#multi_stack_start_date").val(formatDate(aMonthAgo));
$("#timeline_start_date").val(formatDate(aMonthAgo));
}
function waitForGlobalDeadline() {
return new Promise(resolve => {
let intervalId = setInterval(() => {
if (globalDeadline !== undefined) {
clearInterval(intervalId);
resolve(globalDeadline);
}
}, 100); // 100ms마다 globalDeadline 값 확인
});
}
function formatDate(date) {
var year = date.getFullYear();
var month = (date.getMonth() + 1).toString().padStart(2, "0");
var day = date.getDate().toString().padStart(2, "0");
return year + "-" + month + "-" + day;
}
function statisticsMonitor(pdservice_id, pdservice_version_id) {
console.log("[ analysisTime :: statisticsMonitor ] :: 선택된 서비스 ===> " + pdservice_id);
console.log("[ analysisTime :: statisticsMonitor ] :: 선택된 버전 리스트 ===> " + pdservice_version_id);
$(".spinner").html(
' ' +
"진행 현황 정보를 가져오는 중입니다..."
);
//1. 좌상 게이지 차트 및 타임라인
//2. Time ( 작업일정 ) - 버전 개수 삽입
$.ajax({
url: "/auth-user/api/arms/pdService/getNodeWithVersionOrderByCidDesc.do?c_id=" + pdservice_id,
type: "GET",
contentType: "application/json;charset=UTF-8",
dataType: "json",
progress: true,
async: false,
statusCode: {
200: function (json) {
let versionData = json.pdServiceVersionEntities;
versionData.sort((a, b) => a.c_id - b.c_id);
let version_count = versionData.length;
console.log("[ analysisTime :: statisticsMonitor ] :: 등록된 버전 개수 = " + version_count);
if (version_count !== undefined) {
$("#version_prgress").text(version_count);
if (version_count >= 0) {
let today = new Date();
$("#notifyNoVersion").slideUp();
$("#project-start").show();
$("#project-end").show();
$("#versionGaugeChart").html(""); //게이지 차트 초기화
var versionGauge = [];
var versionTimeline = [];
var versionCustomTimeline = [];
versionData.forEach(function (versionElement, idx) {
if (pdservice_version_id.includes(versionElement.c_id)) {
var gaugeElement = {
"current_date": today.toString(),
"version_name": versionElement.c_title,
"version_id": versionElement.c_id,
"start_date": (versionElement.c_pds_version_start_date === "start" ? today : versionElement.c_pds_version_start_date),
"end_date": (versionElement.c_pds_version_end_date === "end" ? today : versionElement.c_pds_version_end_date)
}
versionGauge.push(gaugeElement);
}
var timelineElement = {
"id" : versionElement.c_id,
"title" : "버전: "+versionElement.c_title,
"startDate" : (versionElement.c_pds_version_start_date === "start" ? today : versionElement.c_pds_version_start_date),
"endDate" : (versionElement.c_pds_version_end_date === "end" ? today : versionElement.c_pds_version_end_date)
};
versionTimeline.push(timelineElement);
var versionTimelineCustomData = {
"title" : versionElement.c_title,
"startDate" : (versionElement.c_pds_version_start_date === "start" ? today : versionElement.c_pds_version_start_date),
"endDate" : (versionElement.c_pds_version_end_date === "end" ? today : versionElement.c_pds_version_end_date)
};
versionCustomTimeline.push(versionTimelineCustomData);
});
drawVersionProgress(versionGauge); // 버전 게이지
// 이번 달의 첫째 날 구하기
var firstDay = new Date(today.getFullYear(), today.getMonth(), 1);
// 이번 달의 마지막 날 구하기
var lastDay = new Date(today.getFullYear(), today.getMonth() + 1, 0);
// 이번달 일수 구하기
var daysCount = lastDay.getDate();
// 오늘 일자 구하기
var day = today.getDate();
var today_flag = {
title: "오늘",
startDate: formatDate(firstDay),
endDate: formatDate(lastDay),
id: "today_flag"
};
versionTimeline.push(today_flag);
$("#version-timeline-bar").show();
Timeline.init($("#version-timeline-bar"), versionTimeline);
var basePosition = $("#today_flag").css("left");
var baseWidth = $(".month").css("width");
var calFlagPosition = (parseFloat(baseWidth) / daysCount) * day;
var flagPosition = parseFloat(basePosition) + calFlagPosition + "px";
$("#today_flag").removeAttr("style");
$("#today_flag").removeClass("block");
$("#today_flag").css("position", "absolute");
$("#today_flag").css("height", "170px");
$("#today_flag").css("bottom", "-35px");
$("#today_flag span").remove();
$(".block .label").css("text-align", "left");
$("#today_flag").css("left", flagPosition);
$("#today_flag").css("position", "relative");
$("#today_flag").prepend("
오늘
");
$("#today_flag").css("text-align", "center");
// 박스 위치 수정
versionData.forEach(function(version) {
var id = version.c_id;
var start = new Date(version.c_pds_version_start_date);
var startDate = start.getDate();
var daysCount = new Date(start.getFullYear(), start.getMonth() + 1, 0).getDate();
var end = new Date(version.c_pds_version_end_date);
var pos = $("#"+id).css("left");
var baseWidth =(parseFloat($(".month").css("width")))/daysCount;
var diffTime = Math.abs(end - start);
var diffDays = Math.ceil(diffTime / (1000 * 60 * 60 * 24));
var realWidth = baseWidth*diffDays
var realPos = parseFloat(pos)+ startDate*baseWidth;
$("#"+id).css("left",realPos+"px");
$("#"+id).css("width",realWidth+"px");
});
window.addEventListener("resize", $("#version-timeline-bar").width);
}
}
}
}
});
}
////////////////////
// 두번째 박스
////////////////////
async function drawVersionProgress(data) {
let gaugeChartColor = ColorPalette.d3Chart.gaugeChart;
if (gaugeChartColor.length < 0) {
gaugeChartColor = [
"rgba(158, 1, 66, 0.8)",
"rgba(213, 62, 79, 0.8)",
"rgba(244, 109, 67, 0.8)",
"rgba(253, 174, 97, 0.8)",
"rgba(254, 224, 139, 0.8)",
"rgba(230, 245, 152, 0.8)",
"rgba(171, 221, 164, 0.8)",
"rgba(102, 194, 165, 0.8)",
"rgba(50, 136, 189, 0.8)",
"rgba(94, 79, 162, 0.8)"
];
}
var Needle,
arc,
arcEndRad,
arcStartRad,
barWidth, // 색션의 두께
chart,
chartInset, // 가운데로 들어간 정도
el,
endPadRad,
height,
i,
margin, // 차트가 그려지는 위치 마진
needle, // 침
numSections, // 색션의 수
padRad,
percToDeg,
percToRad,
degToRad, // 고정
percent,
radius, // 반지름
ref,
sectionIndx, // 색션 인덱스
sectionPerc, // 색션의 퍼센트
startPadRad,
svg,
totalPercent,
width,
versionId,
versionName,
waveName;
// percent = 0.55;
barWidth = 25;
padRad = 0;
chartInset = 11;
totalPercent = 0.75;
margin = {
top: 0,
right: 0,
bottom: 0,
left: 0
};
width = 220;
height = width;
radius = Math.min(width, height) / 2.5;
// percToDeg percToRad degToRad 고정
percToDeg = function (perc) {
return perc * 360;
};
percToRad = function (perc) {
return degToRad(percToDeg(perc));
};
degToRad = function (deg) {
return (deg * Math.PI) / 180;
};
svg = d3
.select("#versionGaugeChart")
.append("svg")
.attr("viewBox", [70, 10, width - 150, height - 100])
.append("g");
chart = svg
.append("g")
.attr("transform", "translate(" + (width + margin.left) / 2 + ", " + (height + margin.top) / 2 + ")");
var tooltip = d3
.select("#versionGaugeChart")
.append("div")
.style("opacity", 0)
.attr("class", "tooltip")
.style("background-color", "white")
.style("border", "solid")
.style("border-width", "1px")
.style("border-radius", "5px")
.style("color", "black")
.style("padding", "10px");
var arc = d3
.arc()
.innerRadius(radius * 0.6)
.outerRadius(radius);
var outerArc = d3
.arc()
.innerRadius(radius * 0.9)
.outerRadius(radius * 0.9);
var totalDate;
numSections = data.length; // 전체 색션의 수(버전의 수)
sectionPerc = 1 / numSections / 2; // '/ 2' for Half-circle
var fastestStartDate;
var latestEndDate;
// 가장 빠른날짜, 가장 느린날짜 세팅
for (var idx = 0; idx < data.length; idx++) {
if (idx === 0) {
fastestStartDate = data[idx].start_date;
latestEndDate = data[idx].end_date;
} else {
if (data[idx].start_date < fastestStartDate) {
fastestStartDate = data[idx].start_date;
}
if (data[idx].end_date > latestEndDate) {
latestEndDate = data[idx].end_date;
}
}
}
globalDeadline = formatDate(new Date(latestEndDate));
console.log("[ analysisTime :: globalDeadline ] :: globalDeadline = " + globalDeadline);
$("#fastestStartDate").text(new Date(fastestStartDate).toLocaleDateString());
$("#latestEndDate").text(new Date(latestEndDate).toLocaleDateString());
const today = new Date(data[0].current_date);
today.setHours(0, 0, 0, 0); //시간, 분, 초, 밀리초를 0으로 설정하여 날짜만 비교
// 시작일과 종료일은 'YYYY-MM-DD' 형식의 문자열로 가정
const startDate = new Date(fastestStartDate);
startDate.setHours(0, 0, 0, 0); //시간, 분, 초, 밀리초를 0으로 설정하여 날짜만 비교
const endDate = new Date(latestEndDate);
endDate.setHours(0, 0, 0, 0); //시간, 분, 초, 밀리초를 0으로 설정하여 날짜만 비교
var diffStart = (today - startDate) / (1000 * 60 * 60 * 24); // 오늘 날짜와 시작일 사이의 차이를 일 단위로 계산
var diffEnd = (today - endDate) / (1000 * 60 * 60 * 24); // 오늘 날짜와 종료일 사이의 차이를 일 단위로 계산
$("#startDDay").css("color", "");
$("#endDDay").css("color", "");
if (diffStart > 0) {
$("#startDDay").text("D + " + diffStart);
} else if (diffStart === 0) {
$("#startDDay").text("D - day");
} else {
diffStart *= -1;
$("#startDDay").text("D - " + diffStart);
}
if (diffEnd > 0) {
$("#endDDay")
.css("color", "#FF4D4D")
.css("font-weight", "bold")
.text("D + " + diffEnd)
.append(" 초과");
} else if (diffEnd === 0) {
$("#endDDay").text("D - day");
} else {
diffEnd *= -1;
$("#endDDay").text("D - " + diffEnd);
}
totalDate = Math.floor(Math.abs((new Date(latestEndDate) - new Date(fastestStartDate)) / (1000 * 60 * 60 * 24)) + 1);
var mouseover = function (d) {
var hoverData = d;
var subgroupId = hoverData.version_id;
var subgroupName = hoverData.version_name;
var subgroupValue = new Date(hoverData.start_date).toLocaleDateString() + " ~ " + new Date(hoverData.end_date).toLocaleDateString();
tooltip.html("버전명: " + subgroupName + "
" + "기간: " + subgroupValue).style("opacity", 1);
d3.selectAll(".myWave").style("opacity", 0.2);
d3.selectAll(".myStr").style("opacity", 0.2);
d3.selectAll(".wave-" + subgroupId).style("opacity", 1);
};
var mousemove = function (d) {
var [x, y] = d3.mouse(this);
tooltip.style("left", (x + 120) + "px").style("top", (y + 150) + "px");
};
var mouseleave = function (d) {
tooltip.style("opacity", 0);
d3.selectAll(".myStr").style("opacity", 1);
d3.selectAll(".myWave").style("opacity", 1);
};
for (sectionIndx = i = 1, ref = numSections; 1 <= ref ? i <= ref : i >= ref; sectionIndx = 1 <= ref ? ++i : --i) {
arcStartRad = percToRad(totalPercent);
arcEndRad = arcStartRad + percToRad(sectionPerc);
totalPercent += sectionPerc;
startPadRad = sectionIndx === 0 ? 0 : padRad / 2;
endPadRad = sectionIndx === numSections ? 0 : padRad / 2;
versionId = data[sectionIndx - 1].version_id;
versionName = data[sectionIndx - 1].version_name;
var sectionData = data[sectionIndx - 1];
var arc = d3
.arc()
.outerRadius(radius - chartInset)
.innerRadius(radius - chartInset - barWidth)
.startAngle(arcStartRad + startPadRad)
.endAngle(arcEndRad - endPadRad);
var section = chart.selectAll(".arc.chart-color" + sectionIndx + ".myWave.wave-" + versionId);
section
.data([sectionData])
.enter()
.append("g")
.attr("class", "arc chart-color" + sectionIndx + " myWave wave-" + versionId)
.on("mouseover", mouseover)
.on("mousemove", mousemove)
.on("mouseleave", mouseleave)
.append("path")
.attr("fill", function (d) {
return gaugeChartColor[(sectionIndx - 1) % gaugeChartColor.length];
})
.attr("stroke", "white")
.style("stroke-width", "0.4px")
.attr("d", arc);
chart
.selectAll(".arc.chart-color" + sectionIndx + ".myWave.wave-" + versionId)
.append("text")
.attr("class", "no-select")
.text(function (d) {
return getStrLimit(d.version_name, 9);
})
.attr("x", function (d) {
return arc.centroid(d)[0];
})
.attr("y", function (d) {
return arc.centroid(d)[1] + 2;
})
.style("font-size", "10px")
.style("font-weight", "700")
.attr("text-anchor", "middle");
}
Needle = (function () {
function Needle(len, radius1) {
this.len = len;
this.radius = radius1;
}
Needle.prototype.drawOn = function (el, perc) {
el.append("circle")
.attr("class", "needle-center")
.attr("cx", 0)
.attr("cy", -10)
.attr("r", this.radius)
.attr("stroke", "white")
.style("stroke-width", "0.3px");
return el
.append("path")
.attr("class", "needle")
.attr("d", this.mkCmd(perc))
.attr("stroke", "white")
.style("stroke-width", "0.3px");
};
Needle.prototype.animateOn = function (el, perc) {
var self;
self = this;
return el
.selectAll(".needle")
.transition()
.delay(500)
.ease(d3.easeElasticOut)
.duration(3000)
.attrTween("progress", function () {
return function (percentOfPercent) {
var progress;
progress = percentOfPercent * perc;
return d3.select(".needle").attr("d", self.mkCmd(progress));
};
});
};
Needle.prototype.mkCmd = function (perc) {
var centerX, centerY, leftX, leftY, rightX, rightY, thetaRad, topX, topY;
thetaRad = percToRad(perc / 2);
centerX = 0;
centerY = -10;
topX = centerX - this.len * Math.cos(thetaRad);
topY = centerY - this.len * Math.sin(thetaRad);
leftX = centerX - this.radius * Math.cos(thetaRad - Math.PI / 2);
leftY = centerY - this.radius * Math.sin(thetaRad - Math.PI / 2);
rightX = centerX - this.radius * Math.cos(thetaRad + Math.PI / 2);
rightY = centerY - this.radius * Math.sin(thetaRad + Math.PI / 2);
return "M " + leftX + " " + leftY + " L " + topX + " " + topY + " L " + rightX + " " + rightY;
};
return Needle;
})();
needle = new Needle(35, 3);
needle.drawOn(chart, 0);
var needleAngle = (diffStart + 1) / totalDate;
if (needleAngle > 1) {
needleAngle = 1;
} else if (needleAngle < 0) {
needleAngle = 0;
}
needle.animateOn(chart, needleAngle);
}
////////////////////
// 히트맵 차트
////////////////////
function calendarHeatMap(pdServiceLink, pdServiceVersions) {
$("#calendar_yearview_blocks_chart_1 svg").remove();
$("#calendar_yearview_blocks_chart_2 svg").remove();
$.ajax({
url: "/auth-admin/api/arms/analysis/time/heatmap",
type: "GET",
data: { pdServiceLink: pdServiceLink, pdServiceVersionLinks: pdServiceVersions },
contentType: "application/json;charset=UTF-8",
dataType: "json",
progress: true,
async: false,
statusCode: {
200: function (data) {
console.log("[ analysisTime :: calendarHeatMap ] :: 누적 업데이트 히트맵 차트데이터 = ");
console.log(data);
$(".update-title").show();
$("#calendar_yearview_blocks_chart_1").calendar_yearview_blocks({
data: JSON.stringify(data.requirement),
start_monday: true,
always_show_tooltip: true,
month_names: ["jan", "feb", "mar", "apr", "may", "jun", "jul", "aug", "sept", "oct", "nov", "dec"],
day_names: ["mon", "wed", "fri", "sun"]
//colors: data.requirementColors
});
$("#calendar_yearview_blocks_chart_2").calendar_yearview_blocks({
data: JSON.stringify(data.relationIssue),
start_monday: true,
always_show_tooltip: true,
month_names: ["jan", "feb", "mar", "apr", "may", "jun", "jul", "aug", "sep", "oct", "nov", "dec"],
day_names: ["mon", "wed", "fri", "sun"]
//colors: data.relationIssueColors
});
// d3.select("#heatmap-body").style("overflow-x","scroll");
}
}
});
}
////////////////////
// 스캐터 차트
////////////////////
async function dailyUpdatedStatusScatterChart(pdServiceLink, pdServiceVersionLinks) {
let deadline = await waitForGlobalDeadline();
let startDate = $("#scatter_start_date").val();
let endDate = $("#scatter_end_date").val();
if (!validateSearchDateWithChart(startDate, endDate)) {
return;
}
const url = new UrlBuilder()
.setBaseUrl("/auth-admin/api/arms/analysis/time/standard-daily/jira-issue")
.addQueryParam("pdServiceLink", pdServiceLink)
.addQueryParam("pdServiceVersionLinks", pdServiceVersionLinks)
.addQueryParam("일자기준", "updated")
.addQueryParam("메인_그룹_필드", "isReq")
.addQueryParam("시작일", startDate)
.addQueryParam("종료일", endDate)
.addQueryParam("크기", 1000)
.addQueryParam("하위_크기", 1000)
.addQueryParam("컨텐츠_보기_여부", true)
.build();
$(".spinner").html(
' ' +
"일별 업데이트 상태 차트를 로딩 중입니다..."
);
$.ajax({
url: url,
type: "GET",
contentType: "application/json;charset=UTF-8",
dataType: "json",
progress: true,
async: false,
statusCode: {
200: function (data) {
console.log("[ analysisTime :: dailyUpdatedStatusScatterChart ] :: 일별 업데이트 상태 스캐터 차트데이터 = ");
console.log(data);
let result = Object.keys(data).reduce(
(acc, date) => {
if (data[date].totalRequirements !== 0 || data[date].totalRelationIssues !== 0) {
acc.dates.push(date);
acc.totalRequirements.push(data[date].totalRequirements);
acc.totalRelationIssues.push(data[date].totalRelationIssues);
}
return acc;
},
{
dates: [],
totalRelationIssues: [],
totalRequirements: [],
}
);
let dates = result.dates;
let totalRelationIssues = result.totalRelationIssues;
let totalRequirements = result.totalRequirements;
let deadlineSeries = createDeadlineSeries(dates, totalRequirements, totalRelationIssues, globalDeadline, false, 2);
var dom = document.getElementById("scatter-chart-container");
var myChart = echarts.init(dom, "dark", {
renderer: "canvas",
useDirtyRect: false
});
var option;
if ((totalRequirements && totalRequirements.length > 0) || (totalRelationIssues && totalRelationIssues.length > 0)) {
option = {
aria: {
show: true
},
legend: {
data: ["요구사항", "연결된 이슈"],
textStyle: {
color: "white"
}
},
xAxis: {
type: "category",
axisTick: { show: false },
data: dates,
axisLabel: {
textStyle: {
color: "white"
}
}
},
yAxis: {
type: "value",
splitLine: {
show: true,
lineStyle: {
color: "rgba(255,255,255,0.2)",
width: 1,
type: "dashed"
}
},
axisLabel: {
textStyle: {
color: "white"
}
}
},
series: [
{
name: "요구사항",
data: totalRequirements,
type: "scatter",
symbol: "diamond",
clip: false,
label: {
normal: {
show: false,
color: "#FFFFFF"
},
emphasis: {
show: true,
color: "#FFFFFF"
}
},
symbolSize: function (val) {
var sbSize = 10;
if (val > 10) {
sbSize = val * 1.1;
} else if (val === 0) {
sbSize = 0;
}
return sbSize;
}
},
{
name: "연결된 이슈",
data: totalRelationIssues,
type: "scatter",
clip: false,
label: {
normal: {
show: false,
color: "#FFFFFF"
},
emphasis: {
show: true,
color: "#FFFFFF"
}
},
symbolSize: function (val) {
var sbSize = 10;
if (val > 10) {
sbSize = val * 1.1;
} else if (val === 0) {
sbSize = 0;
}
return sbSize;
},
itemStyle: {
color: "#13de57"
}
},
...deadlineSeries
],
tooltip: {
trigger: "axis",
position: "top",
borderWidth: 1,
axisPointer: {
type: "line",
label: {
formatter: function (params) {
return formatDate(new Date(params.value));
}
}
}
},
backgroundColor: "rgba(255,255,255,0)",
animationDelay: function (idx) {
return idx * 20;
},
animationDelayUpdate: function (idx) {
return idx * 20;
}
};
myChart.on("click", function (params) {
// console.log(params.data);
});
myChart.on("mouseover", function (params) {
// if (params.seriesType === 'line') {
var option = myChart.getOption();
option.series[params.seriesIndex].label.color = 'white';
option.series[params.seriesIndex].label.show = true;
myChart.setOption(option);
// }
});
myChart.on("mouseout", function (params) {
// if (params.seriesType === 'line') {
var option = myChart.getOption();
option.series[params.seriesIndex].label.show = false;
myChart.setOption(option);
// }
});
}
else {
option = {
title: {
text: "데이터가 없습니다",
left: "center",
top: "middle",
textStyle: {
color: "#fff"
}
},
backgroundColor: "rgba(255,255,255,0)"
};
}
if (option && typeof option === "object") {
myChart.setOption(option, true);
}
window.addEventListener("resize", myChart.resize);
}
}
});
}
////////////////
// 멀티 콤비네이션 차트
///////////////
async function dailyCreatedCountAndUpdatedStatusesMultiStackCombinationChart(pdServiceLink, pdServiceVersionLinks) {
let deadline = await waitForGlobalDeadline();
let startDate = $("#multi_stack_start_date").val();
let endDate = $("#multi_stack_end_date").val();
if (!validateSearchDateWithChart(startDate, endDate)) {
return;
}
const url = new UrlBuilder()
.setBaseUrl("/auth-admin/api/arms/analysis/time/standard-daily/jira-issue")
.addQueryParam("pdServiceLink", pdServiceLink)
.addQueryParam("pdServiceVersionLinks", pdServiceVersionLinks)
.addQueryParam("일자기준", "updated")
.addQueryParam("메인_그룹_필드", "isReq")
.addQueryParam("하위_그룹_필드들", "status.status_name.keyword")
.addQueryParam("시작일", startDate)
.addQueryParam("종료일", endDate)
.addQueryParam("크기", 1000)
.addQueryParam("하위_크기", 1000)
.addQueryParam("컨텐츠_보기_여부", true)
.build();
$(".spinner").html(
' ' +
"생성 개수 및 업데이트 상태 현황 정보를 로딩 중입니다..."
);
$.ajax({
url: url,
type: "GET",
contentType: "application/json;charset=UTF-8",
dataType: "json",
progress: true,
async: false,
statusCode: {
200: function (data) {
console.log("[ analysisTime :: dailyCreatedCountAndUpdatedStatusesMultiStackCombinationChart ] :: 일별 이슈 생성 개수 및 업데이트 현황 데이터 = ");
console.log(data);
var accumulateRequirementCount = 0;
var accumulateRelationIssueCount = 0;
let result = Object.keys(data).reduce(
(acc, date) => {
if (data[date].totalRequirements !== 0 || data[date].totalRelationIssues !== 0) {
acc.dates.push(date);
accumulateRequirementCount += data[date].totalRequirements;
accumulateRelationIssueCount += data[date].totalRelationIssues;
acc.totalRequirements.push(accumulateRequirementCount);
acc.totalRelationIssues.push(accumulateRelationIssueCount);
}
if (data[date].requirementStatuses !== null) {
Object.keys(data[date].requirementStatuses).forEach((status) => {
if (!acc.statusKeys.includes(status)) {
acc.statusKeys.push(status);
}
});
}
if (data[date].relationIssueStatuses !== null) {
Object.keys(data[date].relationIssueStatuses).forEach((status) => {
if (!acc.statusKeys.includes(status)) {
acc.statusKeys.push(status);
}
});
}
return acc;
},
{
dates: [],
totalRelationIssues: [],
totalRequirements: [],
statusKeys: []
}
);
var dom = document.getElementById("multi-chart-container");
var myChart = echarts.init(dom, null, {
renderer: "canvas",
useDirtyRect: false
});
var option;
if (result.dates.length > 0) {
var labelOption = {
show: false,
position: "top",
distance: 0,
align: "center",
verticalAlign: "top",
rotate: 0,
formatter: "{c}",
fontSize: 14,
rich: {
name: {}
}
};
let dates = result.dates;
let totalRelationIssues = result.totalRelationIssues;
let totalRequirements = result.totalRequirements;
let statusKeys = result.statusKeys;
let deadlineSeries = createDeadlineSeries(dates, totalRequirements, totalRelationIssues, globalDeadline, true, 4);
let requirementStatusSeries = statusKeys.map((key, i) => {
let stackType = "요구사항";
return {
name: key,
type: "bar",
stack: stackType,
label: labelOption,
emphasis: {
focus: "series"
},
data: dates.map((date) => {
if (data[date] && data[date].requirementStatuses && Object.keys(data[date].requirementStatuses).length > 0) {
return { value: data[date].requirementStatuses[key] || 0, stackType: stackType };
} else {
return { value: 0, stackType: stackType };
}
})
};
});
let relationIssueStatusSeries = statusKeys.map((key, i) => {
let stackType = "연결된 이슈";
return {
name: key,
type: "bar",
stack: stackType,
label: labelOption,
emphasis: {
focus: "series"
},
data: dates.map((date) => {
if (data[date] && data[date].relationIssueStatuses && Object.keys(data[date].relationIssueStatuses).length > 0) {
return { value: data[date].relationIssueStatuses[key] || 0, stackType: stackType };
} else {
return { value: 0, stackType: stackType };
}
})
};
});
// let stackIndex = statusKeys.map((value, index) => index);
statusKeys.push("요구사항");
statusKeys.push("연결된 이슈");
let multiCombinationChartSeries = [
...requirementStatusSeries,
...relationIssueStatusSeries,
{
name: "요구사항",
type: "line",
// yAxisIndex: 1,
emphasis: {
focus: "series"
},
symbolSize: 10,
data: totalRequirements
},
{
name: "연결된 이슈",
type: "line",
// yAxisIndex: 1,
emphasis: {
focus: "series"
},
symbolSize: 10,
data: totalRelationIssues
},
...deadlineSeries
];
var legendData = statusKeys;
var xAiasData = dates;
option = {
tooltip: {
trigger: "axis",
axisPointer: {
type: "shadow"
},
formatter: function(params) {
var tooltipText = "";
tooltipText += params[0].axisValue + "
";
params.forEach(function(item) {
if (item.value !== 0) {
// 0인 데이터는 무시
if (item.seriesType === "bar") {
var stackType = item.data.stackType; // 추가 정보에 접근
tooltipText +=
item.marker +
item.seriesName +
"[" +
stackType +
"]" +
' ' +
item.value +
"" +
"
";
} else if (item.seriesType === "line") {
tooltipText +=
item.marker +
item.seriesName +
' ' +
item.value +
"" +
"
";
}
}
});
return tooltipText;
}
},
legend: {
data: legendData,
textStyle: {
color: "white"
},
tooltip: {
show: true
}
},
grid: {
top: "20%",
containLabel: false
},
toolbox: {
show: true,
orient: "vertical",
left: "right",
bottom: "50px",
feature: {
mark: { show: true },
// dataView: {show: true, readOnly: true},
/*magicType: {
show: true,
type: ['stack'],
seriesIndex: {
stack: stackIndex
}
},*/
dataZoom: {
show: true
}
// restore: { show: true },
//saveAsImage: { show: true }
// myTool: {
// show: true,
// title: '상태 그룹화',
// icon: 'image://http://echarts.baidu.com/images/favicon.png',
// onclick: toggleStack
// },
},
iconStyle: {
borderColor: "white"
}
},
xAxis: [
{
type: "category",
axisTick: { show: false },
data: xAiasData,
axisLabel: {
textStyle: {
color: "white"
}
}
}
],
yAxis: [
{
type: "value",
axisLabel: {
textStyle: {
color: "white"
}
},
splitLine: {
show: true,
lineStyle: {
color: "rgba(255,255,255,0.2)",
width: 1,
type: "dashed"
}
}
},
{
type: "value",
position: "right",
axisLabel: {
textStyle: {
color: "white"
}
}
}
],
series: multiCombinationChartSeries,
backgroundColor: "rgba(255,255,255,0)",
animationDelay: function(idx) {
return idx * 20;
},
animationDelayUpdate: function(idx) {
return idx * 20;
}
};
} else {
option = {
title: {
text: "데이터가 없습니다",
left: "center",
top: "middle",
textStyle: {
color: "#fff" // 제목 색상을 검은색으로 변경
}
},
backgroundColor: "rgba(255,255,255,0)"
};
}
if (option && typeof option === "object") {
myChart.setOption(option, true);
}
window.addEventListener("resize", function() {
myChart.resize();
});
myChart.on("mouseover", function(params) {
var option = myChart.getOption();
option.series[params.seriesIndex].label.show = true;
myChart.setOption(option);
});
myChart.on("mouseout", function(params) {
var option = myChart.getOption();
option.series[params.seriesIndex].label.show = false;
myChart.setOption(option);
});
}
}
});
}
// 마감일 함수
function createDeadlineSeries(dates, totalRelationIssues, totalRequirements, deadline, usePreviousValue, lineWidth) {
var chartStart = dates.reduce((earliest, date) => (date < earliest ? date : earliest), dates[0]);
var chartEnd = dates.reduce((latest, date) => (date > latest ? date : latest), dates[0]);
chartStart = new Date(chartStart);
chartEnd = new Date(chartEnd);
var deadlineSeries = [];
if (new Date(deadline) <= chartEnd) {
if (!dates.includes(deadline)) {
dates.push(deadline);
dates.sort((a, b) => new Date(a) - new Date(b));
let dateIndex = dates.indexOf(deadline);
if (dateIndex > 0 && usePreviousValue) {
totalRequirements.splice(dateIndex, 0, totalRequirements[dateIndex-1]);
totalRelationIssues.splice(dateIndex, 0, totalRelationIssues[dateIndex-1]);
}
else {
totalRequirements.splice(dateIndex, 0, 0);
totalRelationIssues.splice(dateIndex, 0, 0);
}
}
// 데이터 추가
var vs = {
name: "마감일",
type: "line",
data: [
[deadline, 0],
[deadline, 1]
],
tooltip: {
show: false
},
markLine: {
silent: true,
symbol: "none",
data: [
{
xAxis: deadline
}
],
lineStyle: {
color: "red",
width: lineWidth,
type: "dashed"
},
label: {
formatter: "마감일 : {c}",
color: "white",
fontSize: 14,
fontWeight: "bold"
}
},
lineStyle: {
color: "red",
type: "dashed"
},
symbol: "none"
};
deadlineSeries.push(vs);
}
return deadlineSeries;
}
function dailyChartDataSearchEvent() {
$("#scatter-search").on("click", function (params) {
dailyUpdatedStatusScatterChart(selectedPdServiceId, selectedVersionId);
});
$("#multi-stack-search").on("click", function (params) {
dailyCreatedCountAndUpdatedStatusesMultiStackCombinationChart(selectedPdServiceId, selectedVersionId);
});
$("#timeline-search").on("click", function (params) {
timeLineChart(selectedPdServiceId, selectedVersionId);
});
}
function validateSearchDateWithChart(startDate, endDate) {
let result = true;
if(!selectedPdServiceId || !selectedVersionId) {
alert("제품(서비스) 혹은 버전 선택이 되지 않았습니다.");
result = false;
}
if (!startDate || !endDate) {
alert("일자를 지정하지 않았습니다.");
result = false;
}
return result;
}
function onScatterChartDateEndChanged() {
dailyUpdatedStatusScatterChart(selectedPdServiceId, selectedVersionId);
}
function onMultiStackChartDateEndChanged() {
dailyCreatedCountAndUpdatedStatusesMultiStackCombinationChart(selectedPdServiceId, selectedVersionId);
}
function convertVersionIdToTitle(versionId) {
if (versionListData.hasOwnProperty(versionId)) {
var version = versionListData[versionId];
return version.c_title;
}
}
function verticalTimeLineChart(data) {
let contentSet = {}; // 객체로 선언
let items = Object.values(data).reduce((acc, versionData) => {
versionData.forEach(item => {
if (!contentSet[item.summary]) { // 중복 체크
contentSet[item.summary] = {
version: item.pdServiceVersion,
summary: item.summary,
projectName: [item.project.project_name],
date: formatDateTime(item.updated)
};
} else {
// projectName에 item.project.project_name이 없는 경우에만 추가
if (!contentSet[item.summary].projectName.includes(item.project.project_name)) {
contentSet[item.summary].projectName.push(item.project.project_name);
contentSet[item.summary].projectName.sort();
}
}
});
return acc;
}, []);
items = Object.values(contentSet).map(item => ({
...item,
projectName: item.projectName
}));
// 날짜를 기준으로 오름차순 정렬
items.sort((a, b) => new Date(b.date) - new Date(a.date));
makeVerticalTimeline(items);
// mock data
/*
makeVerticalTimeline([
{
title: "BaseVersion",
content: "요구 사항 이슈 1",
type: "Presentation",
date: "2023-11-08"
},
{
title: "",
content: "요구 사항 이슈 2",
type: "Presentation",
date: "2023-11-29"
},
{
title: "1.0",
content: "요구 사항 이슈 3",
type: "Review",
date: "2023-12-01"
},
{
title: "1.1",
content: "요구 사항 이슈 4",
type: "Review",
date: "2023-12-11"
}
]);
*/
}
function makeVerticalTimeline(data) {
// 데이터 세팅
const $container = $(".timeline-container");
$container.empty();
if (data.length == 0) {
const noDataMessage = $('').text('데이터가 없습니다.').css({
position: 'absolute',
top: '50%',
left: '50%',
transform: 'translate(-50%, -50%)'
});
$container.append(noDataMessage);
} else {
// 날짜별로 데이터 그룹화
let groupedData = data.reduce((group, item) => {
let date = item.date;
if (!group[date]) group[date] = [];
group[date].push(item);
return group;
}, {});
const $ul = $('');
Object.entries(groupedData).forEach(([date, items]) => {
items.forEach(({ version, summary, projectName }, index) => {
const $li = $('').addClass('session');
if (index === 0) {
$li.append(`
${date}
`);
}
const $sessionContent = $(`
${convertVersionIdToTitle(version)}
${summary}
`);
const $projectNameDiv = $('').addClass('project-names');
// projectName 배열의 각 요소를 추가
projectName.forEach(name => {
const $button = $('').addClass('project-name').text(name);
$projectNameDiv.append($button);
});
$sessionContent.append($projectNameDiv);
$li.append($sessionContent);
$ul.append($li);
});
});
$container.append($ul);
}
adjustHeight();
}
function formatDateTime(dateTime) {
var date = dateTime.split('T')[0];
return date;
}
async function timeLineChart(pdServiceLink, pdServiceVersionLinks) {
let deadline = await waitForGlobalDeadline();
let startDate = $("#timeline_start_date").val();
let endDate = $("#timeline_end_date").val();
if (!validateSearchDateWithChart(startDate, endDate)) {
return;
}
const verticalUrl = new UrlBuilder()
.setBaseUrl("/auth-admin/api/arms/analysis/time/standard-daily/updated-jira-issue")
.addQueryParam("pdServiceLink", pdServiceLink)
.addQueryParam("pdServiceVersionLinks", pdServiceVersionLinks)
.addQueryParam("일자기준", "updated")
.addQueryParam("isReqType", "REQUIREMENT")
.addQueryParam("시작일", startDate)
.addQueryParam("종료일", endDate)
.addQueryParam("크기", 1000)
.addQueryParam("하위_크기", 1000)
.addQueryParam("컨텐츠_보기_여부", true)
.build();
$(".spinner").html(
' ' +
"수직 타임라인 차트를 로딩 중입니다..."
);
$.ajax({
url: verticalUrl,
type: "GET",
contentType: "application/json;charset=UTF-8",
dataType: "json",
progress: true,
statusCode: {
200: function (data) {
console.log("[ analysisTime :: TimeLineData ] :: = ");
console.log(data);
verticalTimeLineChart(data);
}
}
});
const ridgeLineUrl = new UrlBuilder()
.setBaseUrl("/auth-admin/api/arms/analysis/time/standard-daily/updated-ridgeline")
.addQueryParam("pdServiceLink", pdServiceLink)
.addQueryParam("pdServiceVersionLinks", pdServiceVersionLinks)
.addQueryParam("일자기준", "updated")
.addQueryParam("isReqType", "ISSUE")
.addQueryParam("시작일", startDate)
.addQueryParam("종료일", endDate)
.addQueryParam("크기", 1000)
.addQueryParam("하위_크기", 1000)
.addQueryParam("컨텐츠_보기_여부", true)
.build();
function executeAjaxCall(url) {
$(".spinner").html(
' ' +
"요구사항 업데이트 현황(능선차트)를 로딩 중입니다..."
);
$.ajax({
url: url,
type: "GET",
contentType: "application/json;charset=UTF-8",
dataType: "json",
progress: true,
statusCode: {
200: function (data) {
console.log("[ analysisTime :: ridgeLineData ] :: = ");
console.log(data);
updateRidgeLine(data);
}
}
});
}
executeAjaxCall(ridgeLineUrl);
window.addEventListener("resize", function() {
adjustHeight();
});
}
function getColorByVersion(version) {
var colorPalette = [ //e chart 컬러 팔레트
'#c23531','#2f4554','#61a0a8','#d48265','#91c7ae',
'#749f83','#ca8622','#bda29a','#6e7074','#546570',
'#c4ccd3'
];
var versionNumber = parseInt(version);
return colorPalette[versionNumber % colorPalette.length];
}
function updateRidgeLine(traffic){
// 데이터가 없을 경우
if (!traffic || traffic.length === 0) {
document.getElementById("overlapInputDiv").style.display = "none";
document.getElementById("updateRidgeLine").innerHTML = "" +
"데이터가 없습니다.
";
return;
} else {
document.getElementById("overlapInputDiv").style.display = "flex";
}
function setOverlapInputListener() {
var overlap = this.value;
overlapNumberInput.value = overlap;
drawGraph(traffic, overlap);
}
function setOverlapNumberInputListener() {
var overlap = this.value;
overlapInput.value = overlap;
drawGraph(traffic, overlap);
}
overlapInput.removeEventListener('input', setOverlapInputListener);
overlapNumberInput.removeEventListener('input', setOverlapNumberInputListener);
overlapInput.addEventListener('input', setOverlapInputListener);
overlapNumberInput.addEventListener('input', setOverlapNumberInputListener);
var initialOverlap = traffic.length > 30 ? 5 : 2;
document.getElementById("overlapInput").value = initialOverlap;
document.getElementById("overlapNumberInput").value = initialOverlap;
drawGraph(traffic, initialOverlap);
}
function drawGraph(traffic, overlap){
document.getElementById("updateRidgeLine").innerHTML = "";
var nestedDataByDate = d3.nest()
.key(function(d) { return +new Date(d.date); })
.entries(traffic);
var dates = nestedDataByDate.map(function(d) { return +d.key; }).sort(d3.ascending);
var nestedDataByName = d3.nest()
.key(function(d) { return d.name; })
.entries(traffic);
var series = nestedDataByName.map(function(d) {
var valuesMap = d3.map(d.values, function(e) { return String(+new Date(e.date)); });
var values = dates.map(function(date) {
var valueObj = valuesMap.get(String(date));
return valueObj ? valueObj.value : null;
});
var version = d.values[0] ? d.values[0].version : null; // version 필드 추가
var summary = d.values[0] ? d.values[0].summary : null; // version 필드 추가
var key = d.values[0] ? d.values[0].name : null; // version 필드 추가
return { name: key+": "+summary, values: values, version: version ,key:key}; // version 값 포함하여 반환
});
//const overlap = 4;
const width = 900;
//const height = series.length * 30;
const minHeight = 600;
const maxHeight = 650;
const height = Math.max(minHeight, Math.min(maxHeight, series.length * 16));
const marginTop = 50;
const marginRight = 0;
const marginBottom = 0;
const marginLeft = 280;
// Create the scales.
const x = d3.scaleTime()
.domain(d3.extent(dates))
.range([marginLeft, width - marginRight]);
const y = d3.scalePoint()
.domain(series.map(d => d.name))
.range([marginTop, height - marginBottom]);
const z = d3.scaleLinear()
.domain([0, d3.max(series, d => d3.max(d.values))]).nice()
.range([0, -overlap * y.step()]);
// Create the area generator and its top-line generator.
const area = d3.area()
.curve(d3.curveBasis)
.defined(d => !isNaN(d))
.x((d, i) => x(dates[i]))
.y0(0)
.y1(d => z(d));
const line = area.lineY1();
// Create the SVG container.
const svg = d3.create("svg")
.attr("width", width)
.attr("height", maxHeight)
.attr("viewBox", [0, 0, width, height])
.attr("style", "max-width: 100%; height: auto;");
// Append the axes.
svg.append("g")
.attr("transform", `translate(0,${height - marginBottom})`)
.call(d3.axisBottom(x)
.ticks(width / 80)
.tickSizeOuter(0));
svg.append("g")
.attr("transform", `translate(${marginLeft},0)`)
.call(d3.axisLeft(y).tickSize(0).tickPadding(4))
.call(g => g.select(".domain").remove())
.selectAll(".tick text")
.text(function(d) {
return d.length > 42 ? d.slice(0, 35) + ' . . .' : d; // 긴 레이블은 축약
})
.style("font-size", "10px");
// Append a layer for each series.
const group = svg.append("g")
.selectAll("g")
.data(series)
.join("g")
.attr("transform", d => `translate(0,${y(d.name) + 1})`);
var div = d3.select("body").append("div")
.attr("class", "tooltip")
.style("opacity", 0);
group.append("path")
.attr("fill", d => getColorByVersion(d.version))
.attr("d", d => area(d.values))
.on("mouseover", function(d) {
var event = d3.event;
d3.select(this)
.transition()
.duration(20)
.style("opacity", 0.4);
div.transition()
.duration(20)
.style("opacity", .9);
div.html("버전 정보: " + convertVersionIdToTitle(d.version) +
"
요구사항 키: " + d.key +
"
요구사항 제목: " + d.name)
.style("left", (event.pageX) + "px")
.style("top", (event.pageY - 28) + "px");
})
.on("mouseout", function() {
d3.select(this)
.transition()
.duration(20)
.style("opacity", 1);
div.transition()
.duration(20)
.style("opacity", 0);
});
group.append("path")
.attr("fill", "none")
.attr("stroke","#EBEDF0") //.attr("stroke", d => getColorByVersion(d.version))
.attr("stroke-width", 0.5)
.attr("d", d => line(d.values));
$("#overlapInputDiv").css("display", "flex");
$('#updateRidgeLine').append(svg.node());
adjustHeight();
}
// 차트 높이 조정
function adjustHeight() {
var verticalTimeline = $('#vertical-timeline');
var updateRidgeLine = $('#updateRidgeLine');
if (verticalTimeline && updateRidgeLine) {
verticalTimeline.height(updateRidgeLine.height() + 20);
}
}
// 주식차트
function candleStickChart() {
var dom = document.getElementById("candlestick-chart-container");
var myChart = echarts.init(dom, "dark", {
renderer: "canvas",
useDirtyRect: false
});
var option;
option = {
xAxis: {
data: ["2017-10-24", "2017-10-25", "2017-10-26", "2017-10-27"]
},
yAxis: {},
series: [
{
type: "candlestick",
data: [
[20, 34, 10, 38],
[40, 35, 30, 50],
[31, 38, 33, 44],
[38, 15, 5, 42]
]
}
],
tooltip: {
trigger: "axis",
position: "top",
borderWidth: 1,
axisPointer: {
type: "cross"
}
},
backgroundColor: "rgba(255,255,255,0)"
};
if (option && typeof option === "object") {
myChart.setOption(option, true);
}
window.addEventListener("resize", myChart.resize);
}
function versionTimelineChart(versionData) {
var yVersionData = [];
var xVesrionStartEndData = [];
var yearData = new Set();
versionData.forEach(version => {
yVersionData.push(version.title);
var arrayData = [version.title, +new Date(version.startDate), +new Date(version.endDate)];
yearData.add(new Date(version.startDate).getFullYear());
yearData.add(new Date(version.endDate).getFullYear());
xVesrionStartEndData.push(arrayData);
});
var dom = document.getElementById('version-timeline-chart-container');
var myChart = echarts.init(dom, null, {
renderer: 'canvas',
useDirtyRect: false
});
var colorList = ['#5470C6', '#91CC75', '#FAC858', '#EE6666', '#73C0DE', '#3BA272', '#FC8452', '#9A60B4', '#EA7CCC'];
var versionData = ['v1.0', 'v1.1', 'v1.2', 'v1.3', 'v1.4', 'v1.5', 'v1.6', 'v1.7', 'v1.8', 'v1.9'];
var startEndData= [
['v1.0', +new Date(2023, 0, 1), +new Date(2023, 0, 15)],
['v1.1', +new Date(2023, 0, 10), +new Date(2023, 0, 25)],
['v1.2', +new Date(2023, 1, 1), +new Date(2023, 1, 15)],
['v1.3', +new Date(2023, 3, 1), +new Date(2023, 4, 15)],
['v1.4', +new Date(2023, 2, 1), +new Date(2024, 3, 15)],
['v1.5', +new Date(2023, 6, 1), +new Date(2023, 6, 15)],
['v1.6', +new Date(2023, 5, 1), +new Date(2023, 11, 15)],
['v1.7', +new Date(2024, 2, 1), +new Date(2024, 3, 15)],
['v1.8', +new Date(2024, 5, 1), +new Date(2024, 6, 15)],
['v1.9', +new Date(2024, 1, 1), +new Date(2024, 11, 15)],
];
var today = new Date();
var todayLine = {
name: '오늘',
type: 'line',
data: [[formatDate(today), 0], [formatDate(today), 1]], // y축 전체에 걸쳐 라인을 그립니다.
tooltip: {
show: false
},
markLine : {
silent: true,
symbol: 'none',
data : [{
xAxis : formatDate(today)
}],
lineStyle: {
color: 'red',
width: 2,
type: 'dashed'
},
label: {
formatter: '오늘 : {c}',
color: 'white',
fontSize: 12,
fontWeight: 'bold',
position: 'start'
}
},
lineStyle: {
color: 'red',
type: 'dashed'
},
symbol: 'none'
};
let yearsArray = Array.from(yearData);
let minYear = Math.min(...yearsArray);
let maxYear = Math.max(...yearsArray);
if (maxYear.valueOf() < today.getFullYear()) {
maxYear = today.getFullYear();
}
var option = {
xAxis: {
type: 'time',
min: minYear + '-01-01',
max: maxYear + '-12-31',
axisLabel: {
textStyle: {
color: "white"
},
rotate: 45
},
axisTick: { show: false },
splitLine: {
show: true,
lineStyle: {
color: "rgba(255,255,255,0.2)",
width: 1,
type: "dashed"
}
},
},
yAxis: {
data: yVersionData,
inverse: true,
axisLabel: {
textStyle: {
color: "white"
}
}
},
series: [
{
name: 'Versions',
type: 'custom',
itemStyle: {
color: function(params) {
return colorList[params.dataIndex % colorList.length];
}
},
renderItem: function(params, api) {
var categoryIndex = api.value(0);
var start = api.coord([api.value(1), categoryIndex]);
var end = api.coord([api.value(2), categoryIndex]);
var height = params.coordSys.height / yVersionData.length;
return {
type: 'rect',
shape: {
x: start[0],
y: start[1] - height / 2,
width: end[0] - start[0],
height: height
},
style: api.style(params.dataIndex) // apply color here
};
},
encode: {
x: [1, 2],
y: 0
},
data: xVesrionStartEndData
},
todayLine,
],
tooltip: {
trigger: 'axis',
position: 'top',
borderWidth: 1,
axisPointer: {
type: 'cross',
axis: 'y'
},
formatter: function (params) {
var tooltipText = '';
tooltipText += params[0].marker + params[0].data[0] + '
' + new Date(params[0].data[1]).toLocaleDateString()+ " ~ " + new Date(params[0].data[2]).toLocaleDateString() + '' + '
';
return tooltipText;
}
},
grid: {
left: '15%',
containLabel: false
}
};
if (option && typeof option === 'object') {
myChart.setOption(option, true);
}
window.addEventListener('resize', myChart.resize);
}