Spaces:
Running
Running
/* | |
* 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 * as numberUtil from '../../util/number'; | |
import GlobalModel from '../../model/Global'; | |
import ExtensionAPI from '../../core/ExtensionAPI'; | |
import ThemeRiverSeriesModel, { ThemeRiverSeriesOption } from './ThemeRiverSeries'; | |
import { RectLike } from 'zrender/src/core/BoundingRect'; | |
import SeriesData from '../../data/SeriesData'; | |
export interface ThemeRiverLayoutInfo { | |
rect: RectLike | |
boundaryGap: ThemeRiverSeriesOption['boundaryGap'] | |
} | |
export default function themeRiverLayout(ecModel: GlobalModel, api: ExtensionAPI) { | |
ecModel.eachSeriesByType('themeRiver', function (seriesModel: ThemeRiverSeriesModel) { | |
const data = seriesModel.getData(); | |
const single = seriesModel.coordinateSystem; | |
const layoutInfo = {} as ThemeRiverLayoutInfo; | |
// use the axis boundingRect for view | |
const rect = single.getRect(); | |
layoutInfo.rect = rect; | |
const boundaryGap = seriesModel.get('boundaryGap'); | |
const axis = single.getAxis(); | |
layoutInfo.boundaryGap = boundaryGap; | |
if (axis.orient === 'horizontal') { | |
boundaryGap[0] = numberUtil.parsePercent(boundaryGap[0], rect.height); | |
boundaryGap[1] = numberUtil.parsePercent(boundaryGap[1], rect.height); | |
const height = rect.height - boundaryGap[0] - boundaryGap[1]; | |
doThemeRiverLayout(data, seriesModel, height); | |
} | |
else { | |
boundaryGap[0] = numberUtil.parsePercent(boundaryGap[0], rect.width); | |
boundaryGap[1] = numberUtil.parsePercent(boundaryGap[1], rect.width); | |
const width = rect.width - boundaryGap[0] - boundaryGap[1]; | |
doThemeRiverLayout(data, seriesModel, width); | |
} | |
data.setLayout('layoutInfo', layoutInfo); | |
}); | |
} | |
/** | |
* The layout information about themeriver | |
* | |
* @param data data in the series | |
* @param seriesModel the model object of themeRiver series | |
* @param height value used to compute every series height | |
*/ | |
function doThemeRiverLayout( | |
data: SeriesData<ThemeRiverSeriesModel>, | |
seriesModel: ThemeRiverSeriesModel, | |
height: number | |
) { | |
if (!data.count()) { | |
return; | |
} | |
const coordSys = seriesModel.coordinateSystem; | |
// the data in each layer are organized into a series. | |
const layerSeries = seriesModel.getLayerSeries(); | |
// the points in each layer. | |
const timeDim = data.mapDimension('single'); | |
const valueDim = data.mapDimension('value'); | |
const layerPoints = zrUtil.map(layerSeries, function (singleLayer) { | |
return zrUtil.map(singleLayer.indices, function (idx) { | |
const pt = coordSys.dataToPoint(data.get(timeDim, idx)); | |
pt[1] = data.get(valueDim, idx) as number; | |
return pt; | |
}); | |
}); | |
const base = computeBaseline(layerPoints); | |
const baseLine = base.y0; | |
const ky = height / base.max; | |
// set layout information for each item. | |
const n = layerSeries.length; | |
const m = layerSeries[0].indices.length; | |
let baseY0; | |
for (let j = 0; j < m; ++j) { | |
baseY0 = baseLine[j] * ky; | |
data.setItemLayout(layerSeries[0].indices[j], { | |
layerIndex: 0, | |
x: layerPoints[0][j][0], | |
y0: baseY0, | |
y: layerPoints[0][j][1] * ky | |
}); | |
for (let i = 1; i < n; ++i) { | |
baseY0 += layerPoints[i - 1][j][1] * ky; | |
data.setItemLayout(layerSeries[i].indices[j], { | |
layerIndex: i, | |
x: layerPoints[i][j][0], | |
y0: baseY0, | |
y: layerPoints[i][j][1] * ky | |
}); | |
} | |
} | |
} | |
/** | |
* Compute the baseLine of the rawdata | |
* Inspired by Lee Byron's paper Stacked Graphs - Geometry & Aesthetics | |
* | |
* @param data the points in each layer | |
*/ | |
function computeBaseline(data: number[][][]) { | |
const layerNum = data.length; | |
const pointNum = data[0].length; | |
const sums = []; | |
const y0 = []; | |
let max = 0; | |
for (let i = 0; i < pointNum; ++i) { | |
let temp = 0; | |
for (let j = 0; j < layerNum; ++j) { | |
temp += data[j][i][1]; | |
} | |
if (temp > max) { | |
max = temp; | |
} | |
sums.push(temp); | |
} | |
for (let k = 0; k < pointNum; ++k) { | |
y0[k] = (max - sums[k]) / 2; | |
} | |
max = 0; | |
for (let l = 0; l < pointNum; ++l) { | |
const sum = sums[l] + y0[l]; | |
if (sum > max) { | |
max = sum; | |
} | |
} | |
return { | |
y0, | |
max | |
}; | |
} | |