|  | /* | 
|  | * 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 VisualMapView from './VisualMapView'; | 
|  | import * as graphic from '../../util/graphic'; | 
|  | import {createSymbol} from '../../util/symbol'; | 
|  | import * as layout from '../../util/layout'; | 
|  | import * as helper from './helper'; | 
|  | import PiecewiseModel from './PiecewiseModel'; | 
|  | import { TextAlign } from 'zrender/src/core/types'; | 
|  | import { VisualMappingOption } from '../../visual/VisualMapping'; | 
|  | import { createTextStyle } from '../../label/labelStyle'; | 
|  |  | 
|  | class PiecewiseVisualMapView extends VisualMapView { | 
|  |  | 
|  | static type = 'visualMap.piecewise' as const; | 
|  |  | 
|  | type = PiecewiseVisualMapView.type; | 
|  |  | 
|  | visualMapModel: PiecewiseModel; | 
|  |  | 
|  | protected doRender() { | 
|  | const thisGroup = this.group; | 
|  |  | 
|  | thisGroup.removeAll(); | 
|  |  | 
|  | const visualMapModel = this.visualMapModel; | 
|  | const textGap = visualMapModel.get('textGap'); | 
|  | const textStyleModel = visualMapModel.textStyleModel; | 
|  | const textFont = textStyleModel.getFont(); | 
|  | const textFill = textStyleModel.getTextColor(); | 
|  | const itemAlign = this._getItemAlign(); | 
|  | const itemSize = visualMapModel.itemSize; | 
|  | const viewData = this._getViewData(); | 
|  | const endsText = viewData.endsText; | 
|  | const showLabel = zrUtil.retrieve(visualMapModel.get('showLabel', true), !endsText); | 
|  |  | 
|  | endsText && this._renderEndsText( | 
|  | thisGroup, endsText[0], itemSize, showLabel, itemAlign | 
|  | ); | 
|  |  | 
|  | zrUtil.each(viewData.viewPieceList, function (item: typeof viewData.viewPieceList[number]) { | 
|  | const piece = item.piece; | 
|  |  | 
|  | const itemGroup = new graphic.Group(); | 
|  | itemGroup.onclick = zrUtil.bind(this._onItemClick, this, piece); | 
|  |  | 
|  | this._enableHoverLink(itemGroup, item.indexInModelPieceList); | 
|  |  | 
|  | // TODO Category | 
|  | const representValue = visualMapModel.getRepresentValue(piece) as number; | 
|  |  | 
|  | this._createItemSymbol( | 
|  | itemGroup, representValue, [0, 0, itemSize[0], itemSize[1]] | 
|  | ); | 
|  |  | 
|  | if (showLabel) { | 
|  | const visualState = this.visualMapModel.getValueState(representValue); | 
|  |  | 
|  | itemGroup.add(new graphic.Text({ | 
|  | style: { | 
|  | x: itemAlign === 'right' ? -textGap : itemSize[0] + textGap, | 
|  | y: itemSize[1] / 2, | 
|  | text: piece.text, | 
|  | verticalAlign: 'middle', | 
|  | align: itemAlign as TextAlign, | 
|  | font: textFont, | 
|  | fill: textFill, | 
|  | opacity: visualState === 'outOfRange' ? 0.5 : 1 | 
|  | } | 
|  | })); | 
|  | } | 
|  |  | 
|  | thisGroup.add(itemGroup); | 
|  | }, this); | 
|  |  | 
|  | endsText && this._renderEndsText( | 
|  | thisGroup, endsText[1], itemSize, showLabel, itemAlign | 
|  | ); | 
|  |  | 
|  | layout.box( | 
|  | visualMapModel.get('orient'), thisGroup, visualMapModel.get('itemGap') | 
|  | ); | 
|  |  | 
|  | this.renderBackground(thisGroup); | 
|  |  | 
|  | this.positionGroup(thisGroup); | 
|  | } | 
|  |  | 
|  | private _enableHoverLink(itemGroup: graphic.Group, pieceIndex: number) { | 
|  | itemGroup | 
|  | .on('mouseover', () => onHoverLink('highlight')) | 
|  | .on('mouseout', () => onHoverLink('downplay')); | 
|  |  | 
|  | const onHoverLink = (method?: 'highlight' | 'downplay') => { | 
|  | const visualMapModel = this.visualMapModel; | 
|  |  | 
|  | // TODO: TYPE More detailed action types | 
|  | visualMapModel.option.hoverLink && this.api.dispatchAction({ | 
|  | type: method, | 
|  | batch: helper.makeHighDownBatch( | 
|  | visualMapModel.findTargetDataIndices(pieceIndex), | 
|  | visualMapModel | 
|  | ) | 
|  | }); | 
|  | }; | 
|  | } | 
|  |  | 
|  | private _getItemAlign(): helper.ItemAlign { | 
|  | const visualMapModel = this.visualMapModel; | 
|  | const modelOption = visualMapModel.option; | 
|  |  | 
|  | if (modelOption.orient === 'vertical') { | 
|  | return helper.getItemAlign( | 
|  | visualMapModel, this.api, visualMapModel.itemSize | 
|  | ); | 
|  | } | 
|  | else { // horizontal, most case left unless specifying right. | 
|  | let align = modelOption.align; | 
|  | if (!align || align === 'auto') { | 
|  | align = 'left'; | 
|  | } | 
|  | return align; | 
|  | } | 
|  | } | 
|  |  | 
|  | private _renderEndsText( | 
|  | group: graphic.Group, | 
|  | text: string, | 
|  | itemSize: number[], | 
|  | showLabel: boolean, | 
|  | itemAlign: helper.ItemAlign | 
|  | ) { | 
|  | if (!text) { | 
|  | return; | 
|  | } | 
|  |  | 
|  | const itemGroup = new graphic.Group(); | 
|  | const textStyleModel = this.visualMapModel.textStyleModel; | 
|  |  | 
|  | itemGroup.add(new graphic.Text({ | 
|  | style: createTextStyle(textStyleModel, { | 
|  | x: showLabel ? (itemAlign === 'right' ? itemSize[0] : 0) : itemSize[0] / 2, | 
|  | y: itemSize[1] / 2, | 
|  | verticalAlign: 'middle', | 
|  | align: showLabel ? (itemAlign as TextAlign) : 'center', | 
|  | text | 
|  | }) | 
|  | })); | 
|  |  | 
|  | group.add(itemGroup); | 
|  | } | 
|  |  | 
|  | /** | 
|  | * @private | 
|  | * @return {Object} {peiceList, endsText} The order is the same as screen pixel order. | 
|  | */ | 
|  | private _getViewData() { | 
|  | const visualMapModel = this.visualMapModel; | 
|  |  | 
|  | const viewPieceList = zrUtil.map(visualMapModel.getPieceList(), function (piece, index) { | 
|  | return {piece: piece, indexInModelPieceList: index}; | 
|  | }); | 
|  | let endsText = visualMapModel.get('text'); | 
|  |  | 
|  | // Consider orient and inverse. | 
|  | const orient = visualMapModel.get('orient'); | 
|  | const inverse = visualMapModel.get('inverse'); | 
|  |  | 
|  | // Order of model pieceList is always [low, ..., high] | 
|  | if (orient === 'horizontal' ? inverse : !inverse) { | 
|  | viewPieceList.reverse(); | 
|  | } | 
|  | // Origin order of endsText is [high, low] | 
|  | else if (endsText) { | 
|  | endsText = endsText.slice().reverse(); | 
|  | } | 
|  |  | 
|  | return {viewPieceList: viewPieceList, endsText: endsText}; | 
|  | } | 
|  |  | 
|  | private _createItemSymbol( | 
|  | group: graphic.Group, | 
|  | representValue: number, | 
|  | shapeParam: number[] | 
|  | ) { | 
|  | group.add(createSymbol( | 
|  | // symbol will be string | 
|  | this.getControllerVisual(representValue, 'symbol') as string, | 
|  | shapeParam[0], shapeParam[1], shapeParam[2], shapeParam[3], | 
|  | // color will be string | 
|  | this.getControllerVisual(representValue, 'color') as string | 
|  | )); | 
|  | } | 
|  |  | 
|  | private _onItemClick( | 
|  | piece: VisualMappingOption['pieceList'][number] | 
|  | ) { | 
|  | const visualMapModel = this.visualMapModel; | 
|  | const option = visualMapModel.option; | 
|  | const selectedMode = option.selectedMode; | 
|  | if (!selectedMode) { | 
|  | return; | 
|  | } | 
|  | const selected = zrUtil.clone(option.selected); | 
|  | const newKey = visualMapModel.getSelectedMapKey(piece); | 
|  |  | 
|  | if (selectedMode === 'single' || selectedMode === true) { | 
|  | selected[newKey] = true; | 
|  | zrUtil.each(selected, function (o, key) { | 
|  | selected[key] = key === newKey; | 
|  | }); | 
|  | } | 
|  | else { | 
|  | selected[newKey] = !selected[newKey]; | 
|  | } | 
|  |  | 
|  | this.api.dispatchAction({ | 
|  | type: 'selectDataRange', | 
|  | from: this.uid, | 
|  | visualMapId: this.visualMapModel.id, | 
|  | selected: selected | 
|  | }); | 
|  | } | 
|  | } | 
|  |  | 
|  | export default PiecewiseVisualMapView; |