使用 redux 获取用户位置并将其设置为状态

发布于 2025-01-09 02:54:42 字数 2277 浏览 2 评论 0原文

我有一个反应应用程序,它使用反应传单在地图上显示用户位置。我使用 navigator.geolocation 来获取位置,并在用户允许访问时将其设置为 userData 状态。有时它在地图上显示它,但有时它返回未定义。

redux-action:

export const getUserData = () => {
    return async (dispatch) => {
            if(navigator.geolocation) {
                navigator.geolocation.watchPosition((position) => {
                   dispatch({
                       type: FETCH_USER_DATA,
                       payload: position.coords
                   })
               })
        } else {
            alert('Unable to get location')
        }
    }
}

reducer:

export const rootReducer = (state = initialState, action) => {
    switch (action.type) {
        case FETCH_IP_LOCATION:
            return {
                ...state,
                ipLocation: action.payload,
                isLoading: false,
            }
        case FETCH_USER_DATA:
            return {
                ...state,
                userData: action.payload,
                isLoading: false,
            }
    }
}

渲染地图的组件:

import React from 'react';
import '../styles/MapComponent.css';
import { MapContainer, TileLayer, Marker, Popup } from 'react-leaflet';
import { useEffect } from 'react';
import { getUserData } from '../redux/action/';
import { useDispatch, useSelector } from 'react-redux'
 
const MapComponent = () => {
    const dispatch = useDispatch()
    const isLoading = useSelector((state) => state.isLoading)
    const coord = useSelector((state) => state.userData)
 
    useEffect(() => {
       dispatch(getUserData())
    }, [])

  return (
    <MapContainer center={[coord.latitude, coord.longitude]} zoom={12}>
      <TileLayer
        attribution='&copy; <a href="https://www.openstreetmap.org/copyright">OpenStreetMap</a> contributors'
        url="https://{s}.tile.openstreetmap.org/{z}/{x}/{y}.png"
      />
      <Marker position={[coord.latitude, coord.longitude]}>
        <Popup>
          A pretty CSS3 popup. <br /> Easily customizable.
        </Popup>
      </Marker>
    </MapContainer>
  
  )
}

export default MapComponent

I have a react app which shows the users location on map using react-leaflet. I used navigator.geolocation to get the position and sets it to the userData state when user allows access. Sometimes it is showing it on map but sometimes it is returning undefined.

redux-action :

export const getUserData = () => {
    return async (dispatch) => {
            if(navigator.geolocation) {
                navigator.geolocation.watchPosition((position) => {
                   dispatch({
                       type: FETCH_USER_DATA,
                       payload: position.coords
                   })
               })
        } else {
            alert('Unable to get location')
        }
    }
}

reducer :

export const rootReducer = (state = initialState, action) => {
    switch (action.type) {
        case FETCH_IP_LOCATION:
            return {
                ...state,
                ipLocation: action.payload,
                isLoading: false,
            }
        case FETCH_USER_DATA:
            return {
                ...state,
                userData: action.payload,
                isLoading: false,
            }
    }
}

Component which renders map :

import React from 'react';
import '../styles/MapComponent.css';
import { MapContainer, TileLayer, Marker, Popup } from 'react-leaflet';
import { useEffect } from 'react';
import { getUserData } from '../redux/action/';
import { useDispatch, useSelector } from 'react-redux'
 
const MapComponent = () => {
    const dispatch = useDispatch()
    const isLoading = useSelector((state) => state.isLoading)
    const coord = useSelector((state) => state.userData)
 
    useEffect(() => {
       dispatch(getUserData())
    }, [])

  return (
    <MapContainer center={[coord.latitude, coord.longitude]} zoom={12}>
      <TileLayer
        attribution='© <a href="https://www.openstreetmap.org/copyright">OpenStreetMap</a> contributors'
        url="https://{s}.tile.openstreetmap.org/{z}/{x}/{y}.png"
      />
      <Marker position={[coord.latitude, coord.longitude]}>
        <Popup>
          A pretty CSS3 popup. <br /> Easily customizable.
        </Popup>
      </Marker>
    </MapContainer>
  
  )
}

export default MapComponent

如果你对这篇内容有疑问,欢迎到本站社区发帖提问 参与讨论,获取更多帮助,或者扫码二维码加入 Web 技术交流群。

扫码二维码加入Web技术交流群

发布评论

需要 登录 才能够评论, 你可以免费 注册 一个本站的账号。

评论(1

薔薇婲 2025-01-16 02:54:42

最好将导航器逻辑保留在 redux 之外,一旦获得数据就可以分派它们。

useEffect(() => {
    if (navigator?.geolocation) {
      navigator.geolocation.getCurrentPosition((location) => {
        if (location) dispatch(getUserData(location.coords));
      });
    }
  }, []);
  • 现在,首先您必须提供一些坐标来初始化传单地图,因为 userData 未定义,并且出于同样的原因不渲染标记

  • 收到数据后,您将需要一个额外的组件来将地图视图更改为中心prop 是不可变的。无需经过中心,因为您拥有有关 redux 的信息,因此您可以从那里使用它。

    function ChangeView() {
       const coords = useSelector((state) => state.root.userData);
    
       常量映射 = useMap();
       map.setView([坐标.纬度,坐标.经度],缩放);
    
       返回空值;
     }
    

在下面的演示 您可以使用 redux-toolkit 看到所有部分协同工作。

Its better to keep the navigator logic outside redux and once you get the data you can dispatch them.

useEffect(() => {
    if (navigator?.geolocation) {
      navigator.geolocation.getCurrentPosition((location) => {
        if (location) dispatch(getUserData(location.coords));
      });
    }
  }, []);
  • Now in the first place you have to provide some coords to initialize the leaflet map as userData is undefined and not render the marker for the same reason

  • After you receive the data you will need an extra comp to change the maps view as center prop is immutable. No need to pass the center as you have that info on redux so you can consume it from there.

    function ChangeView() {
       const coords = useSelector((state) => state.root.userData);
    
       const map = useMap();
       map.setView([coords.latitude, coords.longitude], zoom);
    
       return null;
     }
    

In the following demo you can see all the pieces work together using redux-toolkit.

~没有更多了~
我们使用 Cookies 和其他技术来定制您的体验包括您的登录状态等。通过阅读我们的 隐私政策 了解更多相关信息。 单击 接受 或继续使用网站,即表示您同意使用 Cookies 和您的相关数据。
原文