|  | /* | 
|  | * 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); | 
|  | } | 
|  | } |