UNPKG

billboard.js

Version:

Re-usable easy interface JavaScript chart library, based on D3 v4+

477 lines (474 loc) 18 kB
/*! * Copyright (c) 2017 ~ present NAVER Corp. * billboard.js project is licensed under the MIT license * * billboard.js, JavaScript chart library * https://naver.github.io/billboard.js/ * * @version 4.0.1 */ import { select, selectAll } from 'd3-selection'; import { $GRID, $AXIS, $FOCUS, $COMMON } from '../../config/classes.js'; import { AXIS_DEFAULT_TICK_COUNT } from '../../config/const.js'; import { isArray, isValue } from '../../module/util/type-checks.js'; import { getPointer } from '../../module/util/dom.js'; /** * Copyright (c) 2017 ~ present NAVER Corp. * billboard.js project is licensed under the MIT license */ // Grid position and text anchor helpers const GRID_FOCUS_SELECTOR = `line.${$FOCUS.xgridFocus}, line.${$FOCUS.ygridFocus}`; const _getGridTextAnchor = d => isValue(d.position) || "end"; const _getGridTextDx = d => (d.position === "start" ? 4 : (d.position === "middle" ? 0 : -4)); /** * Get current grid focus line selection. * @param {object} $$ ChartInternal context * @returns {d3.selection} Grid focus line selection * @private */ function _getGridFocusEl($$) { const { state, $el: { main } } = $$; const cached = state._gridFocusEl; const mainNode = main.node(); const cachedNodes = cached?.nodes?.() || []; return cachedNodes.length && cachedNodes.every(node => mainNode?.contains(node)) ? cached : (state._gridFocusEl = main.selectAll(GRID_FOCUS_SELECTOR)); } /** * Get grid text x value getter function * @param {boolean} isX Is x Axis * @param {number} width Width value * @param {number} height Height value * @returns {function} * @private */ function _getGridTextX(isX, width, height) { return d => { let x = isX ? 0 : width; if (d.position === "start") { x = isX ? -height : 0; } else if (d.position === "middle") { x = (isX ? -height : width) / 2; } return x; }; } /** * Update coordinate attributes value * @param {d3.selection} el Target node * @param {string} type Type * @private */ function _smoothLines(el, type) { { el.each(function () { const g = select(this); ["x1", "x2", "y1", "y2"] .forEach(v => g.attr(v, +g.attr(v))); }); } } var internalGrid = { hasGrid() { const { config } = this; return ["x", "y"] .some(v => config[`grid_${v}_show`] || config[`grid_${v}_lines`].length); }, initGrid() { const $$ = this; $$.hasGrid() && $$.initGridLines(); $$.initFocusGrid(); }, initGridLines() { const $$ = this; const { config, state: { clip }, $el } = $$; if (config.grid_x_lines.length || config.grid_y_lines.length) { $el.gridLines.main = $el.main.insert("g", `.${$COMMON.chart}${config.grid_lines_front ? " + *" : ""}`) .attr("clip-path", clip.pathGrid) .attr("class", `${$GRID.grid} ${$GRID.gridLines}`); $el.gridLines.main.append("g").attr("class", $GRID.xgridLines); $el.gridLines.main.append("g").attr("class", $GRID.ygridLines); $el.gridLines.x = selectAll([]); } }, updateXGrid(withoutUpdate) { const $$ = this; const { config, scale, state, $el: { main, grid } } = $$; const isRotated = config.axis_rotated; const xgridData = $$.generateGridData(config.grid_x_type, scale.x); const tickOffset = $$.axis.isCategorized() ? $$.axis.x.tickOffset() : 0; const pos = d => (scale.zoom || scale.x)(d) + (tickOffset * (isRotated ? -1 : 1)); state.xgridAttr = isRotated ? { x1: 0, x2: state.width, y1: pos, y2: pos } : { x1: pos, x2: pos, y1: 0, y2: state.height }; grid.x = main.select(`.${$GRID.xgrids}`) .selectAll(`.${$GRID.xgrid}`) .data(xgridData); grid.x.exit().remove(); grid.x = grid.x.enter() .append("line") .attr("class", $GRID.xgrid) .merge(grid.x); if (!withoutUpdate) { grid.x.each(function () { const grid = select(this); Object.keys(state.xgridAttr).forEach(id => { grid.attr(id, state.xgridAttr[id]); }); // hide the gridline overlapping the axis line (attr() returns a string) grid.style("opacity", () => (+grid.attr(isRotated ? "y1" : "x1") === (isRotated ? state.height : 0) ? "0" : null)); }); } }, updateYGrid() { const $$ = this; const { axis, config, scale, state, $el: { grid, main } } = $$; const isRotated = config.axis_rotated; const pos = d => scale.y(d); const gridValues = axis.y.getGeneratedTicks(config.grid_y_ticks) || $$.scale.y.ticks(config.grid_y_ticks); grid.y = main.select(`.${$GRID.ygrids}`) .selectAll(`.${$GRID.ygrid}`) .data(gridValues); grid.y.exit().remove(); grid.y = grid.y .enter() .append("line") .attr("class", $GRID.ygrid) .merge(grid.y); grid.y.attr("x1", isRotated ? pos : 0) .attr("x2", isRotated ? pos : state.width) .attr("y1", isRotated ? 0 : pos) .attr("y2", isRotated ? state.height : pos); _smoothLines(grid.y); }, updateGrid() { const $$ = this; const { $el: { grid, gridLines } } = $$; !gridLines.main && $$.initGridLines(); // hide if arc type grid.main.style("visibility", $$.hasArcType() ? "hidden" : null); $$.hideGridFocus(); $$.updateGridLines("x"); $$.updateGridLines("y"); }, /** * Update Grid lines * @param {string} type x | y * @private */ updateGridLines(type) { const $$ = this; const { config, $el: { gridLines, main }, $T } = $$; const isRotated = config.axis_rotated; const isX = type === "x"; config[`grid_${type}_show`] && $$[`update${type.toUpperCase()}Grid`](); let lines = main.select(`.${$GRID[`${type}gridLines`]}`) .selectAll(`.${$GRID[`${type}gridLine`]}`) .data(config[`grid_${type}_lines`]); // exit $T(lines.exit()) .style("opacity", "0") .remove(); // enter const gridLine = lines.enter().append("g"); gridLine.append("line") .style("opacity", "0"); lines = gridLine.merge(lines); lines.each(function (d) { const g = select(this); if (g.select("text").empty() && d.text) { g.append("text") .style("opacity", "0"); } }); $T(lines .attr("class", d => `${$GRID[`${type}gridLine`]} ${d.class || ""}`.trim()) .select("text") .attr("text-anchor", _getGridTextAnchor) .attr("transform", () => (isX ? (isRotated ? null : "rotate(-90)") : (isRotated ? "rotate(-90)" : null))) .attr("dx", _getGridTextDx) .attr("dy", -5)) .text(function (d) { return d.text ?? this.remove(); }); gridLines[type] = lines; }, redrawGrid(withTransition) { const $$ = this; const { config: { axis_rotated: isRotated }, state: { width, height }, $el: { gridLines }, $T } = $$; const xv = $$.xv.bind($$); const yv = $$.yv.bind($$); let xLines = gridLines.x.select("line"); let xTexts = gridLines.x.select("text"); let yLines = gridLines.y.select("line"); let yTexts = gridLines.y.select("text"); xLines = $T(xLines, withTransition) .attr("x1", isRotated ? 0 : xv) .attr("x2", isRotated ? width : xv) .attr("y1", isRotated ? xv : 0) .attr("y2", isRotated ? xv : height); xTexts = $T(xTexts, withTransition) .attr("x", _getGridTextX(!isRotated, width, height)) .attr("y", xv); yLines = $T(yLines, withTransition) .attr("x1", isRotated ? yv : 0) .attr("x2", isRotated ? yv : width) .attr("y1", isRotated ? 0 : yv) .attr("y2", isRotated ? height : yv); yTexts = $T(yTexts, withTransition) .attr("x", _getGridTextX(isRotated, width, height)) .attr("y", yv); return [ xLines.style("opacity", null), xTexts.style("opacity", null), yLines.style("opacity", null), yTexts.style("opacity", null) ]; }, initFocusGrid() { const $$ = this; const { config, state, state: { clip }, $el } = $$; // Invalidate cached D3 selection in case grid is re-initialized state._gridFocusEl = null; const isFront = config.grid_front; const className = `.${isFront && $el.gridLines.main ? $GRID.gridLines : $COMMON.chart}${isFront ? " + *" : ""}`; const grid = $el.main.insert("g", className) .attr("clip-path", clip.pathGrid) .attr("class", $GRID.grid); $el.grid.main = grid; config.grid_x_show && grid.append("g").attr("class", $GRID.xgrids); config.grid_y_show && grid.append("g").attr("class", $GRID.ygrids); if (config.axis_tooltip) { const axis = grid.append("g").attr("class", "bb-axis-tooltip"); axis.append("line").attr("class", "bb-axis-tooltip-x"); axis.append("line").attr("class", "bb-axis-tooltip-y"); } if (config.interaction_enabled && config.grid_focus_show && !config.axis_tooltip) { grid.append("g") .attr("class", $FOCUS.xgridFocus) .append("line") .attr("class", $FOCUS.xgridFocus); // to show xy focus grid line, should be 'tooltip.grouped=false' if (config.grid_focus_y && !config.tooltip_grouped) { grid.append("g") .attr("class", $FOCUS.ygridFocus) .append("line") .attr("class", $FOCUS.ygridFocus); } } }, showAxisGridFocus() { const $$ = this; const { config, format, state: { event, width, height } } = $$; const isRotated = config.axis_rotated; // get mouse event position const [x, y] = getPointer(event, $$.$el.eventRect?.node()); const pos = { x, y }; for (const [axis, node] of Object.entries($$.$el.axisTooltip)) { const attr = (axis === "x" && !isRotated) || (axis !== "x" && isRotated) ? "x" : "y"; const value = pos[attr]; let scaleText = $$.scale[axis]?.invert(value); if (scaleText) { scaleText = axis === "x" && $$.axis.isTimeSeries() ? format.xAxisTick(scaleText) : scaleText?.toFixed(2); // set position & its text value based on position node?.attr(attr, value) .text(scaleText); } } $$.$el.main.selectAll(`line.bb-axis-tooltip-x, line.bb-axis-tooltip-y`).style("visibility", null) .each(function (d, i) { const line = select(this); if (i === 0) { line .attr("x1", x) .attr("x2", x) .attr("y1", i ? 0 : height) .attr("y2", i ? height : 0); } else { line .attr("x1", i ? 0 : width) .attr("x2", i ? width : 0) .attr("y1", y) .attr("y2", y); } }); }, hideAxisGridFocus() { const $$ = this; $$.$el.main.selectAll(`line.${$AXIS.axisTooltipX}, line.${$AXIS.axisTooltipY}`).style("visibility", "hidden"); Object.values($$.$el.axisTooltip) .forEach((v) => v?.style("display", "none")); }, /** * Show grid focus line * @param {Array} data Selected data * @private */ showGridFocus(data) { const $$ = this; const { config, state: { width, height } } = $$; const isRotated = config.axis_rotated; // Cache grid focus selection to avoid repeated DOM queries on mousemove const focusEl = _getGridFocusEl($$); const dataToShow = (data || [focusEl.datum()]).filter(d => d && isValue($$.getBaseValue(d))); // Hide when bubble/scatter/stanford plot exists if (!config.tooltip_show || dataToShow.length === 0 || (!config.axis_x_forceAsSingle && $$.hasType("bubble")) || $$.hasArcType()) { return; } const isEdge = config.grid_focus_edge && !config.tooltip_grouped; const xx = $$.xx.bind($$); focusEl .style("visibility", null) .data(dataToShow.concat(dataToShow)) .each(function (d) { const el = select(this); const pos = { x: xx(d), y: $$.getYScaleById(d.id)(d.value) }; let xy; if (el.classed($FOCUS.xgridFocus)) { // will contain 'x1, y1, x2, y2' order xy = isRotated ? [ null, // x1 pos.x, // y1 isEdge ? pos.y : width, // x2 pos.x // y2 ] : [ pos.x, isEdge ? pos.y : null, pos.x, height ]; } else { const isY2 = $$.axis.getId(d.id) === "y2"; xy = isRotated ? [ pos.y, // x1 isEdge && !isY2 ? pos.x : null, // y1 pos.y, // x2 isEdge && isY2 ? pos.x : height // y2 ] : [ isEdge && isY2 ? pos.x : null, pos.y, isEdge && !isY2 ? pos.x : width, pos.y ]; } ["x1", "y1", "x2", "y2"] .forEach((v, i) => el.attr(v, xy[i])); }); _smoothLines(focusEl); $$.showCircleFocus?.(data); }, hideGridFocus(force = false) { const $$ = this; const { state: { inputType, resizing } } = $$; if (force || inputType === "mouse" || !resizing) { const focusEl = _getGridFocusEl($$); focusEl.style("visibility", "hidden"); $$.hideCircleFocus?.(); } }, updateGridFocus() { const $$ = this; const { state: { inputType, width, height, resizing }, $el: { grid } } = $$; const xgridFocus = grid.main.select(`line.${$FOCUS.xgridFocus}`); if (inputType === "touch") { if (xgridFocus.empty()) { resizing && $$.showCircleFocus?.(); } else { $$.showGridFocus(); } } else { const isRotated = $$.config.axis_rotated; xgridFocus .attr("x1", isRotated ? 0 : -10) .attr("x2", isRotated ? width : -10) .attr("y1", isRotated ? -10 : 0) .attr("y2", isRotated ? -10 : height); } // need to return 'true' as of being pushed to the redraw list // ref: getRedrawList() return true; }, generateGridData(type, scale) { const $$ = this; const tickNum = $$.$el.main.select(`.${$AXIS.axisX}`) .selectAll(".tick") .size(); let gridData = []; if (type === "year") { const xDomain = $$.getXDomain($$.data.targets); const [firstYear, lastYear] = xDomain.map(v => v.getFullYear()); for (let i = firstYear; i <= lastYear; i++) { gridData.push(new Date(`${i}-01-01 00:00:00`)); } } else { gridData = scale.ticks(AXIS_DEFAULT_TICK_COUNT); if (gridData.length > tickNum) { // use only int gridData = gridData.filter(d => String(d).indexOf(".") < 0); } } return gridData; }, getGridFilterToRemove(params) { return params ? line => { let found = false; (isArray(params) ? params.concat() : [params]).forEach(param => { if ((("value" in param && line.value === param.value) || ("class" in param && line.class === param.class))) { found = true; } }); return found; } : () => true; }, removeGridLines(params, forX) { const $$ = this; const { config, $T } = $$; const toRemove = $$.getGridFilterToRemove(params); const toShow = line => !toRemove(line); const classLines = forX ? $GRID.xgridLines : $GRID.ygridLines; const classLine = forX ? $GRID.xgridLine : $GRID.ygridLine; $T($$.$el.main.select(`.${classLines}`) .selectAll(`.${classLine}`) .filter(toRemove)) .style("opacity", "0") .remove(); const gridLines = `grid_${forX ? "x" : "y"}_lines`; config[gridLines] = config[gridLines].filter(toShow); } }; export { internalGrid as default };