UNPKG

di-echarts

Version:

Apache ECharts is a powerful, interactive charting and data visualization library for browser

215 lines (182 loc) 6.72 kB
/* * Licensed to the Apache Software Foundation (ASF) under one * or more contributor license agreements. See the NOTICE file * distributed with this work for additional information * regarding copyright ownership. The ASF licenses this file * to you under the Apache License, Version 2.0 (the * "License"); you may not use this file except in compliance * with the License. You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, * software distributed under the License is distributed on an * "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY * KIND, either express or implied. See the License for the * specific language governing permissions and limitations * under the License. */ import * as zrUtil from 'zrender/src/core/util'; import {parsePercent} from '../../util/number'; import type GlobalModel from '../../model/Global'; import BoxplotSeriesModel from './BoxplotSeries'; import Axis2D from '../../coord/cartesian/Axis2D'; const each = zrUtil.each; interface GroupItem { seriesModels: BoxplotSeriesModel[] axis: Axis2D boxOffsetList: number[] boxWidthList: number[] } export interface BoxplotItemLayout { ends: number[][] initBaseline: number } export default function boxplotLayout(ecModel: GlobalModel) { const groupResult = groupSeriesByAxis(ecModel); each(groupResult, function (groupItem) { const seriesModels = groupItem.seriesModels; if (!seriesModels.length) { return; } calculateBase(groupItem); each(seriesModels, function (seriesModel, idx) { layoutSingleSeries( seriesModel, groupItem.boxOffsetList[idx], groupItem.boxWidthList[idx] ); }); }); } /** * Group series by axis. */ function groupSeriesByAxis(ecModel: GlobalModel) { const result: GroupItem[] = []; const axisList: Axis2D[] = []; ecModel.eachSeriesByType('boxplot', function (seriesModel: BoxplotSeriesModel) { const baseAxis = seriesModel.getBaseAxis(); let idx = zrUtil.indexOf(axisList, baseAxis); if (idx < 0) { idx = axisList.length; axisList[idx] = baseAxis; result[idx] = { axis: baseAxis, seriesModels: [] } as GroupItem; } result[idx].seriesModels.push(seriesModel); }); return result; } /** * Calculate offset and box width for each series. */ function calculateBase(groupItem: GroupItem) { const baseAxis = groupItem.axis; const seriesModels = groupItem.seriesModels; const seriesCount = seriesModels.length; const boxWidthList: number[] = groupItem.boxWidthList = []; const boxOffsetList: number[] = groupItem.boxOffsetList = []; const boundList: number[][] = []; let bandWidth: number; if (baseAxis.type === 'category') { bandWidth = baseAxis.getBandWidth(); } else { let maxDataCount = 0; each(seriesModels, function (seriesModel) { maxDataCount = Math.max(maxDataCount, seriesModel.getData().count()); }); const extent = baseAxis.getExtent(); bandWidth = Math.abs(extent[1] - extent[0]) / maxDataCount; } each(seriesModels, function (seriesModel) { let boxWidthBound = seriesModel.get('boxWidth'); if (!zrUtil.isArray(boxWidthBound)) { boxWidthBound = [boxWidthBound, boxWidthBound]; } boundList.push([ parsePercent(boxWidthBound[0], bandWidth) || 0, parsePercent(boxWidthBound[1], bandWidth) || 0 ]); }); const availableWidth = bandWidth * 0.8 - 2; const boxGap = availableWidth / seriesCount * 0.3; const boxWidth = (availableWidth - boxGap * (seriesCount - 1)) / seriesCount; let base = boxWidth / 2 - availableWidth / 2; each(seriesModels, function (seriesModel, idx) { boxOffsetList.push(base); base += boxGap + boxWidth; boxWidthList.push( Math.min(Math.max(boxWidth, boundList[idx][0]), boundList[idx][1]) ); }); } /** * Calculate points location for each series. */ function layoutSingleSeries(seriesModel: BoxplotSeriesModel, offset: number, boxWidth: number) { const coordSys = seriesModel.coordinateSystem; const data = seriesModel.getData(); const halfWidth = boxWidth / 2; const cDimIdx = seriesModel.get('layout') === 'horizontal' ? 0 : 1; const vDimIdx = 1 - cDimIdx; const coordDims = ['x', 'y']; const cDim = data.mapDimension(coordDims[cDimIdx]); const vDims = data.mapDimensionsAll(coordDims[vDimIdx]); if (cDim == null || vDims.length < 5) { return; } for (let dataIndex = 0; dataIndex < data.count(); dataIndex++) { const axisDimVal = data.get(cDim, dataIndex) as number; const median = getPoint(axisDimVal, vDims[2], dataIndex); const end1 = getPoint(axisDimVal, vDims[0], dataIndex); const end2 = getPoint(axisDimVal, vDims[1], dataIndex); const end4 = getPoint(axisDimVal, vDims[3], dataIndex); const end5 = getPoint(axisDimVal, vDims[4], dataIndex); const ends: number[][] = []; addBodyEnd(ends, end2, false); addBodyEnd(ends, end4, true); ends.push(end1, end2, end5, end4); layEndLine(ends, end1); layEndLine(ends, end5); layEndLine(ends, median); data.setItemLayout(dataIndex, { initBaseline: median[vDimIdx], ends: ends } as BoxplotItemLayout); } function getPoint(axisDimVal: number, dim: string, dataIndex: number) { const val = data.get(dim, dataIndex) as number; const p = []; p[cDimIdx] = axisDimVal; p[vDimIdx] = val; let point; if (isNaN(axisDimVal) || isNaN(val)) { point = [NaN, NaN]; } else { point = coordSys.dataToPoint(p); point[cDimIdx] += offset; } return point; } function addBodyEnd(ends: number[][], point: number[], start?: boolean) { const point1 = point.slice(); const point2 = point.slice(); point1[cDimIdx] += halfWidth; point2[cDimIdx] -= halfWidth; start ? ends.push(point1, point2) : ends.push(point2, point1); } function layEndLine(ends: number[][], endCenter: number[]) { const from = endCenter.slice(); const to = endCenter.slice(); from[cDimIdx] -= halfWidth; to[cDimIdx] += halfWidth; ends.push(from, to); } }