Merge pull request #166 from antvis/clusterMarker

feat(component): 新增cluster marker
This commit is contained in:
@thinkinggis 2020-01-17 12:17:03 +08:00 committed by GitHub
commit 2ad0549210
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
21 changed files with 771 additions and 7 deletions

View File

@ -68,7 +68,7 @@ scene.addMarker(marker);
```javascript
var el = document.createElement('label');
el.className = 'lableclass';
el.className = 'labelclass';
el.textContent = data[i].v;
el.style.background = getColor(data[i].v);
new L7.Marker({

View File

@ -71,7 +71,7 @@ Marker
```javascript
var el = document.createElement('label');
el.className = 'lableclass';
el.className = 'labelclass';
el.textContent = data[i].v;
el.style.background = getColor(data[i].v);
new L7.Marker({

View File

@ -0,0 +1,97 @@
---
title: Marker Layer
order: 3
---
MarkerLayer 不同于 PointLayer 图层
**技术差异**
- MarkerLayer 地图元素采用 Dom 元素绘制
- PointLayer 通过 WebGL 绘制元素。
**功能差异**
- MarkerLayer 元素的自定义性比较强,任何 HTML+ CSS 的组合都可可以绘制在地图上。
- PointLayer 自定义性比较弱,实现成本比较高,优势可以绘制大量的数据,性能比交互。
## 使用
```javascript
import { Marker, MarkerLayer } from '@antv/l7';
```
### 构造函数
```javascript
const markerLayer = new MarkerLayer(option);
// 调用 addMarker方法 将多个Marker添加到Layer
scene.addMarkerLayer(markerLayer);
```
#### option
- cluster 是部分聚合 `boolean` 默认 `false`
- clusterOption 聚合配置
- cluster 是部分聚合
- element `function`
后续会增加更多配置项目
### 方法
#### addMarker
参数
- marker `Marker` 需要添加的 Marker
添加 Marker
通过 Marker 对象实例化一个 Marker
```javascript
const marker = new Marker().setLnglat(); // 添加进Marker必须设置经纬度才能添加
markerLayer.addMarker(marker);
```
#### removeMarker
从 MarkerLayer 移除 Marker
#### getMarkers
获取 MarkerLayer 中的所有 Marker
#### clear
清除掉所有的 Marker
####
### Scene
#### addMarkerLayer
添加 MarkerLayer
```javascript
scene.addMarkerLayer(layer);
```
#### removeMarkerLayer
移除 MarkerLayer
```javascript
scene.removeMarkerLayer(layer);
```
### demo 地址
[markerLayer ](../../../examples/point/marker#markerlayer)
[markerLayer 聚合](../../../examples/point/marker#clustermarker)

View File

@ -0,0 +1,97 @@
---
title: Marker 图层
order: 3
---
MarkerLayer 不同于 PointLayer 图层
**技术差异**
- MarkerLayer 地图元素采用 Dom 元素绘制
- PointLayer 通过 WebGL 绘制元素。
**功能差异**
- MarkerLayer 元素的自定义性比较强,任何 HTML+ CSS 的组合都可可以绘制在地图上。
- PointLayer 自定义性比较弱,实现成本比较高,优势可以绘制大量的数据,性能比交互。
## 使用
```javascript
import { Marker, MarkerLayer } from '@antv/l7';
```
### 构造函数
```javascript
const markerLayer = new MarkerLayer(option);
// 调用 addMarker方法 将多个Marker添加到Layer
scene.addMarkerLayer(markerLayer);
```
#### option
- cluster 是部分聚合 `boolean` 默认 `false`
- clusterOption 聚合配置
- cluster 是部分聚合
- element `function`
后续会增加更多配置项目
### 方法
#### addMarker
参数
- marker `Marker` 需要添加的 Marker
添加 Marker
通过 Marker 对象实例化一个 Marker
```javascript
const marker = new Marker().setLnglat(); // 添加进Marker必须设置经纬度才能添加
markerLayer.addMarker(marker);
```
#### removeMarker
从 MarkerLayer 移除 Marker
#### getMarkers
获取 MarkerLayer 中的所有 Marker
#### clear
清除掉所有的 Marker
####
### Scene
#### addMarkerLayer
添加 MarkerLayer
```javascript
scene.addMarkerLayer(layer);
```
#### removeMarkerLayer
移除 MarkerLayer
```javascript
scene.removeMarkerLayer(layer);
```
### demo 地址
[markerLayer ](../../../examples/point/marker#markerlayer)
[markerLayer 聚合](../../../examples/point/marker#clustermarker)

View File

@ -7,3 +7,5 @@ order: 7
## 使用
[Marker 文档](../../component)
[MarkerLayer 文档](../../component/markerLayer)

View File

@ -5,4 +5,7 @@ order: 7
可自定义点符号通过自定义dom实现地图标注富文本、动态点状符号都可用于地图上信息的标记。
## 使用
[Marker 文档](../../component)
[MarkerLayer 文档](../../component/markerLayer)

View File

@ -0,0 +1,36 @@
import { Scene, Marker, MarkerLayer } from '@antv/l7';
import { GaodeMap } from '@antv/l7-maps';
const scene = new Scene({
id: 'map',
map: new GaodeMap({
style: 'light',
center: [ 105.790327, 36.495636 ],
pitch: 0,
zoom: 4
})
});
addMarkers();
scene.render();
function addMarkers() {
fetch(
'https://gw.alipayobjects.com/os/basement_prod/d3564b06-670f-46ea-8edb-842f7010a7c6.json'
)
.then(res => res.json())
.then(nodes => {
const markerLayer = new MarkerLayer({
cluster: true
});
for (let i = 0; i < nodes.features.length; i++) {
const { coordinates } = nodes.features[i].geometry;
const marker = new Marker().setLnglat({
lng: coordinates[0],
lat: coordinates[1]
});
markerLayer.addMarker(marker);
}
scene.addMarkerLayer(markerLayer);
scene.addMarkerLayer(markerLayer);
});
}

View File

@ -24,7 +24,7 @@ function addMarkers() {
continue;
}
const el = document.createElement('label');
el.className = 'lableclass';
el.className = 'labelclass';
el.textContent = nodes[i].v + '℃';
el.style.background = getColor(nodes[i].v);
el.style.borderColor = getColor(nodes[i].v);

View File

@ -0,0 +1,58 @@
import { Scene, Marker, MarkerLayer } from '@antv/l7';
import { GaodeMap } from '@antv/l7-maps';
const scene = new Scene({
id: 'map',
map: new GaodeMap({
style: 'light',
center: [ 105.790327, 36.495636 ],
pitch: 0,
zoom: 4
})
});
addMarkers();
scene.render();
function addMarkers() {
fetch(
'https://gw.alipayobjects.com/os/basement_prod/67f47049-8787-45fc-acfe-e19924afe032.json'
)
.then(res => res.json())
.then(nodes => {
const markerLayer = new MarkerLayer();
for (let i = 0; i < nodes.length; i++) {
if (nodes[i].g !== '1' || nodes[i].v === '') {
continue;
}
const el = document.createElement('label');
el.className = 'labelclass';
el.textContent = nodes[i].v + '℃';
el.style.background = getColor(nodes[i].v);
el.style.borderColor = getColor(nodes[i].v);
const marker = new Marker({
element: el
}).setLnglat({ lng: nodes[i].x * 1, lat: nodes[i].y });
markerLayer.addMarker(marker);
}
scene.addMarkerLayer(markerLayer);
});
}
function getColor(v) {
const colors = [ '#ffffe5', '#f7fcb9', '#d9f0a3', '#addd8e', '#78c679', '#41ab5d', '#238443', '#005a32' ];
return v > 50
? colors[7]
: v > 40
? colors[6]
: v > 30
? colors[5]
: v > 20
? colors[4]
: v > 10
? colors[3]
: v > 5
? colors[2]
: v > 0
? colors[1]
: colors[0];
}

View File

@ -4,6 +4,18 @@
"en": "Category"
},
"demos": [
{
"filename": "markerlayer.js",
"title": "MarkerLayer 统一管理Marker",
"screenshot":"https://gw.alipayobjects.com/mdn/rms_855bab/afts/img/A*ng-FSqu67kYAAAAAAAAAAABkARQnAQ"
},
{
"filename": "clustermarker.js",
"title": "MarkerLayer 聚合Marker",
"screenshot":"https://gw.alipayobjects.com/mdn/rms_855bab/afts/img/A*2vBbRYT2bgIAAAAAAAAAAABkARQnAQ"
},
{
"filename": "marker.js",
"title": "温度",

View File

@ -11,6 +11,27 @@
z-index: 5;
cursor: pointer;
}
.l7-marker-cluster {
background-clip: padding-box;
border-radius: 20px;
background-color: rgba(181, 226, 140, 0.6);
width: 40px;
height: 40px;
}
.l7-marker-cluster div {
width: 30px;
height: 30px;
margin-left: 5px;
margin-top: 5px;
text-align: center;
border-radius: 15px;
font: 12px "Helvetica Neue", Arial, Helvetica, sans-serif;
background-color: rgba(110, 204, 57, 0.6);
}
.l7-marker-cluster span {
line-height: 30px;
}
.l7-popup-anchor-top,
.l7-popup-anchor-top-left,
.l7-popup-anchor-top-right {

View File

@ -4,10 +4,11 @@ import Logo from './control/logo';
import Scale from './control/scale';
import Zoom from './control/zoom';
import Marker from './marker';
import MarkerLayer from './markerlayer';
import Popup from './popup';
// 引入样式
// TODO: 使用 Less 或者 Sass每个组件单独引用自身样式
import './css/l7.css';
export { Control, Logo, Scale, Zoom, Layers, Marker, Popup };
export { Control, Logo, Scale, Zoom, Layers, Marker, Popup, MarkerLayer };

View File

@ -0,0 +1,209 @@
import { IMapService, IMarker, TYPES } from '@antv/l7-core';
import { bindAll, DOM } from '@antv/l7-utils';
import { EventEmitter } from 'eventemitter3';
import { Container } from 'inversify';
import { isFunction } from 'lodash';
import Supercluster from 'supercluster';
import Marker from './marker';
type CallBack = (...args: any[]) => any;
interface IMarkerStyleOption {
element: CallBack;
style: { [key: string]: any } | CallBack;
className: string;
radius: number;
maxZoom: number;
minZoom: number;
zoom: number;
}
interface IMarkerLayerOption {
cluster: boolean;
clusterOption: IMarkerStyleOption;
}
interface IPointFeature {
geometry: {
type: 'Point';
coordinates: [number, number];
};
properties: any;
}
export default class MarkerLayer extends EventEmitter {
private markers: IMarker[] = [];
private markerLayerOption: IMarkerLayerOption;
private clusterIndex: Supercluster;
private points: IPointFeature[] = [];
private clusterMarkers: IMarker[] = [];
private mapsService: IMapService<unknown>;
private scene: Container;
private zoom: number;
constructor(option?: Partial<IMarkerLayerOption>) {
super();
this.markerLayerOption = {
...this.getDefault(),
...option,
};
bindAll(['update'], this);
this.zoom = this.markerLayerOption.clusterOption?.zoom || -99;
}
public getDefault() {
return {
cluster: false,
clusterOption: {
radius: 80,
maxZoom: 20,
minZoom: 0,
zoom: -99,
style: {},
className: '',
element: this.generateElement,
},
};
}
public addTo(scene: Container) {
// this.remove();
this.scene = scene;
this.mapsService = scene.get<IMapService>(TYPES.IMapService);
if (this.markerLayerOption.cluster) {
this.initCluster();
this.update();
this.mapsService.on('zoom', this.update);
this.mapsService.on('zoomchange', this.update);
}
this.addMarkers();
return this;
}
public addMarker(marker: IMarker) {
const cluster = this.markerLayerOption.cluster;
if (cluster) {
this.addPoint(marker);
}
this.markers.push(marker);
}
public removeMarker(marker: IMarker) {
this.markers.indexOf(marker);
const markerIndex = this.markers.indexOf(marker);
if (markerIndex > -1) {
this.markers.splice(markerIndex, 1);
}
}
public getMarkers() {
const cluster = this.markerLayerOption.cluster;
return cluster ? this.clusterMarkers : this.markers;
}
public addMarkers() {
this.getMarkers().forEach((marker: IMarker) => {
marker.addTo(this.scene);
});
}
public clear() {
this.markers.forEach((marker: IMarker) => {
marker.remove();
});
this.mapsService.off('zoom', this.update);
this.mapsService.off('zoomchange', this.update);
this.markers = [];
}
public destroy() {
this.clear();
this.removeAllListeners();
}
private addPoint(marker: IMarker) {
const { lng, lat } = marker.getLnglat();
const feature: IPointFeature = {
geometry: {
type: 'Point',
coordinates: [lng, lat],
},
properties: marker.getExtData(),
};
this.points.push(feature);
}
private initCluster() {
if (!this.markerLayerOption.cluster) {
return;
}
const { radius, minZoom = 0, maxZoom } = this.markerLayerOption
.clusterOption as IMarkerStyleOption;
this.clusterIndex = new Supercluster({
radius,
minZoom,
maxZoom,
});
// @ts-ignore
this.clusterIndex.load(this.points);
}
private getClusterMarker(zoom: number) {
const clusterPoint = this.clusterIndex.getClusters(
[-180, -85, 180, 85],
zoom,
);
this.clusterMarkers.forEach((marker: IMarker) => {
marker.remove();
});
this.clusterMarkers = [];
clusterPoint.forEach((feature) => {
const marker =
feature.properties && feature.properties.hasOwnProperty('point_count')
? this.clusterMarker(feature)
: this.normalMarker(feature);
this.clusterMarkers.push(marker);
marker.addTo(this.scene);
});
}
private clusterMarker(feature: any) {
const clusterOption = this.markerLayerOption.clusterOption;
const { element } = clusterOption as IMarkerStyleOption;
const marker = new Marker({
element: element(feature),
}).setLnglat({
lng: feature.geometry.coordinates[0],
lat: feature.geometry.coordinates[1],
});
return marker;
}
private normalMarker(feature: any) {
const marker = new Marker().setLnglat({
lng: feature.geometry.coordinates[0],
lat: feature.geometry.coordinates[1],
});
return marker;
}
private update() {
const zoom = this.mapsService.getZoom();
if (Math.abs(zoom - this.zoom) > 1) {
this.getClusterMarker(Math.floor(zoom));
this.zoom = Math.floor(zoom);
}
}
private generateElement(feature: any) {
const el = DOM.create('div', 'l7-marker-cluster');
const label = DOM.create('div', '', el);
const span = DOM.create('span', '', label);
span.textContent = feature.properties.point_count;
// if (className !== '') {
// DOM.addClass(el, className);
// }
// span.textContent = feature.properties.point_count;
// const elStyle = isFunction(style)
// ? style(feature.properties.point_count)
// : style;
// Object.keys(elStyle).forEach((key: string) => {
// // @ts-ignore
// el.style[key] = elStyle[key];
// });
return el;
}
}

View File

@ -15,15 +15,29 @@ export interface IMarker {
setLnglat(lngLat: ILngLat | IPoint): this;
getLnglat(): ILngLat;
getElement(): HTMLElement;
getExtData(): any;
setExtData(data: any): void;
setPopup(popup: IPopup): void;
togglePopup(): this;
}
export interface IMarkerService {
container: HTMLElement;
addMarkerLayer(Marker: IMarkerLayer): void;
removeMarkerLayer(Marker: IMarkerLayer): void;
addMarker(Marker: IMarker): void;
addMarkers(): void;
addMarkerLayers(): void;
removeMarker(Marker: IMarker): void;
removeAllMarkers(): void;
init(scene: Container): void;
destroy(): void;
}
export interface IMarkerLayer {
addMarker(marker: IMarker): void;
getMarkers(): IMarker[];
addTo(scene: Container): void;
removeMarker(marker: IMarker): void;
clear(): void;
destroy(): void;
}

View File

@ -1,7 +1,12 @@
import { Container, injectable } from 'inversify';
import { TYPES } from '../../types';
import { IMapService } from '../map/IMapService';
import { IMarker, IMarkerService, IMarkerServiceCfg } from './IMarkerService';
import {
IMarker,
IMarkerLayer,
IMarkerService,
IMarkerServiceCfg,
} from './IMarkerService';
@injectable()
export default class MarkerService implements IMarkerService {
@ -9,7 +14,28 @@ export default class MarkerService implements IMarkerService {
private scene: Container;
private mapsService: IMapService;
private markers: IMarker[] = [];
private markerLayers: IMarkerLayer[] = [];
private unAddMarkers: IMarker[] = [];
private unAddMarkerLayers: IMarkerLayer[] = [];
public addMarkerLayer(markerLayer: IMarkerLayer): void {
if (this.mapsService.map && this.mapsService.getMarkerContainer()) {
this.markerLayers.push(markerLayer);
markerLayer.addTo(this.scene);
} else {
this.unAddMarkerLayers.push(markerLayer);
}
}
public removeMarkerLayer(layer: IMarkerLayer): void {
layer.destroy();
this.markerLayers.indexOf(layer);
const markerIndex = this.markerLayers.indexOf(layer);
if (markerIndex > -1) {
this.markerLayers.splice(markerIndex, 1);
}
}
public addMarker(marker: IMarker): void {
if (this.mapsService.map && this.mapsService.getMarkerContainer()) {
this.markers.push(marker);
@ -27,8 +53,21 @@ export default class MarkerService implements IMarkerService {
this.unAddMarkers = [];
}
public addMarkerLayers(): void {
this.unAddMarkerLayers.forEach((markerLayer: IMarkerLayer) => {
this.markerLayers.push(markerLayer);
markerLayer.addTo(this.scene);
});
this.unAddMarkers = [];
}
public removeMarker(marker: IMarker): void {
marker.remove();
this.markers.indexOf(marker);
const markerIndex = this.markers.indexOf(marker);
if (markerIndex > -1) {
this.markers.splice(markerIndex, 1);
}
}
public removeAllMarkers(): void {
@ -44,5 +83,14 @@ export default class MarkerService implements IMarkerService {
this.markers.forEach((marker: IMarker) => {
marker.remove();
});
this.markers = [];
this.markerLayers.forEach((layer: IMarkerLayer) => {
layer.destroy();
});
this.markerLayers = [];
}
private removeMakerLayerMarker(layer: IMarkerLayer) {
layer.destroy();
}
}

View File

@ -141,6 +141,7 @@ export default class Scene extends EventEmitter implements ISceneService {
this.map.addMarkerContainer();
// 初始化未加载的marker;
this.markerService.addMarkers();
this.markerService.addMarkerLayers();
// 地图初始化之后 才能初始化 container 上的交互
this.interactionService.init();
this.logger.debug('map loaded');

View File

@ -12,6 +12,7 @@ import {
ILngLat,
IMapService,
IMarker,
IMarkerLayer,
IMarkerService,
IPoint,
IPopup,
@ -162,6 +163,14 @@ class Scene
this.markerService.addMarker(marker);
}
public addMarkerLayer(layer: IMarkerLayer) {
this.markerService.addMarkerLayer(layer);
}
public removeMarkerLayer(layer: IMarkerLayer) {
this.markerService.removeMarkerLayer(layer);
}
public removeAllMakers() {
this.markerService.removeAllMarkers();
}

View File

@ -1,4 +1,4 @@
.lableclass {
.labelclass {
position: absolute;
display: inline;
cursor: pointer;

View File

@ -1,7 +1,9 @@
import { storiesOf } from '@storybook/react';
import * as React from 'react';
import Chart from './components/chart';
import ClusterMarkerLayer from './components/clusterMarker';
import Marker from './components/Marker';
import MarkerLayerComponent from './components/markerlayer';
import Popup from './components/Popup';
import Scale from './components/Scale';
import Zoom from './components/Zoom';
@ -11,4 +13,6 @@ storiesOf('UI 组件', module)
.add('Scale', () => <Scale />)
.add('Marker', () => <Marker />)
.add('Chart', () => <Chart />)
.add('Popup', () => <Popup />);
.add('Popup', () => <Popup />)
.add('MarkerLayer', () => <MarkerLayerComponent />)
.add('ClusterMarkerLayer', () => <ClusterMarkerLayer />);

View File

@ -0,0 +1,74 @@
// @ts-ignore
import { Marker, MarkerLayer, PolygonLayer, Scene } from '@antv/l7';
import { Mapbox, GaodeMap } from '@antv/l7-maps';
import * as React from 'react';
export default class ClusterMarkerLayer extends React.Component {
private scene: Scene;
public componentWillUnmount() {
this.scene.destroy();
}
public async componentDidMount() {
function getColor(v: number) {
return v > 50
? '#800026'
: v > 40
? '#BD0026'
: v > 30
? '#E31A1C'
: v > 20
? '#FC4E2A'
: v > 10
? '#FD8D3C'
: v > 5
? '#FEB24C'
: v > 0
? '#FED976'
: '#FFEDA0';
}
const response = await fetch(
'https://gw.alipayobjects.com/os/basement_prod/d3564b06-670f-46ea-8edb-842f7010a7c6.json',
);
const nodes = await response.json();
const markerLayer = new MarkerLayer({
cluster: true,
});
const scene = new Scene({
id: 'map',
map: new Mapbox({
style: 'dark',
center: [110.19382669582967, 30.258134],
pitch: 0,
zoom: 3,
}),
});
// tslint:disable-next-line:prefer-for-of
for (let i = 0; i < nodes.features.length; i++) {
const { coordinates } = nodes.features[i].geometry;
const marker = new Marker().setLnglat({
lng: coordinates[0],
lat: coordinates[1],
});
markerLayer.addMarker(marker);
}
scene.addMarkerLayer(markerLayer);
this.scene = scene;
}
public render() {
return (
<div
id="map"
style={{
position: 'absolute',
top: 0,
left: 0,
right: 0,
bottom: 0,
}}
/>
);
}
}

View File

@ -0,0 +1,78 @@
// @ts-ignore
import { Marker, MarkerLayer, PolygonLayer, Scene } from '@antv/l7';
import { GaodeMap } from '@antv/l7-maps';
import * as React from 'react';
export default class MarkerLayerComponent extends React.Component {
private scene: Scene;
public componentWillUnmount() {
this.scene.destroy();
}
public async componentDidMount() {
function getColor(v: number) {
return v > 50
? '#800026'
: v > 40
? '#BD0026'
: v > 30
? '#E31A1C'
: v > 20
? '#FC4E2A'
: v > 10
? '#FD8D3C'
: v > 5
? '#FEB24C'
: v > 0
? '#FED976'
: '#FFEDA0';
}
const response = await fetch(
'https://gw.alipayobjects.com/os/basement_prod/67f47049-8787-45fc-acfe-e19924afe032.json',
);
const nodes = await response.json();
const markerLayer = new MarkerLayer();
const scene = new Scene({
id: 'map',
map: new GaodeMap({
style: 'dark',
center: [110.19382669582967, 30.258134],
pitch: 0,
zoom: 3,
}),
});
// tslint:disable-next-line:prefer-for-of
for (let i = 0; i < nodes.length; i++) {
if (nodes[i].g !== '1' || nodes[i].v === '') {
continue;
}
const el = document.createElement('label');
el.className = 'labelclass';
el.textContent = nodes[i].v + '℃';
el.style.background = getColor(nodes[i].v);
el.style.borderColor = getColor(nodes[i].v);
const marker = new Marker({
element: el,
}).setLnglat({ lng: nodes[i].x * 1, lat: nodes[i].y });
markerLayer.addMarker(marker);
}
scene.addMarkerLayer(markerLayer);
this.scene = scene;
}
public render() {
return (
<div
id="map"
style={{
position: 'absolute',
top: 0,
left: 0,
right: 0,
bottom: 0,
}}
/>
);
}
}