deck.gl GeoJsonLayer:点击更新要素颜色
deck.gl GeoJsonLayer: update feature color on click
我正在使用 deck.gl GeoJsonLayer 在地图上显示区域。
通过一个事件,我能够使这些区域可选择。
我正在努力解决的问题是,在点击某个特征后更改特征选择中的特征颜色。
这是渲染层函数(改编自here)
_renderLayers(props) {
const { geofeats } = props;
return [
new GeoJsonLayer({
id: 'geojson',
data: geofeats,
opacity: 0.8,
stroked: true,
filled: true,
extruded: true,
wireframe: true,
getElevation: f => 10000,
getFillColor: f =>
{
if(f.properties.selected)
{
return [200, 200, 100];
} else return [200, 100, 150];
},
getLineColor: [255, 255, 255],
pickable: true,
onHover: this._onHover,
onClick: this._onClick
})
];
}
问题是,当我通过 setState() 更新特征集合中特征的选择状态时,即使数据中表示了状态变化,渲染也不会更新。
这就是我传递 'geofeats' 对象的方式:
render() {
const {features} = this.state;
const {mapStyle = 'mapbox://styles/mapbox/light-v9'} = this.props;
return (
<DeckGL
layers={this._renderLayers({geofeats: features})}
effects={this._effects}
initialViewState={INITIAL_VIEW_STATE}
controller={true}
>
<StaticMap
reuseMaps
mapStyle={mapStyle}
preventStyleDiffing={true}
mapboxApiAccessToken={MAPBOX_TOKEN}
/>
{this._renderTooltip}
</DeckGL>
);
}
我尝试通过 setState 而不是通过 props - 但结果是一样的。要素集合已移交给 GeoJsonLayer 但从未更新。
有人可以告诉我,我做错了什么吗?
更新:错误再现的要点示例:https://gist.github.com/jaronimoe/efdbb58b3f52c2aac63362a921802cfe
您的问题有两个方面。
首先,您永远不会为每个功能分配一个唯一的 ID,这最终意味着当您将鼠标悬停在一个项目上时,每个功能都会被分配给选定项。这是因为在 setFeatureSelected 中,您正在比较 undefined == undefined
aka (currentID == selectedId
)。作为旁注,您应该默认为三重相等
其次,您需要告诉 GeoJsonLayer
如何更新颜色。这些是您需要进行的更改:
function setFeatureSelected(features, selfeat) {
let selectedID = selfeat.properties.id;
for (let i = 0; i < features.features.length; i++) {
let currentID = features.features[i].properties.id;
if (selectedID === currentID) {
features.features[i].properties.selected = true;
} else {
// Make sure to update the others to be false, so that way only one is ever selected
features.features[i].properties.selected = false;
}
}
return features;
}
那么你想更新 initFeatureSelected
以实际注入 ID。现在我只使用数组中的索引,但如果你愿意,你可以使用更明确的东西
function initFeatureSelected(features) {
for (let i = 0; i < features.features.length; i++) {
features.features[i].properties.selected = false;
// Track each feature individually with a unique ID.
features.features[i].properties.id = i;
}
return features;
}
您需要的最后一部分是告诉 GeoJsonLayer
您有新值来重新计算 getFillColor
函数。这是通过 updateTriggers
完成的,您可以在其中传递确定颜色是否应更改的事物的值。其中,在您的情况下是您的功能的选定 ID
new GeoJsonLayer({
id: 'geojson',
data,
opacity: 0.8,
stroked: false,
filled: true,
extruded: true,
wireframe: true,
getElevation: 50, //Math.sqrt(f.properties.valuePerSqm) * 10,
getFillColor: d => d.properties.selected ? [100, 105, 155] : [55, 205, 155], //COLOR_SCALE(f.properties.growth),
getLineColor: [255, 255, 255],
updateTriggers: {
getFillColor: [
this.state.hoveredObject
? this.state.hoveredObject.properties.id
: null
]
},
pickable: true,
onHover: this._onHover
})
我正在使用 deck.gl GeoJsonLayer 在地图上显示区域。 通过一个事件,我能够使这些区域可选择。 我正在努力解决的问题是,在点击某个特征后更改特征选择中的特征颜色。
这是渲染层函数(改编自here)
_renderLayers(props) {
const { geofeats } = props;
return [
new GeoJsonLayer({
id: 'geojson',
data: geofeats,
opacity: 0.8,
stroked: true,
filled: true,
extruded: true,
wireframe: true,
getElevation: f => 10000,
getFillColor: f =>
{
if(f.properties.selected)
{
return [200, 200, 100];
} else return [200, 100, 150];
},
getLineColor: [255, 255, 255],
pickable: true,
onHover: this._onHover,
onClick: this._onClick
})
];
}
问题是,当我通过 setState() 更新特征集合中特征的选择状态时,即使数据中表示了状态变化,渲染也不会更新。
这就是我传递 'geofeats' 对象的方式:
render() {
const {features} = this.state;
const {mapStyle = 'mapbox://styles/mapbox/light-v9'} = this.props;
return (
<DeckGL
layers={this._renderLayers({geofeats: features})}
effects={this._effects}
initialViewState={INITIAL_VIEW_STATE}
controller={true}
>
<StaticMap
reuseMaps
mapStyle={mapStyle}
preventStyleDiffing={true}
mapboxApiAccessToken={MAPBOX_TOKEN}
/>
{this._renderTooltip}
</DeckGL>
);
}
我尝试通过 setState 而不是通过 props - 但结果是一样的。要素集合已移交给 GeoJsonLayer 但从未更新。
有人可以告诉我,我做错了什么吗?
更新:错误再现的要点示例:https://gist.github.com/jaronimoe/efdbb58b3f52c2aac63362a921802cfe
您的问题有两个方面。
首先,您永远不会为每个功能分配一个唯一的 ID,这最终意味着当您将鼠标悬停在一个项目上时,每个功能都会被分配给选定项。这是因为在 setFeatureSelected 中,您正在比较 undefined == undefined
aka (currentID == selectedId
)。作为旁注,您应该默认为三重相等
其次,您需要告诉 GeoJsonLayer
如何更新颜色。这些是您需要进行的更改:
function setFeatureSelected(features, selfeat) {
let selectedID = selfeat.properties.id;
for (let i = 0; i < features.features.length; i++) {
let currentID = features.features[i].properties.id;
if (selectedID === currentID) {
features.features[i].properties.selected = true;
} else {
// Make sure to update the others to be false, so that way only one is ever selected
features.features[i].properties.selected = false;
}
}
return features;
}
那么你想更新 initFeatureSelected
以实际注入 ID。现在我只使用数组中的索引,但如果你愿意,你可以使用更明确的东西
function initFeatureSelected(features) {
for (let i = 0; i < features.features.length; i++) {
features.features[i].properties.selected = false;
// Track each feature individually with a unique ID.
features.features[i].properties.id = i;
}
return features;
}
您需要的最后一部分是告诉 GeoJsonLayer
您有新值来重新计算 getFillColor
函数。这是通过 updateTriggers
完成的,您可以在其中传递确定颜色是否应更改的事物的值。其中,在您的情况下是您的功能的选定 ID
new GeoJsonLayer({
id: 'geojson',
data,
opacity: 0.8,
stroked: false,
filled: true,
extruded: true,
wireframe: true,
getElevation: 50, //Math.sqrt(f.properties.valuePerSqm) * 10,
getFillColor: d => d.properties.selected ? [100, 105, 155] : [55, 205, 155], //COLOR_SCALE(f.properties.growth),
getLineColor: [255, 255, 255],
updateTriggers: {
getFillColor: [
this.state.hoveredObject
? this.state.hoveredObject.properties.id
: null
]
},
pickable: true,
onHover: this._onHover
})