UNPKG

billboard.js

Version:

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

149 lines (122 loc) 3.49 kB
/** * Copyright (c) 2017 ~ present NAVER Corp. * billboard.js project is licensed under the MIT license */ import {select as d3Select} from "d3-selection"; // selection import CLASS from "../../config/classes"; import {isValue, parseDate} from "../../module/util"; import {AxisType} from "../../../types/types"; export default { initRegion() { const $$ = this; const {$el} = $$; $el.region.main = $el.main.append("g") .attr("clip-path", $$.state.clip.path) .attr("class", CLASS.regions); }, updateRegion(duration: number): void { const $$ = this; const {config, $el} = $$; if (!$el.region.main) { $$.initRegion(); } // hide if arc type $el.region.main.style("visibility", $$.hasArcType() ? "hidden" : "visible"); // select <g> element let list = $el.main.select(`.${CLASS.regions}`) .selectAll(`.${CLASS.region}`) .data(config.regions); list.exit() .transition() .duration(duration) .style("opacity", "0") .remove(); list = list.enter() .append("g") .merge(list) .attr("class", $$.classRegion.bind($$)); list .append("rect") .style("fill-opacity", "0"); $el.region.list = list; }, redrawRegion(withTransition) { const $$ = this; let regions = $$.$el.region.list.select("rect"); regions = (withTransition ? regions.transition() : regions) .attr("x", $$.regionX.bind($$)) .attr("y", $$.regionY.bind($$)) .attr("width", $$.regionWidth.bind($$)) .attr("height", $$.regionHeight.bind($$)); return [ (withTransition ? regions.transition() : regions) .style("fill-opacity", d => (isValue(d.opacity) ? d.opacity : "0.1")) .on("end", function() { // remove unnecessary rect after transition d3Select(this.parentNode) .selectAll("rect:not([x])") .remove(); }) ]; }, getRegionXY(type: AxisType, d): number { const $$ = this; const {config, scale} = $$; const isRotated = config.axis_rotated; const isX = type === "x"; let key = "start"; let currScale; let pos = 0; if (d.axis === "y" || d.axis === "y2") { if (!isX) { key = "end"; } if ((isX ? isRotated : !isRotated) && key in d) { currScale = scale[d.axis]; pos = currScale(d[key]); } } else if ((isX ? !isRotated : isRotated) && key in d) { currScale = scale.zoom || scale.x; pos = currScale($$.axis.isTimeSeries() ? parseDate.call($$, d[key]) : d[key]); } return pos; }, regionX(d): number { return this.getRegionXY("x", d); }, regionY(d): number { return this.getRegionXY("y", d); }, getRegionSize(type, d): number { const $$ = this; const {config, scale, state} = $$; const isRotated = config.axis_rotated; const isWidth = type === "width"; const start = $$[isWidth ? "regionX" : "regionY"](d); let currScale; let key = "end"; let end = state[type]; if (d.axis === "y" || d.axis === "y2") { if (!isWidth) { key = "start"; } if ((isWidth ? isRotated : !isRotated) && key in d) { currScale = scale[d.axis]; end = currScale(d[key]); } } else if ((isWidth ? !isRotated : isRotated) && key in d) { currScale = scale.zoom || scale.x; end = currScale($$.axis.isTimeSeries() ? parseDate.call($$, d[key]) : d[key]); } return end < start ? 0 : end - start; }, regionWidth(d): number { return this.getRegionSize("width", d); }, regionHeight(d): number { return this.getRegionSize("height", d); }, isRegionOnX(d): boolean { return !d.axis || d.axis === "x"; }, };