blob: f0049defc86041f5496a89b309cb0fec6ae034c4 [file] [log] [blame]
/*
* 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 createListSimply from '../helper/createListSimply';
import SeriesModel from '../../model/Series';
import geoSourceManager from '../../coord/geo/geoSourceManager';
import {makeSeriesEncodeForNameBased} from '../../data/helper/sourceHelper';
import {
SeriesOption,
BoxLayoutOptionMixin,
SeriesEncodeOptionMixin,
OptionDataItemObject,
OptionDataValueNumeric,
ParsedValue,
SeriesOnGeoOptionMixin,
StatesOptionMixin,
SeriesLabelOption
} from '../../util/types';
import { Dictionary } from 'zrender/src/core/types';
import GeoModel, { GeoCommonOptionMixin, GeoItemStyleOption } from '../../coord/geo/GeoModel';
import List from '../../data/List';
import Model from '../../model/Model';
import Geo from '../../coord/geo/Geo';
import { createTooltipMarkup } from '../../component/tooltip/tooltipMarkup';
import {createSymbol, ECSymbol} from '../../util/symbol';
import {LegendIconParams} from '../../component/legend/LegendModel';
import {Group} from '../../util/graphic';
export interface MapStateOption {
itemStyle?: GeoItemStyleOption
label?: SeriesLabelOption
}
export interface MapDataItemOption extends MapStateOption, StatesOptionMixin<MapStateOption>,
OptionDataItemObject<OptionDataValueNumeric> {
cursor?: string
}
export type MapValueCalculationType = 'sum' | 'average' | 'min' | 'max';
export interface MapSeriesOption extends
SeriesOption<MapStateOption>, MapStateOption,
GeoCommonOptionMixin,
// If `geoIndex` is not specified, a exclusive geo will be
// created. Otherwise use the specified geo component, and
// `map` and `mapType` are ignored.
SeriesOnGeoOptionMixin,
BoxLayoutOptionMixin,
SeriesEncodeOptionMixin {
type?: 'map'
coordinateSystem?: string;
silent?: boolean;
// FIXME:TS add marker types
markLine?: any;
markPoint?: any;
markArea?: any;
mapValueCalculation?: MapValueCalculationType;
showLegendSymbol?: boolean;
// @deprecated. Only for echarts2 backward compat.
geoCoord?: Dictionary<number[]>;
data?: (OptionDataValueNumeric | OptionDataValueNumeric[] | MapDataItemOption)[]
nameProperty?: string;
}
class MapSeries extends SeriesModel<MapSeriesOption> {
static type = 'series.map' as const;
type = MapSeries.type;
static dependencies = ['geo'];
static layoutMode = 'box' as const;
coordinateSystem: Geo;
// -----------------
// Injected outside
originalData: List;
mainSeries: MapSeries;
// Only first map series of same mapType will drawMap.
needsDrawMap: boolean = false;
// Group of all map series with same mapType
seriesGroup: MapSeries[] = [];
getInitialData(this: MapSeries, option: MapSeriesOption): List {
const data = createListSimply(this, {
coordDimensions: ['value'],
encodeDefaulter: zrUtil.curry(makeSeriesEncodeForNameBased, this)
});
const dataNameMap = zrUtil.createHashMap();
const toAppendNames = [] as string[];
for (let i = 0, len = data.count(); i < len; i++) {
const name = data.getName(i);
dataNameMap.set(name, true);
}
const geoSource = geoSourceManager.load(this.getMapType(), this.option.nameMap, this.option.nameProperty);
zrUtil.each(geoSource.regions, function (region) {
const name = region.name;
if (!dataNameMap.get(name)) {
toAppendNames.push(name);
}
});
// Complete data with missing regions. The consequent processes (like visual
// map and render) can not be performed without a "full data". For example,
// find `dataIndex` by name.
data.appendValues([], toAppendNames);
return data;
}
/**
* If no host geo model, return null, which means using a
* inner exclusive geo model.
*/
getHostGeoModel(): GeoModel {
const geoIndex = this.option.geoIndex;
return geoIndex != null
? this.ecModel.getComponent('geo', geoIndex) as GeoModel
: null;
}
getMapType(): string {
return (this.getHostGeoModel() || this).option.map;
}
// _fillOption(option, mapName) {
// Shallow clone
// option = zrUtil.extend({}, option);
// option.data = geoCreator.getFilledRegions(option.data, mapName, option.nameMap);
// return option;
// }
getRawValue(dataIndex: number): ParsedValue {
// Use value stored in data instead because it is calculated from multiple series
// FIXME Provide all value of multiple series ?
const data = this.getData();
return data.get(data.mapDimension('value'), dataIndex);
}
/**
* Get model of region
*/
getRegionModel(regionName: string): Model<MapDataItemOption> {
const data = this.getData();
return data.getItemModel(data.indexOfName(regionName));
}
/**
* Map tooltip formatter
*/
formatTooltip(
dataIndex: number,
multipleSeries: boolean,
dataType: string
) {
// FIXME orignalData and data is a bit confusing
const data = this.getData();
const value = this.getRawValue(dataIndex);
const name = data.getName(dataIndex);
const seriesGroup = this.seriesGroup;
const seriesNames = [];
for (let i = 0; i < seriesGroup.length; i++) {
const otherIndex = seriesGroup[i].originalData.indexOfName(name);
const valueDim = data.mapDimension('value');
if (!isNaN(seriesGroup[i].originalData.get(valueDim, otherIndex) as number)) {
seriesNames.push(seriesGroup[i].name);
}
}
return createTooltipMarkup('section', {
header: seriesNames.join(', '),
noHeader: !seriesNames.length,
blocks: [createTooltipMarkup('nameValue', {
name: name, value: value
})]
});
}
getTooltipPosition = function (this: MapSeries, dataIndex: number): number[] {
if (dataIndex != null) {
const name = this.getData().getName(dataIndex);
const geo = this.coordinateSystem;
const region = geo.getRegion(name);
return region && geo.dataToPoint(region.getCenter());
}
};
setZoom(zoom: number): void {
this.option.zoom = zoom;
}
setCenter(center: number[]): void {
this.option.center = center;
}
getLegendIcon(opt: LegendIconParams): ECSymbol | Group {
const iconType = opt.icon || 'roundRect';
const icon = createSymbol(
iconType,
0,
0,
opt.itemWidth,
opt.itemHeight,
opt.itemStyle.fill
);
icon.setStyle(opt.itemStyle);
// Map do not use itemStyle.borderWidth as border width
icon.style.stroke = 'none';
// No rotation because no series visual symbol for map
if (iconType.indexOf('empty') > -1) {
icon.style.stroke = icon.style.fill;
icon.style.fill = '#fff';
icon.style.lineWidth = 2;
}
return icon;
}
static defaultOption: MapSeriesOption = {
// 一级层叠
zlevel: 0,
// 二级层叠
z: 2,
coordinateSystem: 'geo',
// map should be explicitly specified since ec3.
map: '',
// If `geoIndex` is not specified, a exclusive geo will be
// created. Otherwise use the specified geo component, and
// `map` and `mapType` are ignored.
// geoIndex: 0,
// 'center' | 'left' | 'right' | 'x%' | {number}
left: 'center',
// 'center' | 'top' | 'bottom' | 'x%' | {number}
top: 'center',
// right
// bottom
// width:
// height
// Aspect is width / height. Inited to be geoJson bbox aspect
// This parameter is used for scale this aspect
// Default value:
// for geoSVG source: 1,
// for geoJSON source: 0.75.
aspectScale: null,
///// Layout with center and size
// If you wan't to put map in a fixed size box with right aspect ratio
// This two properties may more conveninet
// layoutCenter: [50%, 50%]
// layoutSize: 100
showLegendSymbol: true,
// Define left-top, right-bottom coords to control view
// For example, [ [180, 90], [-180, -90] ],
// higher priority than center and zoom
boundingCoords: null,
// Default on center of map
center: null,
zoom: 1,
scaleLimit: null,
selectedMode: true,
label: {
show: false,
color: '#000'
},
// scaleLimit: null,
itemStyle: {
borderWidth: 0.5,
borderColor: '#444',
areaColor: '#eee'
},
emphasis: {
label: {
show: true,
color: 'rgb(100,0,0)'
},
itemStyle: {
areaColor: 'rgba(255,215,0,0.8)'
}
},
select: {
label: {
show: true,
color: 'rgb(100,0,0)'
},
itemStyle: {
color: 'rgba(255,215,0,0.8)'
}
},
nameProperty: 'name'
};
}
export default MapSeries;