import { memo, useMemo, useEffect, useContext, PureComponent, type ContextType, } from 'react' import invariant from 'invariant' import { unregisterEvents, applyUpdatersToPropsAndRegisterEvents, } from '../../utils/helper.js' import { noop } from '../../utils/noop.js' import MapContext from '../../map-context.js' const eventMap = { onDblClick: 'dblclick', onClick: 'click', } const updaterMap = { opacity(instance: google.maps.GroundOverlay, opacity: number): void { instance.setOpacity(opacity) }, } type GroundOverlayState = { groundOverlay: google.maps.GroundOverlay | null } export type GroundOverlayProps = { options?: google.maps.GroundOverlayOptions | undefined /** The opacity of the overlay, expressed as a number between 0 and 1. Optional. Defaults to 1. */ opacity?: number | undefined /** This event is fired when the DOM dblclick event is fired on the GroundOverlay. */ onDblClick?: ((e: google.maps.MapMouseEvent) => void) | undefined /** This event is fired when the DOM click event is fired on the GroundOverlay. */ onClick?: ((e: google.maps.MapMouseEvent) => void) | undefined /** The url of the projected image */ url: string /** The bounds that the image will be scaled to fit */ bounds: google.maps.LatLngBoundsLiteral /** This callback is called when the groundOverlay instance has loaded. It is called with the groundOverlay instance. */ onLoad?: ((groundOverlay: google.maps.GroundOverlay) => void) | undefined /** This callback is called when the component unmounts. It is called with the groundOverlay instance. */ onUnmount?: ((groundOverlay: google.maps.GroundOverlay) => void) | undefined visible?: boolean } function GroundOverlayFunctional({ url, bounds, options, visible, }: GroundOverlayProps) { const map = useContext(MapContext) const imageBounds = new google.maps.LatLngBounds( new google.maps.LatLng(bounds.south, bounds.west), new google.maps.LatLng(bounds.north, bounds.east) ) const groundOverlay = useMemo(() => { return new google.maps.GroundOverlay(url, imageBounds, options) }, []) useEffect(() => { if (groundOverlay !== null) { groundOverlay.setMap(map) } }, [map]) useEffect(() => { if (typeof url !== 'undefined' && groundOverlay !== null) { groundOverlay.set('url', url) groundOverlay.setMap(map) } }, [groundOverlay, url]) useEffect(() => { if (typeof visible !== 'undefined' && groundOverlay !== null) { groundOverlay.setOpacity(visible ? 1 : 0) } }, [groundOverlay, visible]) useEffect(() => { const newBounds = new google.maps.LatLngBounds( new google.maps.LatLng(bounds.south, bounds.west), new google.maps.LatLng(bounds.north, bounds.east) ) if (typeof bounds !== 'undefined' && groundOverlay !== null) { groundOverlay.set('bounds', newBounds) groundOverlay.setMap(map) } }, [groundOverlay, bounds]) return null } export const GroundOverlayF = memo(GroundOverlayFunctional) export class GroundOverlay extends PureComponent< GroundOverlayProps, GroundOverlayState > { public static defaultProps = { onLoad: noop, } static override contextType = MapContext declare context: ContextType registeredEvents: google.maps.MapsEventListener[] = [] override state: GroundOverlayState = { groundOverlay: null, } setGroundOverlayCallback = (): void => { if (this.state.groundOverlay !== null && this.props.onLoad) { this.props.onLoad(this.state.groundOverlay) } } override componentDidMount(): void { invariant( !!this.props.url || !!this.props.bounds, `For GroundOverlay, url and bounds are passed in to constructor and are immutable after instantiated. This is the behavior of Google Maps JavaScript API v3 ( See https://developers.google.com/maps/documentation/javascript/reference#GroundOverlay) Hence, use the corresponding two props provided by \`react-google-maps-api\`, url and bounds. In some cases, you'll need the GroundOverlay component to reflect the changes of url and bounds. You can leverage the React's key property to remount the component. Typically, just \`key={url}\` would serve your need. See https://github.com/tomchentw/react-google-maps/issues/655` ) const groundOverlay = new google.maps.GroundOverlay( this.props.url, this.props.bounds, { ...this.props.options, map: this.context, } ) this.registeredEvents = applyUpdatersToPropsAndRegisterEvents({ updaterMap, eventMap, prevProps: {}, nextProps: this.props, instance: groundOverlay, }) this.setState(function setGroundOverlay() { return { groundOverlay, } }, this.setGroundOverlayCallback) } override componentDidUpdate(prevProps: GroundOverlayProps): void { if (this.state.groundOverlay !== null) { unregisterEvents(this.registeredEvents) this.registeredEvents = applyUpdatersToPropsAndRegisterEvents({ updaterMap, eventMap, prevProps, nextProps: this.props, instance: this.state.groundOverlay, }) } } override componentWillUnmount(): void { if (this.state.groundOverlay) { if (this.props.onUnmount) { this.props.onUnmount(this.state.groundOverlay) } this.state.groundOverlay.setMap(null) } } override render(): null { return null } } export default GroundOverlay