* extract preventDefaultWrapper into utils * add click handling to beacon list item * add click handling to dialog sidebar * focus in on beacons when clicked in list * stylelint * fussy import ordering * test beacon focusing in beaocnviewdialog
132 lines
4.0 KiB
TypeScript
132 lines
4.0 KiB
TypeScript
/*
|
|
Copyright 2022 The Matrix.org Foundation C.I.C.
|
|
|
|
Licensed under the Apache License, Version 2.0 (the "License");
|
|
you may not use this file except in compliance with the License.
|
|
You may obtain a copy of the License at
|
|
|
|
http://www.apache.org/licenses/LICENSE-2.0
|
|
|
|
Unless required by applicable law or agreed to in writing, software
|
|
distributed under the License is distributed on an "AS IS" BASIS,
|
|
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
See the License for the specific language governing permissions and
|
|
limitations under the License.
|
|
*/
|
|
|
|
import React, { ReactNode, useContext, useEffect } from 'react';
|
|
import classNames from 'classnames';
|
|
import maplibregl from 'maplibre-gl';
|
|
import { ClientEvent, IClientWellKnown } from 'matrix-js-sdk/src/matrix';
|
|
import { logger } from 'matrix-js-sdk/src/logger';
|
|
|
|
import MatrixClientContext from '../../../contexts/MatrixClientContext';
|
|
import { useEventEmitterState } from '../../../hooks/useEventEmitter';
|
|
import { parseGeoUri } from '../../../utils/location';
|
|
import { tileServerFromWellKnown } from '../../../utils/WellKnownUtils';
|
|
import { useMap } from '../../../utils/location/useMap';
|
|
import { Bounds } from '../../../utils/beacon/bounds';
|
|
|
|
const useMapWithStyle = ({ id, centerGeoUri, onError, interactive, bounds }) => {
|
|
const bodyId = `mx_Map_${id}`;
|
|
|
|
// style config
|
|
const context = useContext(MatrixClientContext);
|
|
const mapStyleUrl = useEventEmitterState(
|
|
context,
|
|
ClientEvent.ClientWellKnown,
|
|
(clientWellKnown: IClientWellKnown) => tileServerFromWellKnown(clientWellKnown)?.["map_style_url"],
|
|
);
|
|
|
|
const map = useMap({ interactive, bodyId, onError });
|
|
|
|
useEffect(() => {
|
|
if (mapStyleUrl && map) {
|
|
map.setStyle(mapStyleUrl);
|
|
}
|
|
}, [mapStyleUrl, map]);
|
|
|
|
useEffect(() => {
|
|
if (map && centerGeoUri) {
|
|
try {
|
|
const coords = parseGeoUri(centerGeoUri);
|
|
map.setCenter({ lon: coords.longitude, lat: coords.latitude });
|
|
} catch (_error) {
|
|
logger.error('Could not set map center');
|
|
}
|
|
}
|
|
}, [map, centerGeoUri]);
|
|
|
|
useEffect(() => {
|
|
if (map && bounds) {
|
|
try {
|
|
const lngLatBounds = new maplibregl.LngLatBounds(
|
|
[bounds.west, bounds.south],
|
|
[bounds.east, bounds.north],
|
|
);
|
|
map.fitBounds(lngLatBounds, { padding: 100, maxZoom: 15 });
|
|
} catch (_error) {
|
|
logger.error('Invalid map bounds');
|
|
}
|
|
}
|
|
}, [map, bounds]);
|
|
|
|
return {
|
|
map,
|
|
bodyId,
|
|
};
|
|
};
|
|
|
|
interface MapProps {
|
|
id: string;
|
|
interactive?: boolean;
|
|
/**
|
|
* set map center to geoUri coords
|
|
* Center will only be set to valid geoUri
|
|
* this prop is only simply diffed by useEffect, so to trigger *recentering* of the same geoUri
|
|
* append the uri with a var not used by the geoUri spec
|
|
* eg a timestamp: `geo:54,42;mxTs=123`
|
|
*/
|
|
centerGeoUri?: string;
|
|
bounds?: Bounds;
|
|
className?: string;
|
|
onClick?: () => void;
|
|
onError?: (error: Error) => void;
|
|
children?: (renderProps: {
|
|
map: maplibregl.Map;
|
|
}) => ReactNode;
|
|
}
|
|
|
|
const Map: React.FC<MapProps> = ({
|
|
bounds,
|
|
centerGeoUri,
|
|
children,
|
|
className,
|
|
id,
|
|
interactive,
|
|
onError, onClick,
|
|
}) => {
|
|
const { map, bodyId } = useMapWithStyle({ centerGeoUri, onError, id, interactive, bounds });
|
|
|
|
const onMapClick = (
|
|
event: React.MouseEvent<HTMLDivElement, MouseEvent>,
|
|
) => {
|
|
// Eat click events when clicking the attribution button
|
|
const target = event.target as Element;
|
|
if (target.classList.contains("maplibregl-ctrl-attrib-button")) {
|
|
return;
|
|
}
|
|
|
|
onClick && onClick();
|
|
};
|
|
|
|
return <div className={classNames('mx_Map', className)}
|
|
id={bodyId}
|
|
onClick={onMapClick}
|
|
>
|
|
{ !!children && !!map && children({ map }) }
|
|
</div>;
|
|
};
|
|
|
|
export default Map;
|