billboard.js
Version:
Re-usable easy interface JavaScript chart library, based on D3 v4+
371 lines (314 loc) • 10.7 kB
text/typescript
/**
* Copyright (c) 2017 ~ present NAVER Corp.
* billboard.js project is licensed under the MIT license
*/
import {select as d3Select} from "d3-selection";
import type {d3Selection, DataRow} from "../../../types/types";
import {$BAR, $COMMON} from "../../config/classes";
import {getRandom, isNumber} from "../../module/util";
import type {IBarData} from "../data/IData";
import {getBarRadiusInfo, getBarRadiusResolver, getStackingBarRadiusSet} from "./core/barRadius";
import {getShapeColorWithGradient, updateTargetsForShape} from "./shape";
type BarTypeDataRow = DataRow<number | number[]>;
type BarConnectLine = {x: number, y: number, width: number, height: number};
type BarPath = (string | BarConnectLine)[];
/**
* Get the type of connect line for bar chart
* @param {string} id Data id
* @returns {string|null} Connect line type or null if not applicable
* @private
*/
function _getConnectLineType(id: string): string | null {
const connectLine = this.config.bar_connectLine;
const type = connectLine?.[id] || connectLine;
return (/^(start|end)\-(start|end)$/.test(type)) ? type : null;
}
export default {
initBar(): void {
const {$el, config, state: {clip}} = this;
$el.bar = $el.main.select(`.${$COMMON.chart}`);
$el.bar = config.bar_front ? $el.bar.append("g") : $el.bar.insert("g", ":first-child");
$el.bar
.attr("class", $BAR.chartBars)
.call(this.setCssRule(false, `.${$BAR.chartBars}`, ["pointer-events:none"]));
// set clip-path attribute when condition meet
// https://github.com/naver/billboard.js/issues/2421
if (
config.clipPath === false && (
config.bar_radius || config.bar_radius_ratio
)
) {
$el.bar.attr("clip-path", clip.pathXAxis.replace(/#[^)]*/, `#${clip.id}`));
}
},
updateTargetsForBar(targets: BarTypeDataRow[]): void {
const $$ = this;
const {config} = $$;
const classBars = $$.getClass("bars", true);
const isSelectable = config.interaction_enabled && config.data_selection_isselectable;
const mainBarEnter = updateTargetsForShape.call($$, targets, {
type: "Bar",
elKey: "bar",
containerClass: $BAR.chartBars,
itemClass: $BAR.chartBar,
initFn: $$.initBar
});
// Bars for each data
mainBarEnter.append("g")
.attr("class", classBars)
.style("cursor", d => (isSelectable?.bind?.($$.api)(d) ? "pointer" : null))
.call(selection => {
$$.setCssRule(true, ` .${$BAR.bar}`, ["fill"], $$.color)(selection);
// add bar connect line
selection.each(function(d) {
if (_getConnectLineType.call($$, d.id)) {
d3Select(this).append("path")
.attr("class", $BAR.barConnectLine);
}
});
});
},
/**
* Generate/Update elements
* @param {boolean} withTransition Transition for exit elements
* @param {boolean} isSub Subchart draw
* @private
*/
updateBar(withTransition: boolean, isSub = false): void {
const $$ = this;
if ($$.state.isCanvasMode) {
return;
}
const {config, $el, $T} = $$;
const $root = isSub ? $el.subchart : $el;
const classBar = $$.getClass("bar", true);
const initialOpacity = $$.initialOpacity.bind($$);
config.bar_linearGradient && $$.updateLinearGradient();
const bar = $root.main.selectAll(`.${$BAR.bars}`)
.selectAll(`.${$BAR.bar}`)
.data($$.labelishData.bind($$));
$T(bar.exit(), withTransition)
.style("opacity", "0")
.remove();
$root.bar = bar.enter().append("path")
.attr("class", classBar)
.style("fill", $$.updateBarColor.bind($$))
.merge(bar)
.style("opacity", initialOpacity);
// calculate ratio if grouped data exists
$$.setRatioForGroupedData($root.bar.data());
},
/**
* Update bar color
* @param {object} d Data object
* @returns {string} Color string
* @private
*/
updateBarColor(d: IBarData): string | null {
const $$ = this;
const fn = $$.getStylePropValue($$.color);
return getShapeColorWithGradient.call($$, d, "bar_linearGradient", fn || (() => null));
},
/**
* Redraw function
* @param {function} drawFn Retuned function from .getDrawShape() => .generateDrawBar()
* @param {boolean} withTransition With or without transition
* @param {boolean} isSub Subchart draw
* @returns {Array}
* @private
*/
redrawBar(drawFn, withTransition?: boolean, isSub = false) {
const $$ = this;
if ($$.state.isCanvasMode) {
return [];
}
const {bar} = isSub ? $$.$el.subchart : $$.$el;
const barPath: BarConnectLine[] = [];
const connectLineCache = new Map<string, string | null>();
return [
$$.$T(bar, withTransition, getRandom())
.attr("d", function(d, i, arr) {
const path = (isNumber(d.value) || $$.isBarRangeType(d)) && drawFn(d, i);
// Memoize per series id: config lookup + regex runs once per id, not per bar
let connectLineType = connectLineCache.get(d.id);
if (connectLineType === undefined) {
connectLineType = _getConnectLineType.call($$, d.id);
connectLineCache.set(d.id, connectLineType);
}
// for bar.connectLine option
if (path.length > 1) {
barPath.push(path[1]);
}
// flush per series even when the last datum is null,
// otherwise the accumulated path leaks into the next series
if (i === arr.length - 1 && barPath.length) {
const barConnectLineNode = $$.$T(
d3Select(this.parentNode.querySelector(`.${$BAR.barConnectLine}`)),
withTransition,
getRandom()
);
$$.updateConnectLine(barConnectLineNode, connectLineType, barPath);
barPath.splice(0);
}
return path[0];
})
.style("fill", $$.updateBarColor.bind($$))
.style("clip-path", d => d.clipPath)
.style("opacity", null)
];
},
/**
* Generate draw function
* @param {object} barIndices data order within x axis.
* barIndices ==> {data1: 0, data2: 0, data3: 1, data4: 1, __max__: 1}
*
* When gropus given as:
* groups: [
* ["data1", "data2"],
* ["data3", "data4"]
* ],
*
* Will be rendered as:
* data1 data3 data1 data3
* data2 data4 data2 data4
* -------------------------
* 0 1
* @param {boolean} isSub If is for subchart
* @returns {function}
* @private
*/
generateDrawBar(barIndices, isSub?: boolean): (d: IBarData, i: number) => BarPath {
const $$ = this;
const {config} = $$;
const getPoints = $$.generateGetBarPoints(barIndices, isSub);
const getRadius = getBarRadiusResolver($$);
const stackingRadiusSet = getRadius ? getStackingBarRadiusSet($$) : new Set<string>();
return (d: IBarData, i: number): BarPath => {
// 4 points that make a bar
const points = getPoints(d, i);
const {
indexX,
indexY,
isNegative,
pos,
radius,
clipPath
} = getBarRadiusInfo(
$$,
d,
points,
getRadius,
stackingRadiusSet,
$$.isStackingRadiusData.bind($$)
);
const pathRadius = ["", ""];
// initialize as null to not set attribute if isn't needed
d.clipPath = clipPath;
if (getRadius) {
const arc = `a${radius} ${radius} ${isNegative ? "1 0 0" : "0 0 1"} `;
pathRadius[indexY] = `${arc}${radius},${radius}`;
pathRadius[indexX] = `${arc}${
[-radius, radius][
config.axis_rotated ? "sort" : "reverse"
]()
}`;
isNegative && pathRadius.reverse();
}
// path string data shouldn't be containing new line chars
// https://github.com/naver/billboard.js/issues/530
const path = config.axis_rotated ?
`H${pos} ${pathRadius[0]}V${points[2][indexY] - radius} ${pathRadius[1]}H${
points[3][indexX]
}` :
`V${pos} ${pathRadius[0]}H${points[2][indexX] - radius} ${pathRadius[1]}V${
points[3][indexY]
}`;
const coords: BarPath = [`M${points[0][indexX]},${points[0][indexY]}${path}z`];
if (_getConnectLineType.call($$, d.id)) {
coords.push(config.axis_rotated ?
{
x: points[0][indexX],
y: points[0][indexY],
width: points[0][indexX] - pos,
height: points[2][indexY] - points[0][indexY]
} :
{
x: points[0][indexX],
y: pos,
width: points[2][indexX] - points[0][indexX],
height: points[3][indexY] - pos
});
}
return coords;
};
},
/**
* Determine if given stacking bar data is radius type
* @param {object} d Data row
* @returns {boolean}
*/
isStackingRadiusData(d: IBarData): boolean {
const $$ = this;
const {$el, config, data, state} = $$;
const {id, index, value} = d;
// when the data is hidden, check if has rounded edges
if (state.hiddenTargetIds.has(id)) {
const target = $el.bar.filter(d => d.id === id && d.value === value);
return !target.empty() && /a\d+/i.test(target.attr("d"));
}
// Find same grouped ids
const keys = config.data_groups.find(v => v.indexOf(id) > -1);
// Get sorted list
const sortedList = $$.orderTargets(
$$.filterTargetsToShow(data.targets.filter($$.isBarType, $$))
).filter(v => keys.indexOf(v.id) > -1);
// Get sorted Ids. Filter positive or negative values Ids from given value
const sortedIds = sortedList
.map(v => {
// Direct index access (values are sorted by index from convertDataToTargets)
const v2 = v.values[index];
if (v2 && (isNumber(value) && value > 0 ? v2.value > 0 : v2.value < 0)) {
return v2;
}
return undefined;
})
.filter(Boolean)
.map(v => v.id);
// If the given id stays in the last position, then radius should be applied.
return value !== 0 && (sortedIds.indexOf(id) === sortedIds.length - 1);
},
/**
* Update the bar connect line path
* @param {d3Selection} node d3 selection of bar connect line
* @param {string} type Type of connect line, one of "start-start", "start-end", "end-start", "end-end"
* @param {Array} barPath d3 path data for the bar
*/
updateConnectLine(
node: d3Selection,
type: "start-start" | "start-end" | "end-start" | "end-end",
barPath: BarConnectLine[]
): void {
const path = barPath.map((v: BarConnectLine, i: number, arr: BarConnectLine[]): string => {
const isRotated = this.config.axis_rotated;
const isStart = /^start-(start|end)$/.test(type);
const isEnd = /^end-(start|end)$/.test(type);
const path: string[] = [];
const x = isRotated ? (isEnd ? v.x - v.width : v.x) : (v.x + v.width);
const y = isRotated ? v.y + v.height : isStart ? v.y + v.height : v.y;
if (i === 0) {
path.push(`${x},${y}`);
} else {
path.push(
isRotated ?
`L${v.x - (/\w+-end$/.test(type) ? v.width : 0)},${v.y}` :
`L${v.x},${v.y + (/\w+-start$/.test(type) ? v.height : 0)}`
);
if (i < arr.length - 1) {
path.push(`M${x},${y}`);
}
}
return path.join(" ");
});
node.attr("d", `M${path.join("")}z`);
}
};