React Leaflet - 在地图中心对齐弹出窗口
React Leaflet - Align Popup on the center of the map
如果单击标记,我想将地图视图平移到标记的中心。到目前为止,这适用于我的项目。此外,我希望打开的弹出窗口在地图视图的中心对齐。
我想我可以像这样用 leaflet-popup
class 硬编码:
.leaflet-popup {
transform: translate(-40%, 50%) !important;
}
但这不起作用。我的代码的问题是,弹出窗口的位置独立于地图视图。但它应该根据当前视图居中。我可以向您展示我的地图设置:
编辑
我提供了一个 CodeSandBox link 来玩。单击顶部的标记,您会看到弹出窗口未在屏幕中心对齐:
地图组件
const {Map, TileLayer, Marker, GeoJSON} = ReactLeaflet;
function MapOverlay({setSwipeState}) {
const mapRef = useRef(null);
const [donationLocations] = useState([
[48.135125, 11.581981], [58.403, 20.420], [43.300, 40],
[70.505, -20], [40.505, -80], [-40.505, -10]
]);
function centerMapView(e) {
const {leafletElement} = mapRef.current;
if(e) {
leafletElement.setView(e.popup._latlng); // center view to where popup opens
// todo: align popup in the middle of the screen
// Get bounds of map view, divide it by 2 and apply coorditanes to the popup position
}
}
function getMapData() {
return (
<div>
{
donationLocations.map((position, i) =>
(
<Marker position={position} key={i}>
<MarkerPopup/>
</Marker>
)
)
}
</div>
)
}
return (
<Map
ref={mapRef}
center={[45.000, 10.000]}
zoom={3}
onPopupopen={centerMapView.bind(this)}
zoomControl={false}
minZoom={3}
bounceAtZoomLimits={true}
maxBoundsViscosity={.95}
maxBounds={[[-180, -90], [180, 90]]}
>
<TileLayer
noWrap={true}
attribution='&copy <a href="http://osm.org/copyright">OpenStreetMap</a> contributors | &copy <a href="https://apps.mapbox.com/feedback/">Mapbox</a>'
url={'https://api.mapbox.com/styles/v1/mapbox/streets-v11/tiles/{z}/{x}/{y}?access_token=' + process.env.REACT_APP_MAPBOX_KEY}
/>
{getMapData()}
</Map>
)
}
标记弹出组件
const {Popup} = ReactLeaflet;
export default function MarkerPopup() {
return (
<Popup
autoPan={false} // Important part here
>
<Card>
...
</Card>
</Popup>
);
}
function centerMapView(e) {
const { leafletElement } = mapRef.current;
if (e) {
const popupLatlng = e.popup._latlng;
const zoom = leafletElement.getZoom();
const point = leafletElement.project(popupLatlng, zoom);
const newPoint = point.subtract([0, 180]);
const newLatlng = leafletElement.unproject(newPoint, zoom);
leafletElement.panTo(newLatlng, { animate: true });
}
}
您可以将 centerMapView
函数更新为以下内容。这将首先将您的 Latlng
值转换为 point
值,然后通过减去特定数量的像素( offset )修改该值,最后将 point
值转换回 Latlng
并使用值调用 panTo
。
最佳解决方案是制作一个始终位于屏幕中央的模态弹出窗口。
在('popupopen')
的活动中展示
我设法让它工作了。解决方案是制作一个与 DOM.
分离的模态对话框
可以看到工作代码:Here
// Pass open state an setOpen function from the parent
function MarkerPopup({open, setOpen}) {
const classes = useStyles();
function handleClose() {
setOpen(false);
}
return (
<Popup
autoPan={false}
>
<Modal
className={classes.modal}
open={open}
classes={{root: classes.root}}
onClose={handleClose.bind(this)}
BackdropComponent={Backdrop}
BackdropProps={{
timeout: 0,
invisible: true
}}
>
<Card className={classes.card}>
<CardMedia
className={classes.media}
image="https://material-ui.com/static/images/cards/contemplative-reptile.jpg"
title="Contemplative Reptile"
/>
<CardContent>
<Typography gutterBottom variant="h5" component="h2">
Lizard
</Typography>
<Typography variant="body2" color="textSecondary" component="p">
Lizards are a widespread group of squamate reptiles, with over 6,000 species, ranging
across all continents except Antarctica
</Typography>
</CardContent>
<CardActions>
<Button size="small" color="primary">
Details
</Button>
<Button size="small" color="primary">
Donate
</Button>
</CardActions>
</Card>
</Modal>
</Popup>
);
}
如果单击标记,我想将地图视图平移到标记的中心。到目前为止,这适用于我的项目。此外,我希望打开的弹出窗口在地图视图的中心对齐。
我想我可以像这样用 leaflet-popup
class 硬编码:
.leaflet-popup {
transform: translate(-40%, 50%) !important;
}
但这不起作用。我的代码的问题是,弹出窗口的位置独立于地图视图。但它应该根据当前视图居中。我可以向您展示我的地图设置:
编辑
我提供了一个 CodeSandBox link 来玩。单击顶部的标记,您会看到弹出窗口未在屏幕中心对齐:
地图组件
const {Map, TileLayer, Marker, GeoJSON} = ReactLeaflet;
function MapOverlay({setSwipeState}) {
const mapRef = useRef(null);
const [donationLocations] = useState([
[48.135125, 11.581981], [58.403, 20.420], [43.300, 40],
[70.505, -20], [40.505, -80], [-40.505, -10]
]);
function centerMapView(e) {
const {leafletElement} = mapRef.current;
if(e) {
leafletElement.setView(e.popup._latlng); // center view to where popup opens
// todo: align popup in the middle of the screen
// Get bounds of map view, divide it by 2 and apply coorditanes to the popup position
}
}
function getMapData() {
return (
<div>
{
donationLocations.map((position, i) =>
(
<Marker position={position} key={i}>
<MarkerPopup/>
</Marker>
)
)
}
</div>
)
}
return (
<Map
ref={mapRef}
center={[45.000, 10.000]}
zoom={3}
onPopupopen={centerMapView.bind(this)}
zoomControl={false}
minZoom={3}
bounceAtZoomLimits={true}
maxBoundsViscosity={.95}
maxBounds={[[-180, -90], [180, 90]]}
>
<TileLayer
noWrap={true}
attribution='&copy <a href="http://osm.org/copyright">OpenStreetMap</a> contributors | &copy <a href="https://apps.mapbox.com/feedback/">Mapbox</a>'
url={'https://api.mapbox.com/styles/v1/mapbox/streets-v11/tiles/{z}/{x}/{y}?access_token=' + process.env.REACT_APP_MAPBOX_KEY}
/>
{getMapData()}
</Map>
)
}
标记弹出组件
const {Popup} = ReactLeaflet;
export default function MarkerPopup() {
return (
<Popup
autoPan={false} // Important part here
>
<Card>
...
</Card>
</Popup>
);
}
function centerMapView(e) {
const { leafletElement } = mapRef.current;
if (e) {
const popupLatlng = e.popup._latlng;
const zoom = leafletElement.getZoom();
const point = leafletElement.project(popupLatlng, zoom);
const newPoint = point.subtract([0, 180]);
const newLatlng = leafletElement.unproject(newPoint, zoom);
leafletElement.panTo(newLatlng, { animate: true });
}
}
您可以将 centerMapView
函数更新为以下内容。这将首先将您的 Latlng
值转换为 point
值,然后通过减去特定数量的像素( offset )修改该值,最后将 point
值转换回 Latlng
并使用值调用 panTo
。
最佳解决方案是制作一个始终位于屏幕中央的模态弹出窗口。
在('popupopen')
的活动中展示我设法让它工作了。解决方案是制作一个与 DOM.
分离的模态对话框可以看到工作代码:Here
// Pass open state an setOpen function from the parent
function MarkerPopup({open, setOpen}) {
const classes = useStyles();
function handleClose() {
setOpen(false);
}
return (
<Popup
autoPan={false}
>
<Modal
className={classes.modal}
open={open}
classes={{root: classes.root}}
onClose={handleClose.bind(this)}
BackdropComponent={Backdrop}
BackdropProps={{
timeout: 0,
invisible: true
}}
>
<Card className={classes.card}>
<CardMedia
className={classes.media}
image="https://material-ui.com/static/images/cards/contemplative-reptile.jpg"
title="Contemplative Reptile"
/>
<CardContent>
<Typography gutterBottom variant="h5" component="h2">
Lizard
</Typography>
<Typography variant="body2" color="textSecondary" component="p">
Lizards are a widespread group of squamate reptiles, with over 6,000 species, ranging
across all continents except Antarctica
</Typography>
</CardContent>
<CardActions>
<Button size="small" color="primary">
Details
</Button>
<Button size="small" color="primary">
Donate
</Button>
</CardActions>
</Card>
</Modal>
</Popup>
);
}