@goongmaps/goong-map-react
Version:
A fork of react-map-gl. React components for Goong JS
112 lines (91 loc) • 3.52 kB
JavaScript
import * as React from 'react';
import {useMemo} from 'react';
import * as PropTypes from 'prop-types';
import MapState from '../utils/map-state';
import {LINEAR_TRANSITION_PROPS} from '../utils/map-controller';
import useMapControl, {mapControlDefaultProps, mapControlPropTypes} from './use-map-control';
const noop = () => {};
const propTypes = Object.assign({}, mapControlPropTypes, {
// Custom className
className: PropTypes.string,
style: PropTypes.object,
// Callbacks fired when the user interacted with the map. The object passed to the callbacks
// contains viewport properties such as `longitude`, `latitude`, `zoom` etc.
onViewStateChange: PropTypes.func,
onViewportChange: PropTypes.func,
// Show/hide compass button
showCompass: PropTypes.bool,
// Show/hide zoom buttons
showZoom: PropTypes.bool,
// Custom labels assigned to the controls
zoomInLabel: PropTypes.string,
zoomOutLabel: PropTypes.string,
compassLabel: PropTypes.string
});
const defaultProps = Object.assign({}, mapControlDefaultProps, {
className: '',
showCompass: true,
showZoom: true,
zoomInLabel: 'Zoom In',
zoomOutLabel: 'Zoom Out',
compassLabel: 'Reset North'
});
function updateViewport(context, props, opts) {
const {viewport} = context;
const mapState = new MapState(Object.assign({}, viewport, opts));
const viewState = Object.assign({}, mapState.getViewportProps(), LINEAR_TRANSITION_PROPS);
const onViewportChange = props.onViewportChange || context.onViewportChange || noop;
const onViewStateChange = props.onViewStateChange || context.onViewStateChange || noop;
// Call new style callback
onViewStateChange({viewState});
// Call old style callback
onViewportChange(viewState);
}
function renderButton(type, label, callback, children) {
return (
<button
key={type}
className={`mapboxgl-ctrl-icon mapboxgl-ctrl-${type}`}
type="button"
title={label}
onClick={callback}
>
{children || <span className="mapboxgl-ctrl-icon" aria-hidden="true" />}
</button>
);
}
function renderCompass(context) {
const {bearing} = context.viewport;
const style = {transform: `rotate(${-bearing}deg)`};
return <span className="mapboxgl-ctrl-icon" aria-hidden="true" style={style} />;
}
/*
* PureComponent doesn't update when context changes, so
* implementing our own shouldComponentUpdate here.
*/
function NavigationControl(props) {
const {context, containerRef} = useMapControl(props);
const onZoomIn = () => {
updateViewport(context, props, {zoom: context.viewport.zoom + 1});
};
const onZoomOut = () => {
updateViewport(context, props, {zoom: context.viewport.zoom - 1});
};
const onResetNorth = () => {
updateViewport(context, props, {bearing: 0, pitch: 0});
};
const {className, showCompass, showZoom, zoomInLabel, zoomOutLabel, compassLabel} = props;
const style = useMemo(() => ({position: 'absolute', ...props.style}), [props.style]);
return (
<div style={style} className={className}>
<div className="mapboxgl-ctrl mapboxgl-ctrl-group" ref={containerRef}>
{showZoom && renderButton('zoom-in', zoomInLabel, onZoomIn)}
{showZoom && renderButton('zoom-out', zoomOutLabel, onZoomOut)}
{showCompass && renderButton('compass', compassLabel, onResetNorth, renderCompass(context))}
</div>
</div>
);
}
NavigationControl.propTypes = propTypes;
NavigationControl.defaultProps = defaultProps;
export default React.memo(NavigationControl);