import React, { useState, useCallback, useRef, useEffect } from 'react'; import { GoogleMap, useJsApiLoader, Marker, Autocomplete, } from '@react-google-maps/api'; import { useMaps } from '../stores/useMaps'; import { LocateFixed, MapPinIcon } from 'lucide-react'; import { Button } from '@chakra-ui/react'; const containerStyle = { width: '100%', height: '400px', }; const defaultCenter = { lat: -6.2, lng: 106.816666, }; const PinpointLocation = ({ initialLatitude, initialLongitude, initialAddress }) => { const { isLoaded } = useJsApiLoader({ googleMapsApiKey: process.env.NEXT_PUBLIC_GOOGLE_API_KEY, libraries: ['places'], }); const { addressMaps, setAddressMaps, selectedPosition, setSelectedPosition, setDetailAddress, setPinedMaps, } = useMaps(); const [tempAddress, setTempAddress] = useState(initialAddress || ''); const [tempPosition, setTempPosition] = useState( initialLatitude && initialLongitude ? { lat: parseFloat(initialLatitude), lng: parseFloat(initialLongitude) } : selectedPosition.lat && selectedPosition.lng ? selectedPosition : defaultCenter ); const [markerIcon, setMarkerIcon] = useState(null); const autocompleteRef = useRef(null); useEffect(() => { if (isLoaded && window.google) { setMarkerIcon({ url: 'https://cdn.pixabay.com/photo/2014/04/03/10/03/google-309740_1280.png', scaledSize: new window.google.maps.Size(25, 40), }); } // If we have initial coordinates but no address, fetch the address if (initialLatitude && initialLongitude && !initialAddress) { getAddress(parseFloat(initialLatitude), parseFloat(initialLongitude)); } }, [isLoaded, initialLatitude, initialLongitude, initialAddress]); const getAddressComponent = (components, type) => { const component = components.find((comp) => comp.types.includes(type)); return component ? component.long_name : ''; }; const getAddress = async (lat, lng) => { try { const response = await fetch( `https://maps.googleapis.com/maps/api/geocode/json?latlng=${lat},${lng}&key=${process.env.NEXT_PUBLIC_GOOGLE_API_KEY}` ); const data = await response.json(); if (data.results[0]) { const addressComponents = data.results[0].address_components; const formattedAddress = data.results[0].formatted_address; const details = { street: getAddressComponent(addressComponents, 'route') + ' ' + getAddressComponent(addressComponents, 'street_number'), province: getAddressComponent(addressComponents, 'administrative_area_level_1'), district: getAddressComponent(addressComponents, 'administrative_area_level_2'), subDistrict: getAddressComponent(addressComponents, 'administrative_area_level_3'), village: getAddressComponent(addressComponents, 'administrative_area_level_4'), postalCode: getAddressComponent(addressComponents, 'postal_code'), }; setDetailAddress(details); setTempAddress(formattedAddress); } } catch (error) { console.error('Error fetching address:', error); } }; const onMapClick = useCallback((event) => { const lat = event.latLng.lat(); const lng = event.latLng.lng(); const newPosition = { lat, lng }; setTempPosition(newPosition); getAddress(lat, lng); }, []); const handlePlaceSelect = () => { const place = autocompleteRef.current.getPlace(); if (place && place.geometry) { const lat = place.geometry.location.lat(); const lng = place.geometry.location.lng(); const newPosition = { lat, lng }; setTempPosition(newPosition); setTempAddress(place.formatted_address); getAddress(lat, lng); } }; const handleUseCurrentLocation = () => { if (navigator.geolocation) { navigator.geolocation.getCurrentPosition( (position) => { const lat = position.coords.latitude; const lng = position.coords.longitude; const newPosition = { lat, lng }; setTempPosition(newPosition); getAddress(lat, lng); }, (error) => { console.error('Error getting current location:', error); } ); } }; const handleSavePinpoint = (event) => { event.preventDefault(); if (tempAddress === '') { alert('Silahkan pilih lokasi terlebih dahulu'); return; } setSelectedPosition(tempPosition); setAddressMaps(tempAddress); setPinedMaps(false); }; return (

Tentukan Pinpoint Lokasi

{isLoaded ? ( (autocompleteRef.current = ref)} onPlaceChanged={handlePlaceSelect} > setTempAddress(e.target.value)} style={{ width: '100%', padding: '8px' }} /> ) : (

Loading autocomplete...

)}
{isLoaded ? ( {markerIcon && ( { const lat = e.latLng.lat(); const lng = e.latLng.lng(); const newPosition = { lat, lng }; setTempPosition(newPosition); getAddress(lat, lng); }} icon={markerIcon} /> )} ) : (

Loading map...

)}

PinPoint :

); }; export default PinpointLocation;