앞서의 예제는 기본적인 D3의 개념과 사용법을 익히기 위한 것이었다.
여기에서는 일반적인 바(column) 차트에서
흔히 보는 주요 기능 몇 가지(척도 중심)를 구현하면서
D3에 대한 보다 깊은 개념과 사용법을 정리한다.
1. X / Y 축 (척도) 만들기
2. 그리드 만들기
3. 도움말 만들기
앞서의 예제 중에서 마지막 예제를 다음과 같이 수정해 준다.
차트의 가로(X)와 세로 (Y) 축에
주어진 데이터의 값이 출력되는 것이
보기에도 좋고 차트에서 일반적으로 제공되는 기능이기도 하다.
따라서, 이전에는 데이터를 세로 축(Y)의 값만 배열로 작성했지만
가로축(X)에도 값을 출력하기 위해 Json을 배열로 작성했다 [라인 24].
예로, X축의 값 A, Y축의 값 9와 같이 ({x:'A', y:9})
X, Y의 값을 하나의 세트로 구성하였다.
이러한 데이터 구조의 변화에 따라,
33, 36, 44 라인에 작성된 것 같이 사용법에 변화가 있다.
.attr("height", function(d, i) {return (d.y*5)})
.attr("y", function(d, i) {return (250-d.y*5)});
.attr("y", function(d, i) {return 250-d.y*5 + 15});
바의 높이와 위치를 계산할 때
변수 d로 받아서 사용하던 데이터 사용법을 바꾸어 주어야 한다.
D3에서 data 지정을 배열로 하면
각각의 속성 지정에 넘어오는 값은 각 배열의 원소(d)가 넘어오고,
이 배열 원소는 그냥 값이니 그대로 사용하면 된다.
즉, d * 5와 같이 바로 사용했다.
수정한 예제는 Json 배열이니 각 원소는 Json이 되고
Json은 x, y로 지정되어 있기 때문에
d.y * 5와 같은 방식으로 사용되게 된다.
이상의 코드는 앞서의 예제를 배열에서 Json으로 바꾼 것 외에
다음 실행 결과와 같이 CSS를 이용하여 SVG 테그에 외곽선을 그렸다 [라인 4].
외곽선이 있는 실행 결과를 보면
차트가 왼쪽으로 치우친 것을 알 수 있다.
즉, 주어진 SVG의 크기에 맞게 제작되지 않았다.
앞서의 예제에서 크기에 맞춰서 바(rect)를 생성하는 수식을 사용하지 않았다.
즉, 각 개별 바의 크기는
바의 간격과 바의 수 (데이터의 수)를 고려하여
SVG의 크기(width)에 맞게 계산해야 한다.
이전 예제에서는 바의 크기(width)는 40, 간격은 10으로 고정해서 구현했다.
바의 높이도 주어진 값에 5를 곱했을 뿐이다.
바의 높이는 주어진 값 중 최고 값(Max)과
SVG의 높이(height)를 비율로 계산해서 처리해야 한다.
서술이 길었는데
핵심은 제법 복잡할 수 있는 이러한 처리를
D3에서는 간단한 코드 몇 줄로 구현할 수 있다는 것이다.
먼저 다음 그림과 같이
SVG 크기에 맞추어 X축부터 출력해 본다.
기능을 구현하기 전에 D3에서 제공하는 scale에 대해서 알아야 하는데
v3에서 v4로 바뀌면서 변화된 것이 많고
종류도 많은데 정리된 자료가 별로 없다.
D3의 주요 기능 중 하나이니
영문 자료라도 확인해 두면 도움이 될 것이다.
v4에 대한 자료는 별로 없지만
v3는 검색해 보거나 다음 자료들을 참고하면 도움이 될 것이다.
여기서는 필요한 기능만 정리하고 넘어간다.
http://blog.naver.com/PostView.nhn?blogId=zero_kjy&logNo=생략
http://blog.naver.com/PostView.nhn?blogId=zero_kjy&logNo=생략
http://blog.naver.com/PostView.nhn?blogId=zero_kjy&logNo=생략
https://www.slideshare.net/neuroassociates/week16-d3js
http://www.bsidesoft.com/?p=2382
http://hunjae.com/d3-js-scale/
두 버전의 차이를 간단하게 정리하면
v3에서는 scale(척도)과 axix(축)을 나누어서 구현했다.
v4에서는 하나(scale)로 처리하는 차이가 있다.
v4에 대한 영문 자료를 대충만 봐도 알 수 있지만
아주 많은 척도가 정의되어 있다.
이 중에서 scaleBand라는 척도가 사용된다는 것만 기억하고 넘어간다.
척도에서 중요한 개념(함수)이 domain과 range로
v3와 v4에서 같은 개념과 방식으로 사용된다.
domain은 데이터 값들의 정보(범위, 값)를 지정하고,
range는 출력되는 화면의 정보(범위, 픽셀)을 의미한다.
즉, 척도는 데이터 정보과 출력 정보를 결합하여 계산하는 역활을 하는 것이다.
예로, 1부터 5까지의 값을 가지는 데이터가 있고
출력하고자 하는 차트(SVG)의 너비가 100픽셀(px)이라고 하면,
domain([1,5])가 되고 range([1,100])로 작성한다.
그리고, 척도(scaleBand)에서
데이터 값이 1일때는 20픽셀, 2일때는 40픽셀과 같은
적당한 위치값을 계산해서 반환하게 된다.
즉, 척도(scaleBand)라는 클래스에
데이터 값(domain)과 출력 범위(range)를 넣어주고
필요한 값을 꺼내어 사용하면 된다.
필요한 값은 한 데이터(차트 도형, Rect)의 크기와 위치이다.
크기는 x축이니 width로 bandwidth란 메소드를 이용하고
위치는 척도의 인스턴스(xScale) 함수를 호출하면 된다.
이러한 기초 지식을 가지고 다음 코드를 이해해 본다.
앞서 정리한 내용을 코드로 작성하고 빨강색으로 표시하였다.
var dataset = [{x:'A', y:9 }, {x:'B', y:19}, {x:'C', y:29}, {x:'D', y:39},
{x:'E', y:29}, {x:'F', y:19}, {x:'G', y:9 }];
var svg = d3.select("svg");
var width = parseInt(svg.style("width"), 10);
var height = parseInt(svg.style("height"), 10)-20;
var xScale = d3.scaleBand() ①
.domain(dataset.map(function(d) { return d.x;} ))
.range([0, width]).padding(0.2);
svg.selectAll("rect")
.data(dataset)
.enter().append("rect")
.attr("class", "bar")
.attr("height", function(d, i) {return (d.y*5)})
.attr("width", xScale.bandwidth()) ②
.attr("x", function(d, i) {return xScale(d.x)})
.attr("y", function(d, i) {return (height-d.y*5)});
svg.selectAll("text")
.data(dataset)
.enter().append("text")
.text(function(d) {return d.y})
.attr("class", "text")
.attr("x", function(d, i) {return xScale(d.x)+xScale.bandwidth()/2})
.style("text-anchor", "middle")
.attr("y", function(d, i) {return height-d.y*5 + 15});
svg.append("g") ③
.attr("transform", "translate(0," + (height) + ")")
.call(d3.axisBottom(xScale));
① domain에 지정하는 데이터 값이
연속된 값이면 시작값과 종료값(최대값)을 지정하면 된다.
여기서는 A, B, C 등의 문자열이기 때문에
배열(dataset)에 사용된 값 모두를 나열하여(map) 지정하였다.
range의 화면 범위를 지정하기 위해
SVG의 CSS width값을 구해서 지정하였다 (svg.style("width")).
② 바를 생성하면서(append("rect"))
바의 크기(attr-width)와 위치(attr-x)를 척도를 이용해서 지정한다.
이렇게 척도를 이용하여 차트의 크기(SVG)나
입력되는 데이터의 개수에 따라 적절한 도형(rect)이 생성되게 된다.
배열 개수를 바꿔서 확인 해보길 바란다.
③ 마지막으로 척도를 이용하여 축(axis)을 작성한다.
X축이니 화면(SVG) 바닥(bottom)에 놓고
위치는 화면의 높이(height = Height-20)가 된다.
즉, translate를 이용하여 X 좌표는 0, Y 좌표는 height인 지점에
라인을 그리고 축의 값(xScale)을 출력한다.
height에 20px을 뺀 것은 X 축의 값을 출력하는 공간을 계산한 것이다.
지금까지 사용하지 않은 그룹 개념이 사용되었다 (svgG.append("g")).
즉, 눈금 개수(tick count) 만큼 SVG 선(Line)이 생성된다.
이것을 간편하게 관리하기 위해 그룹으로 관리한다.
웹 브라우저의 개발자 도구(F12)로 확인하면
SVG group (X축)에 여러 개의 눈금(Line)이 있는 것을 확인할 수 있다.
그림을 보면 몇개의 그룹이 생성된 것을 볼 수 있다.
첫 번째 그룹(g)이 코드로 직접 생성한 것이고
(코드에 사용된 transform과 height값 280 (300-20)이 보인다.)
첫 번째 그룹 하위에 있는 그룹들은
D3에서 생성한 척도의 눈금(tick)이다.
눈금 그룹은 눈금(line)과 X축의 값(text)로 구성된 것을 볼 수 있다.
차트에 출력된 값을 보면 앞서의 예제와 다른 점이 있다.
숫자가 도형의 정중앙에 출력된다.
(이전 예제는 19, 29와 같은 두자리 숫자와 9와 같은 한자리 숫자의 출력 좌표가 안 맞다)
이전 예제는 도형의 중앙을 계산하기 어려웠지만
척도를 이용하여 간단하게 계산할 수 있다.
도형을 출력할 좌표(xScale)에 도형 크기(width)의 반(/2)을 더해주면
숫자가 찍힐 정확한 중앙 지점을 계산할 수 있다.
(위 코드에서 파란색으로 표시된 코드를 의미한다.)
SVG의 Text 테그의 text-anchor 스타일을 middle로 지정하면
지정된 좌표가 문자열의 중앙에 오도록 출력해 준다.
이번에는 다음 그림과 같이 Y축을 구현해 본다.
앞서의 코드에서 X축을 구현하기 위해
빨간색으로 표시된 코드와 같은 방식으로 구현하면 되니
직접 해본 후 진행하는 것이 좋다.
다음 코드를 보면 알 수 있듯이
앞서의 예제 코드에 빨간색으로 표시한 부분이
Y축을 보이기 위해 작성한 코드이다.
var svg = d3.select("svg");
var width = parseInt(svg.style("width"), 10) -30;
var height = parseInt(svg.style("height"), 10)-20;
var svgG = svg.append("g") ③
.attr("transform", "translate(30, 0)");
var yScale = d3.scaleLinear() ①
.domain([0, d3.max(dataset, function(d){ return d.y; })])
.range([height, 0]);
svgG.selectAll("rect")
.data(dataset)
.enter().append("rect")
.attr("class", "bar")
.attr("height", function(d, i) {return height-yScale(d.y)})
.attr("width", xScale.bandwidth())
.attr("x", function(d, i) {return xScale(d.x)})
.attr("y", function(d, i) {return yScale(d.y)});
svgG.selectAll("text")
.data(dataset)
.enter().append("text")
.text(function(d) {return d.y})
.attr("class", "text")
.attr("x", function(d, i) {return xScale(d.x)+xScale.bandwidth()/2})
.style("text-anchor", "middle")
.attr("y", function(d, i) {return yScale(d.y) + 15});
svgG.append("g") ②
.call(d3.axisLeft(yScale).ticks(5));
① X축은 척도로 scaleBand를 사용한 반면,
Y축은 척도로 scaleLinear를 사용하였다.
X축의 값은 A, B, C 등의 문자열이었고,
Y축의 값은 1, 2, 3 등으로 연속된 숫자 값이다.
즉, 고정된 문자열에는 scaleBand, 일반 숫자값은 scaleLinear을 사용한다.
domain과 range는 모든 척도에서 같은 개념으로 사용된다.
다만, Y축이 숫자이기 때문에
0 부터 주어진 배열내 최대값(max(d.y))을 domain에 지정한다.
range 사용에서도 X축과 차이가 있다.
X축에서는 0부터 width로 지정했고
Y축에서는 height에서 0으로 지정했다.
즉, 큰 값이 먼저 나왔다.
X축은 A가 왼쪽, 즉 x좌표의 값이 가장 작다.
Y축은 1이 가장 하단, 즉 차트의 높이(Height) 값, 가장 큰 값을 가진다.
따라서, Y축은 domain 지정시 작은 값에서 큰 값으로
range 지정은 큰 값에서 작은 값으로 지정하여 서로 매핑 시켜준다.
② 마지막으로 X축은 axisBottom으로 하단에 축을 그리고
Y축은 axisLeft로 좌측에 축을 그린다.
다만, X축은 height값에 -20을 줘서 축의 위치를 바꾸어 줬지만
Y축은 다른 처리가 필요해서 축의 위치를 바꾸어 주지 않았다.
이것에 대한 자세한 설명은 뒤에 할 것이다.
Y축에는 척도의 개수(tick)을 5개로 지정하였다.
지정하지 않으면 D3에서 자동으로 지정한다.
③ 코드 작성이 완료 되었지만
이상의 코드에는 x축 생성할때 사용한 코드와 차이가 있다.
x축에 사용한 방식으로 Y축 코드를 작성하면
다음 표의 왼쪽처럼 작성해야 한다.
기대 되는 코드 |
현재 작성된 코드 |
svg.append("g") |
var svgG = svg.append("g") |
Y축을 이렇게 작성하면 다음 그림에서 왼쪽 하단 처럼
X축과 섞여서 이상하게 보인다.
이 문제를 해결하기 위해
차트 전체를 구성하는 그룹을 만들고 (svgG = svg.append("g"))
그룹 자체를 오른쪽으로 30 px 정도 이동시켜준다 (translate(30, 0)).
그리고, 차트를 위해 생성되는 도형(rect), 라벨(text), 척도등을
이전에는 svg에 생성했으나 svgG에 생성한다.
앞서 작성된 코드를 보면
X축 예제는 svg에 생성하였고, Y축 예제는 svgG에 생성하였다.
척도와 관련된 구현을 마쳤다.
데이터 개수를 추가해서 실행해 보면
그림과 같이 도형(바)의 크기가 바뀌면서 실행된 것을 볼 수 있다.
var dataset = [{x:'A', y:9 }, {x:'B', y:19}, {x:'C', y:29}, {x:'D', y:39},
{x:'E', y:29}, {x:'F', y:19}, {x:'G', y:9 }, {x:'H', y:29 }, {x:'I', y:39 }, {x:'J', y:49 }];
척도를 이용하므로서,
이번에는 그림처럼 차트에 그리드(Grid)을 추가한다.
그리드는 값의 정확한 위치를 보여 주기 위해 사용되는 것으로
D3에서는 간단하게 구현할 수 있다.
구현 방법은 코드에서 보듯이
그림과 같이 차트 도형(bar) 뒤편에
축을 그릴 때 사용한 눈금(tick)을 크게 그리는 것(tickSize)이다.
따라서, 눈금의 크기를 차트 영역의
너비(width)나 높이(height)만큼 크게 그리면 된다.
<style>
~~ 생략 ~~
.grid line {
stroke: lightgrey;
stroke-opacity: 0.7;
}
</style>
<script>
~~ 생략 ~~
svgG.append("g")
.attr("class", "grid")
.attr("transform", "translate(0," + height + ")")
.call(d3.axisBottom(xScale)
.tickSize(-height)
);
svgG.append("g")
.attr("class", "grid")
.call(d3.axisLeft(yScale)
.ticks(5)
.tickSize(-width)
);
~~ 생략 ~~
X축의 그리드 선은
지정된 X좌표에 0부터 높이(height)까지 이어진 선이다.
따라서, X축의 그리드 선은 0부터 높이,
Y축의 그리드 선은 0부터 너비까지가 된다.
여기에 더 고려해야 할 것은 마이너스(-) 값이란 것이다.
X축의 경우 바닥(axisBottom)에 선을 그리기 때문
현재 그리는 기준이 바닥(height)이다.
즉, 바닥(height)이 0이되니, 0부터 –height까지 그려야 한다.
Y축은 왼쪽(axisLeft)을 기준으로 설정했기 때문에
축의 왼쪽은 플러스(+) 값, 오른쪽은 마이너스(-) 값으로 지정해서 그린다.
작성된 그리드 선에 대한 효과는 CSS로 지정한다 (.attr("class", "grid")).
CSS에서 선의 색상(stroke)을 회색(lightgrey)으로 지정하고
선의 투명도(stroke-opacity)를 70%(0.7)로 잘 안보이게 해서
차트 도형을 보는데 지장이 없게 했다.
그리드와 관련해서 마지막으로 정리 할 것은
그리드의 작성된 코드가 차트 도형을 생성하기 전에 있어야 한다.
그리드의 작성된 코드가 차트 도형보다 앞에 있어야
그림과 같이 선 위에 도형이 놓이게 된다.
(그리드가 배경처럼 보이게 된다)
반대의 경우에는 도형 위에 선이 놓여
선이 더 중요하게 강조되는 것처럼 보이게 된다.
문제: 앞의 그림을 보면 도형의 값(label)이 사라졌다.
다시 보이게 할 방법은 무엇일까?
그림과 같이 차트 도형(바)에 마우스를 올리면
별도의 작은 창에 관련 데이터를 보여주는 기능을 구현한다.
이 기능을 구현하는 방법은
다음 문장을 그대로 코드로 작성하면 된다.
차트 도형에 마우스를 올리면(mouseover) 보이고,
도형에서 마우스가 벗어나면(mouseout) 안보이게 구현 한다.
마우스가 도형에 올라있는 동안은 마우스를 따라다니면서(mousemove)
마우스 근처에 작은 창(rect)을 하나 보여준다.
작은 창은 해당 도형의 데이터 값(Y 값)을 출력한다.
var barG = svgG.append("g");
barG.selectAll("rect")
.data(dataset)
.enter().append("rect")
.attr("class", "bar")
.attr("height", function(d, i) {return height-yScale(d.y)})
.attr("width", xScale.bandwidth())
.attr("x", function(d, i) {return xScale(d.x)})
.attr("y", function(d, i) {return yScale(d.y)})
.on("mouseover", function() { tooltip.style("display", null); })
.on("mouseout", function() { tooltip.style("display", "none"); })
.on("mousemove", function(d) {
var xP = d3.mouse(this)[0];
var yP = d3.mouse(this)[1] - 25;
tooltip.attr("transform", "translate(" + xP + "," + yP + ")");
tooltip.select("text").text(d.y);
});
barG.selectAll("text");
var tooltip = svg.append("g")
.attr("class", "tooltip")
.style("display", "none");
tooltip.append("rect")
.attr("width", 30)
.attr("height", 20)
.attr("fill", "white");
tooltip.append("text")
.attr("x", 15)
.attr("dy", "1em")
.style("text-anchor", "middle");
이 문장 다시 코드로 작성하면
툴팁은 rect(배경, 선등)와 text(값출력)로 구성된
tooltip이라는 SVG 그룹 태그로 (var tooltip = svg.append("g"))
평소에는 눈에 보이지 않는다 (display: "none").
이 tooltip은 도형의
mouseover 이벤트에서 display 값을 지워서(빈문자열) 보이게 하고
mouseout 이벤트에서 display 값을 none로 해서 보이지 않게 한다.
mousemove에서는 현재 마우스의 좌표(xP, yP)을 구해서
tooltip의 위치를 지정하고 값을 출력한다.
구해진 좌표값에 적절한 계산(-20)을 해서
툴팁이 마우스에 가려지지 않게 했다.
앞서 제시한 문제에서 도형의 값(label)이 사라진 문제를 제시하였다.
해결 방법은 위 코드에 사용된 barG 변수이다.
앞서의 예제에서는 svgG에 모든 기능을 구현했다.
이번 예에서는 그리드/척도(svgG)와
챠트 도형의 그룹(barG)을 다르게 작성했다.
하나의 그룹에 많은 개체가 있어서 위치 충돌이 있는 것으로 추측한다.(정확한 이유를 아는 분은 댓글을)
이번에는 툴팁을 DIV로 구현해 본다.
개인적으로 rect와 같은 SVG 도형들이 덜 익숙한 것도 있고
보기 좋게 구현하는 것이 다소 어렵게 느껴져서
CSS로 간편하게 지정하는 DIV를 선호한다.
.toolTip {
position: absolute;
border: 0 none;
border-radius: 4px 4px 4px 4px;
background-color: white;
padding: 5px;
text-align: center;
font-size: 11px;
}
~~ 생략 ~~
barG.selectAll("rect")
.data(dataset)
.enter().append("rect")
.attr("class", "bar")
.attr("height", function(d, i) {return height-yScale(d.y)})
.attr("width", xScale.bandwidth())
.attr("x", function(d, i) {return xScale(d.x)})
.attr("y", function(d, i) {return yScale(d.y)})
.on("mouseover", function() { tooltip.style("display", null); })
.on("mouseout", function() { tooltip.style("display", "none"); })
.on("mousemove", function(d) {
tooltip.style("left", (d3.event.pageX+10)+"px");
tooltip.style("top", (d3.event.pageY-10)+"px");
tooltip.html(d.y);
});
barG.selectAll("text")
var tooltip = d3.select("body").append("div").attr("class", "toolTip").style("display", "none");
다음 코드와 같이 D3의 append로 생성하고
toolTip이라는 CSS 클래스를 지정한다.
해당 클래스에서 CSS 속성으로 원하는 데로 지정하면 된다.
여기서는 툴팁 도형을 일반 사각(rect)이 아닌
모서리가 둥근 사각형으로 구현했다.
SVG Rect는 좌표(x, y)로 위치를 지정했지만
DIV는 left와 right 값으로 지정하고
위치 값도
SVG Rect는 SVG 내에서 위치로 지정했지만
DIV는 부모가 문서(body)이므로
문서 기준(pageX, pageY)으로 구현하였다.
문제 D3에서 척도는 매우 활용이 많고 중요한 클래스이다. 이 척도를 잘 이용하면 다양한 기능을 구현 할 수 있는데, 척도를 이용해서 각 도형의 색깔을 모두 다르게 부여하는 기능을 직접 구현해 본다. |
'JavaScript > Chart' 카테고리의 다른 글
1. D3 (SVG) 차트 만들기 - Bar I (0) | 2017.04.15 |
---|---|
3. D3 (SVG) 차트 만들기 - Line (6) | 2017.04.15 |