持续三天不断尝试之后获得了一些不为人知的辛酸 orz
关于OpenLayers的Cluster
ol.source.Cluster
ol利用这个js对象实现对地图上feature进行聚合展示的控制,所有的features需要被放在一个feature数组中作为ol.source.Vector中features属性进行载入。
下面就是官方针对Cluster的API文档:
new ol.source.Cluster(options)
options: Constructor options.
Name | Type | Description |
---|---|---|
attributions | ol.AttributionLike | Attributions. |
distance | number | Minimum distance in pixels between clusters. Default is 20. |
extent | ol.Extent | Extent. |
geometryFunction | function | Function that takes an ol.Feature as argument and returns an ol.geom.Point as cluster calculation point for the feature. When a feature should not be considered for clustering, the function should return null. The default, which works when the underyling source contains point features only, is function(feature) {return feature.getGeometry();} See ol.geom.Polygon#getInteriorPoint for a way to get a cluster calculation point for polygons. |
format | ol.format.Feature | Format. |
logo | string | Logo. |
projection | ol.ProjectionLike | Projection. |
source | ol.source.Vector | Source. Required. |
wrapX | boolean | WrapX. Default is true |
之所以把这个贴出来是为了让大家了解一个事情,那就是如果在Openlayers的官方API文档中找你想要的东西,恐怕应该是一件需要碰运气的事情。并不是说他们的文档写的不够全面和完善,但当你想要从这种API文档中快速的找到一些能够立即实现你想要的开发效果的东西,那恐怕你得把他当前API所涉及到的所有js对象的API都看一遍才行了(事实上我就是这么做的)。本文的目的只是让有GIS js开发基础的人能够快速的用openlayers实现多边形聚合效果,因为在网上搜了很久都没有类似的贴,所以才决定记录下来,希望有可能帮助到遇到类似情况的人吧。
实现多边形聚合效果
在网上翻了很多关于ol实现聚合的贴,基本上都是照本宣科搞几个ol.geom.Point随便弄一弄,然而我遇到的需求是我可能会get两百个左右的ol.geom.Polygon,按照官网给的例子,根本行不通啊喂,报的错也让我很崩溃,说我的操作有问题。马德我一个塞尔达可以单挑人马活下来的人说我操作有问题我摔。
愤怒的我立即打开F12开始不断的试错,试图证明我操作的问题是ol本身的问题引起的,后来API中的一个option让我产生了想法,就是上面的那个geometryFunction,这个function可以将feature获取到并且返回一个Point作为feature本身的聚合坐标,只要不妨碍进行聚合的计算,即使返回极点他也不拦着你,不过一般的Polygon都有一个getInteriorPoint方法可以获取多边形的内点,所以在geometryFunction中可以做到返回多边形内点坐标(那么问题就来了,内点是哪个点呢,其实我也不知道,原谅我调试的时候没有多去试试)。
代码中把geometryFunction属性加上,写一个新的方法去获取多边形内点并返回。
var geometryFunc = function (feature) {
return feature.getGeometry().getInteriorPoint();
}
var vectorLayer = new ol.layer.Vector({
source: new ol.source.Cluster({
distance: distance, //调用方法时候输入的聚合距离
source: new ol.source.Vector({ features: features }), //获取到的feature数组
geometryFunction: geometryFunc //获取多边形内点的方法
})
});
然后保存刷新页面,查询一下,最后发现聚合效果正常运转,我的操作终于没问题了。
但是放大以后,我勒个擦,这些小圆圈是什么鬼,我的多边形呢?
于是陷入了第二波痛苦。
直到在官方demo里面看到一个这样的example:
Earthquake Cluster
然后打开了通往新世界的大门。
在ol.layer.Vector中有一个style属性,用来将获取到的feature和resolution进行处理,如下:
var maxFeatureCount, vector;
// 计算当前resolution下feature之间的距离以形成聚合图形的基础信息
function calculateClusterInfo(resolution) {
maxFeatureCount = 0;
var features = vector.getSource().getFeatures();
var feature, radius;
for (var i = features.length - 1; i >= 0; --i) {
feature = features[i];
var originalFeatures = feature.get('features');
var extent = ol.extent.createEmpty();
var j, jj;
for (j = 0, jj = originalFeatures.length; j < jj; ++j) {
ol.extent.extend(extent, originalFeatures[j].getGeometry().getExtent());
}
maxFeatureCount = Math.max(maxFeatureCount, jj);
radius = 0.25 * (ol.extent.getWidth(extent) + ol.extent.getHeight(extent)) /resolution;
feature.set('radius', radius);
}
}
var currentResolution;
function styleFunction(feature, resolution) {
if (resolution != currentResolution) {
// 根据resolution处理图形聚合状态
calculateClusterInfo(resolution);
currentResolution = resolution;
}
var style;
var size = feature.get('features').length;
if (size > 1) {
// 聚合图形样式,自动获取对应半径,调整颜色深度
style = new ol.style.Style({
image: new ol.style.Circle({
radius: feature.get('radius'),
fill: new ol.style.Fill({
color: [255, 153, 0, Math.min(0.8, 0.4 + (size / maxFeatureCount))]
})
}),
text: new ol.style.Text({
text: size.toString(),
fill: textFill,
stroke: textStroke
})
});
} else {
var originalFeature = feature.get('features')[0];
// 单个图形样式处理方法
style = createPolygonStyle(originalFeature);
}
return style;
}
vector = new ol.layer.Vector({
source: new ol.source.Cluster({
distance: 40,
source: new ol.source.Vector({
//获取到的feature数组
feature: features
})
}),
style: styleFunction
});
这个方法首先对返回的resolution进行了判断,如果地图进行了缩放,则重新计算图面上点与点之间的距离以规划聚合图形的半径,然后对其返回对应单独的图形样式。
接下来就要单独讲讲上面单个图形样式处理方法的部分,由于在我的需求中,对每个图形还单独增加了点击效果,聚合图形不考虑在内,但凡出现没有在聚合范围内的Polygon,都要求能够点击并且高亮展示,所以上文的createPolygonStyle方法还需进行进一步的处理;首先在全局变量中需要添加一个highLightFeatureId用于记录哪个Polygon是需要高亮的,然后在styleFunction回调的时候对需要高亮的图形样式进行调整,代码如下:
var createPolygonStyle = function (feature) {
if (_highlightfeatureId == feature.attributes.id) {
// 高亮样式
var style = new ol.style.Style({
geometry: feature.getGeometry(),
fill: new ol.style.Fill({
color: 'rgba(204,102,102,0.8)'
}),
stroke: new ol.style.Stroke({
color: '#CC3333',
width: 3
}),
image: new ol.style.Circle({
stroke: new ol.style.Stroke({
color: 'rgb(204,102,102)',
width: 1
}),
radius: 6,
fill: new ol.style.Fill({
color: [255, 255, 255, 0.6]
})
}),
text: new ol.style.Text({
text: feature.text,
textAlign: "center",
textBaseline: 'middle',
font: "14px serif",
fill: new ol.style.Fill({ color: "#FFFFFF" }),
stroke: new ol.style.Stroke({ color: "#000000", width: 3 })
})
});
return style;
} else {
// 普通多边形的样式
var style = new ol.style.Style({
geometry: feature.getGeometry(),
fill: new ol.style.Fill({
color: 'rgba(0,204,204,0.2)'
}),
stroke: new ol.style.Stroke({
lineDash: [20, 20],
color: '#0099CC',
width: 1
}),
image: new ol.style.Circle({
stroke: new ol.style.Stroke({
color: 'rgb(0,204,204)',
width: 1
}),
radius: 6,
fill: new ol.style.Fill({
color: [255, 255, 255, 0.6]
})
}),
text: new ol.style.Text({
text: feature.text,
textAlign: "center",
textBaseline: 'middle',
font: "14px serif",
fill: new ol.style.Fill({ color: "#FFFFFF" }),
stroke: new ol.style.Stroke({ color: "#000000", width: 3 })
})
});
return style;
}
}
最后附上实现效果图
小比例尺
大比例尺
小比例尺高亮
大比例尺高亮
高亮后聚合