| /* | 
 | * 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 {ECPolygon} from '../line/poly'; | 
 | import * as graphic from '../../util/graphic'; | 
 | import { setStatesStylesFromModel, toggleHoverEmphasis } from '../../util/states'; | 
 | import {setLabelStyle, getLabelStatesModels} from '../../label/labelStyle'; | 
 | import {bind} from 'zrender/src/core/util'; | 
 | import DataDiffer from '../../data/DataDiffer'; | 
 | import ChartView from '../../view/Chart'; | 
 | import ThemeRiverSeriesModel from './ThemeRiverSeries'; | 
 | import GlobalModel from '../../model/Global'; | 
 | import ExtensionAPI from '../../core/ExtensionAPI'; | 
 | import { RectLike } from 'zrender/src/core/BoundingRect'; | 
 | import { ColorString } from '../../util/types'; | 
 | import { saveOldStyle } from '../../animation/basicTransition'; | 
 |  | 
 | type LayerSeries = ReturnType<ThemeRiverSeriesModel['getLayerSeries']>; | 
 |  | 
 | class ThemeRiverView extends ChartView { | 
 |  | 
 |     static readonly type = 'themeRiver'; | 
 |     readonly type = ThemeRiverView.type; | 
 |  | 
 |     private _layersSeries: LayerSeries; | 
 |     private _layers: graphic.Group[] = []; | 
 |  | 
 |     render(seriesModel: ThemeRiverSeriesModel, ecModel: GlobalModel, api: ExtensionAPI) { | 
 |         const data = seriesModel.getData(); | 
 |         const self = this; | 
 |  | 
 |         const group = this.group; | 
 |  | 
 |         const layersSeries = seriesModel.getLayerSeries(); | 
 |  | 
 |         const layoutInfo = data.getLayout('layoutInfo'); | 
 |         const rect = layoutInfo.rect; | 
 |         const boundaryGap = layoutInfo.boundaryGap; | 
 |  | 
 |         group.x = 0; | 
 |         group.y = rect.y + boundaryGap[0]; | 
 |  | 
 |         function keyGetter(item: LayerSeries[number]) { | 
 |             return item.name; | 
 |         } | 
 |         const dataDiffer = new DataDiffer( | 
 |             this._layersSeries || [], layersSeries, | 
 |             keyGetter, keyGetter | 
 |         ); | 
 |  | 
 |         const newLayersGroups: graphic.Group[] = []; | 
 |  | 
 |         dataDiffer | 
 |             .add(bind(process, this, 'add')) | 
 |             .update(bind(process, this, 'update')) | 
 |             .remove(bind(process, this, 'remove')) | 
 |             .execute(); | 
 |  | 
 |         function process(status: 'add' | 'update' | 'remove', idx: number, oldIdx?: number) { | 
 |             const oldLayersGroups = self._layers; | 
 |             if (status === 'remove') { | 
 |                 group.remove(oldLayersGroups[idx]); | 
 |                 return; | 
 |             } | 
 |             const points0: number[] = []; | 
 |             const points1: number[] = []; | 
 |             let style; | 
 |             const indices = layersSeries[idx].indices; | 
 |             let j = 0; | 
 |             for (; j < indices.length; j++) { | 
 |                 const layout = data.getItemLayout(indices[j]); | 
 |                 const x = layout.x; | 
 |                 const y0 = layout.y0; | 
 |                 const y = layout.y; | 
 |  | 
 |                 points0.push(x, y0); | 
 |                 points1.push(x, y0 + y); | 
 |  | 
 |                 style = data.getItemVisual(indices[j], 'style'); | 
 |             } | 
 |  | 
 |             let polygon: ECPolygon; | 
 |             const textLayout = data.getItemLayout(indices[0]); | 
 |             const labelModel = seriesModel.getModel('label'); | 
 |             const margin = labelModel.get('margin'); | 
 |             const emphasisModel = seriesModel.getModel('emphasis'); | 
 |  | 
 |             if (status === 'add') { | 
 |                 const layerGroup = newLayersGroups[idx] = new graphic.Group(); | 
 |                 polygon = new ECPolygon({ | 
 |                     shape: { | 
 |                         points: points0, | 
 |                         stackedOnPoints: points1, | 
 |                         smooth: 0.4, | 
 |                         stackedOnSmooth: 0.4, | 
 |                         smoothConstraint: false | 
 |                     }, | 
 |                     z2: 0 | 
 |                 }); | 
 |                 layerGroup.add(polygon); | 
 |                 group.add(layerGroup); | 
 |  | 
 |                 if (seriesModel.isAnimationEnabled()) { | 
 |                     polygon.setClipPath(createGridClipShape(polygon.getBoundingRect(), seriesModel, function () { | 
 |                         polygon.removeClipPath(); | 
 |                     })); | 
 |                 } | 
 |             } | 
 |             else { | 
 |                 const layerGroup = oldLayersGroups[oldIdx]; | 
 |                 polygon = layerGroup.childAt(0) as ECPolygon; | 
 |                 group.add(layerGroup); | 
 |  | 
 |                 newLayersGroups[idx] = layerGroup; | 
 |  | 
 |                 graphic.updateProps(polygon, { | 
 |                     shape: { | 
 |                         points: points0, | 
 |                         stackedOnPoints: points1 | 
 |                     } | 
 |                 }, seriesModel); | 
 |  | 
 |                 saveOldStyle(polygon); | 
 |             } | 
 |  | 
 |             setLabelStyle(polygon, getLabelStatesModels(seriesModel), { | 
 |                 labelDataIndex: indices[j - 1], | 
 |                 defaultText: data.getName(indices[j - 1]), | 
 |                 inheritColor: style.fill as ColorString | 
 |             }, { | 
 |                 normal: { | 
 |                     verticalAlign: 'middle' | 
 |                     // align: 'right' | 
 |                 } | 
 |             }); | 
 |             polygon.setTextConfig({ | 
 |                 position: null, | 
 |                 local: true | 
 |             }); | 
 |  | 
 |             const labelEl = polygon.getTextContent(); | 
 |             // TODO More label position options. | 
 |             if (labelEl) { | 
 |                 labelEl.x = textLayout.x - margin; | 
 |                 labelEl.y = textLayout.y0 + textLayout.y / 2; | 
 |             } | 
 |  | 
 |             polygon.useStyle(style); | 
 |  | 
 |             data.setItemGraphicEl(idx, polygon); | 
 |  | 
 |             setStatesStylesFromModel(polygon, seriesModel); | 
 |             toggleHoverEmphasis( | 
 |                 polygon, emphasisModel.get('focus'), emphasisModel.get('blurScope'), emphasisModel.get('disabled') | 
 |             ); | 
 |         } | 
 |  | 
 |         this._layersSeries = layersSeries; | 
 |         this._layers = newLayersGroups; | 
 |     } | 
 | }; | 
 |  | 
 | // add animation to the view | 
 | function createGridClipShape(rect: RectLike, seriesModel: ThemeRiverSeriesModel, cb: () => void) { | 
 |     const rectEl = new graphic.Rect({ | 
 |         shape: { | 
 |             x: rect.x - 10, | 
 |             y: rect.y - 10, | 
 |             width: 0, | 
 |             height: rect.height + 20 | 
 |         } | 
 |     }); | 
 |     graphic.initProps(rectEl, { | 
 |         shape: { | 
 |             x: rect.x - 50, | 
 |             width: rect.width + 100, | 
 |             height: rect.height + 20 | 
 |         } | 
 |     }, seriesModel, cb); | 
 |  | 
 |     return rectEl; | 
 | } | 
 |  | 
 | export default ThemeRiverView; |