Live location sharing - consolidate maps (#8236)
* extract location markers into generic Marker Signed-off-by: Kerry Archibald <kerrya@element.io> * wrap marker in smartmarker Signed-off-by: Kerry Archibald <kerrya@element.io> * test smartmarker Signed-off-by: Kerry Archibald <kerrya@element.io> * working map in location body Signed-off-by: Kerry Archibald <kerrya@element.io> * test Map Signed-off-by: Kerry Archibald <kerrya@element.io> * remove skinned sdk Signed-off-by: Kerry Archibald <kerrya@element.io> * update snaps with new mocks Signed-off-by: Kerry Archibald <kerrya@element.io> * use new ZoomButtons in MLocationBody Signed-off-by: Kerry Archibald <kerrya@element.io> * make LocationViewDialog map interactive Signed-off-by: Kerry Archibald <kerrya@element.io> * test MLocationBody Signed-off-by: Kerry Archibald <kerrya@element.io> * test LocationViewDialog Signed-off-by: Kerry Archibald <kerrya@element.io> * add copyrights, shrink snapshot Signed-off-by: Kerry Archibald <kerrya@element.io> * update comment Signed-off-by: Kerry Archibald <kerrya@element.io> * lint Signed-off-by: Kerry Archibald <kerrya@element.io> * lint Signed-off-by: Kerry Archibald <kerrya@element.io>
This commit is contained in:
parent
944e11d7d6
commit
9ba55d1d14
16 changed files with 890 additions and 235 deletions
|
@ -6,6 +6,8 @@ class MockMap extends EventEmitter {
|
||||||
removeControl = jest.fn();
|
removeControl = jest.fn();
|
||||||
zoomIn = jest.fn();
|
zoomIn = jest.fn();
|
||||||
zoomOut = jest.fn();
|
zoomOut = jest.fn();
|
||||||
|
setCenter = jest.fn();
|
||||||
|
setStyle = jest.fn();
|
||||||
}
|
}
|
||||||
const MockMapInstance = new MockMap();
|
const MockMapInstance = new MockMap();
|
||||||
|
|
||||||
|
|
|
@ -48,49 +48,10 @@ limitations under the License.
|
||||||
background-color: $dialog-close-external-color;
|
background-color: $dialog-close-external-color;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
|
||||||
.mx_MLocationBody {
|
.mx_LocationViewDialog_map {
|
||||||
position: absolute;
|
|
||||||
|
|
||||||
.mx_MLocationBody_map {
|
|
||||||
width: 80vw;
|
width: 80vw;
|
||||||
height: 80vh;
|
height: 80vh;
|
||||||
}
|
border-radius: 8px;
|
||||||
|
|
||||||
.mx_MLocationBody_zoomButtons {
|
|
||||||
position: absolute;
|
|
||||||
display: grid;
|
|
||||||
grid-template-columns: auto;
|
|
||||||
grid-row-gap: 8px;
|
|
||||||
|
|
||||||
right: 24px;
|
|
||||||
bottom: 48px;
|
|
||||||
|
|
||||||
.mx_AccessibleButton {
|
|
||||||
background-color: $background;
|
|
||||||
box-shadow: 0px 4px 12px rgba(0, 0, 0, 0.25);
|
|
||||||
border-radius: 4px;
|
|
||||||
width: 24px;
|
|
||||||
height: 24px;
|
|
||||||
|
|
||||||
.mx_MLocationBody_zoomButton {
|
|
||||||
background-color: $primary-content;
|
|
||||||
margin: 4px;
|
|
||||||
width: 16px;
|
|
||||||
height: 16px;
|
|
||||||
mask-repeat: no-repeat;
|
|
||||||
mask-size: contain;
|
|
||||||
mask-position: center;
|
|
||||||
}
|
|
||||||
|
|
||||||
.mx_MLocationBody_plusButton {
|
|
||||||
mask-image: url('$(res)/img/element-icons/plus-button.svg');
|
|
||||||
}
|
|
||||||
|
|
||||||
.mx_MLocationBody_minusButton {
|
|
||||||
mask-image: url('$(res)/img/element-icons/minus-button.svg');
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
|
@ -225,6 +225,7 @@ class LocationPicker extends React.Component<ILocationPickerProps, IState> {
|
||||||
return (
|
return (
|
||||||
<div className="mx_LocationPicker">
|
<div className="mx_LocationPicker">
|
||||||
<div id="mx_LocationPicker_map" />
|
<div id="mx_LocationPicker_map" />
|
||||||
|
|
||||||
{ this.props.shareType === LocationShareType.Pin && <div className="mx_LocationPicker_pinText">
|
{ this.props.shareType === LocationShareType.Pin && <div className="mx_LocationPicker_pinText">
|
||||||
<span>
|
<span>
|
||||||
{ this.state.position ? _t("Click to move the pin") : _t("Click to drop a pin") }
|
{ this.state.position ? _t("Click to move the pin") : _t("Click to drop a pin") }
|
||||||
|
|
|
@ -16,13 +16,14 @@ limitations under the License.
|
||||||
|
|
||||||
import React from 'react';
|
import React from 'react';
|
||||||
import { MatrixEvent } from 'matrix-js-sdk/src/models/event';
|
import { MatrixEvent } from 'matrix-js-sdk/src/models/event';
|
||||||
import { ClientEvent, IClientWellKnown, MatrixClient } from 'matrix-js-sdk/src/client';
|
import { MatrixClient } from 'matrix-js-sdk/src/client';
|
||||||
|
|
||||||
import BaseDialog from "../dialogs/BaseDialog";
|
import BaseDialog from "../dialogs/BaseDialog";
|
||||||
import { IDialogProps } from "../dialogs/IDialogProps";
|
import { IDialogProps } from "../dialogs/IDialogProps";
|
||||||
import { LocationBodyContent } from '../messages/MLocationBody';
|
import { locationEventGeoUri, isSelfLocation } from '../../../utils/location';
|
||||||
import { tileServerFromWellKnown } from '../../../utils/WellKnownUtils';
|
import Map from './Map';
|
||||||
import { parseGeoUri, locationEventGeoUri, createMapWithCoords } from '../../../utils/location';
|
import SmartMarker from './SmartMarker';
|
||||||
|
import ZoomButtons from './ZoomButtons';
|
||||||
|
|
||||||
interface IProps extends IDialogProps {
|
interface IProps extends IDialogProps {
|
||||||
matrixClient: MatrixClient;
|
matrixClient: MatrixClient;
|
||||||
|
@ -34,78 +35,54 @@ interface IState {
|
||||||
}
|
}
|
||||||
|
|
||||||
export default class LocationViewDialog extends React.Component<IProps, IState> {
|
export default class LocationViewDialog extends React.Component<IProps, IState> {
|
||||||
private coords: GeolocationCoordinates;
|
|
||||||
private map?: maplibregl.Map;
|
|
||||||
|
|
||||||
constructor(props: IProps) {
|
constructor(props: IProps) {
|
||||||
super(props);
|
super(props);
|
||||||
|
|
||||||
this.coords = parseGeoUri(locationEventGeoUri(this.props.mxEvent));
|
|
||||||
this.map = null;
|
|
||||||
this.state = {
|
this.state = {
|
||||||
error: undefined,
|
error: undefined,
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
|
|
||||||
componentDidMount() {
|
|
||||||
if (this.state.error) {
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
this.props.matrixClient.on(ClientEvent.ClientWellKnown, this.updateStyleUrl);
|
|
||||||
|
|
||||||
this.map = createMapWithCoords(
|
|
||||||
this.coords,
|
|
||||||
true,
|
|
||||||
this.getBodyId(),
|
|
||||||
this.getMarkerId(),
|
|
||||||
(e: Error) => this.setState({ error: e }),
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
componentWillUnmount() {
|
|
||||||
this.props.matrixClient.off(ClientEvent.ClientWellKnown, this.updateStyleUrl);
|
|
||||||
}
|
|
||||||
|
|
||||||
private updateStyleUrl = (clientWellKnown: IClientWellKnown) => {
|
|
||||||
const style = tileServerFromWellKnown(clientWellKnown)?.["map_style_url"];
|
|
||||||
if (style) {
|
|
||||||
this.map?.setStyle(style);
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
private getBodyId = () => {
|
private getBodyId = () => {
|
||||||
return `mx_LocationViewDialog_${this.props.mxEvent.getId()}`;
|
return `mx_LocationViewDialog_${this.props.mxEvent.getId()}`;
|
||||||
};
|
};
|
||||||
|
|
||||||
private getMarkerId = () => {
|
private onError = (error) => {
|
||||||
return `mx_MLocationViewDialog_marker_${this.props.mxEvent.getId()}`;
|
this.setState({ error });
|
||||||
};
|
|
||||||
|
|
||||||
private onZoomIn = () => {
|
|
||||||
this.map?.zoomIn();
|
|
||||||
};
|
|
||||||
|
|
||||||
private onZoomOut = () => {
|
|
||||||
this.map?.zoomOut();
|
|
||||||
};
|
};
|
||||||
|
|
||||||
render() {
|
render() {
|
||||||
|
const { mxEvent } = this.props;
|
||||||
|
|
||||||
|
// only pass member to marker when should render avatar marker
|
||||||
|
const markerRoomMember = isSelfLocation(mxEvent.getContent()) ? mxEvent.sender : undefined;
|
||||||
|
const geoUri = locationEventGeoUri(mxEvent);
|
||||||
return (
|
return (
|
||||||
<BaseDialog
|
<BaseDialog
|
||||||
className='mx_LocationViewDialog'
|
className='mx_LocationViewDialog'
|
||||||
onFinished={this.props.onFinished}
|
onFinished={this.props.onFinished}
|
||||||
fixedWidth={false}
|
fixedWidth={false}
|
||||||
>
|
>
|
||||||
<LocationBodyContent
|
<Map
|
||||||
mxEvent={this.props.mxEvent}
|
id={this.getBodyId()}
|
||||||
bodyId={this.getBodyId()}
|
centerGeoUri={geoUri}
|
||||||
markerId={this.getMarkerId()}
|
onError={this.onError}
|
||||||
error={this.state.error}
|
interactive
|
||||||
zoomButtons={true}
|
className="mx_LocationViewDialog_map"
|
||||||
onZoomIn={this.onZoomIn}
|
>
|
||||||
onZoomOut={this.onZoomOut}
|
{
|
||||||
|
({ map }) =>
|
||||||
|
<>
|
||||||
|
<SmartMarker
|
||||||
|
map={map}
|
||||||
|
id={`${this.getBodyId()}-marker`}
|
||||||
|
geoUri={geoUri}
|
||||||
|
roomMember={markerRoomMember}
|
||||||
/>
|
/>
|
||||||
|
<ZoomButtons map={map} />
|
||||||
|
</>
|
||||||
|
}
|
||||||
|
</Map>
|
||||||
</BaseDialog>
|
</BaseDialog>
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
101
src/components/views/location/Map.tsx
Normal file
101
src/components/views/location/Map.tsx
Normal file
|
@ -0,0 +1,101 @@
|
||||||
|
/*
|
||||||
|
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 { 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';
|
||||||
|
|
||||||
|
const useMapWithStyle = ({ id, centerGeoUri, onError, interactive }) => {
|
||||||
|
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', centerGeoUri);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}, [map, centerGeoUri]);
|
||||||
|
|
||||||
|
return {
|
||||||
|
map,
|
||||||
|
bodyId,
|
||||||
|
};
|
||||||
|
};
|
||||||
|
|
||||||
|
interface MapProps {
|
||||||
|
id: string;
|
||||||
|
interactive?: boolean;
|
||||||
|
centerGeoUri?: string;
|
||||||
|
className?: string;
|
||||||
|
onClick?: () => void;
|
||||||
|
onError?: (error: Error) => void;
|
||||||
|
children?: (renderProps: {
|
||||||
|
map: maplibregl.Map;
|
||||||
|
}) => ReactNode;
|
||||||
|
}
|
||||||
|
|
||||||
|
const Map: React.FC<MapProps> = ({
|
||||||
|
centerGeoUri, className, id, onError, onClick, children, interactive,
|
||||||
|
}) => {
|
||||||
|
const { map, bodyId } = useMapWithStyle({ centerGeoUri, onError, id, interactive });
|
||||||
|
|
||||||
|
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;
|
|
@ -15,28 +15,23 @@ limitations under the License.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
import React from 'react';
|
import React from 'react';
|
||||||
import maplibregl from 'maplibre-gl';
|
|
||||||
import { MatrixEvent } from 'matrix-js-sdk/src/models/event';
|
import { MatrixEvent } from 'matrix-js-sdk/src/models/event';
|
||||||
import { ClientEvent, IClientWellKnown } from 'matrix-js-sdk/src/client';
|
|
||||||
|
|
||||||
import { IBodyProps } from "./IBodyProps";
|
|
||||||
import { _t } from '../../../languageHandler';
|
import { _t } from '../../../languageHandler';
|
||||||
import Modal from '../../../Modal';
|
import Modal from '../../../Modal';
|
||||||
import {
|
import {
|
||||||
parseGeoUri,
|
|
||||||
locationEventGeoUri,
|
locationEventGeoUri,
|
||||||
createMapWithCoords,
|
|
||||||
getLocationShareErrorMessage,
|
getLocationShareErrorMessage,
|
||||||
LocationShareError,
|
LocationShareError,
|
||||||
isSelfLocation,
|
isSelfLocation,
|
||||||
} from '../../../utils/location';
|
} from '../../../utils/location';
|
||||||
import LocationViewDialog from '../location/LocationViewDialog';
|
import MatrixClientContext from '../../../contexts/MatrixClientContext';
|
||||||
import TooltipTarget from '../elements/TooltipTarget';
|
import TooltipTarget from '../elements/TooltipTarget';
|
||||||
import { Alignment } from '../elements/Tooltip';
|
import { Alignment } from '../elements/Tooltip';
|
||||||
import AccessibleButton from '../elements/AccessibleButton';
|
import LocationViewDialog from '../location/LocationViewDialog';
|
||||||
import { tileServerFromWellKnown } from '../../../utils/WellKnownUtils';
|
import Map from '../location/Map';
|
||||||
import MatrixClientContext from '../../../contexts/MatrixClientContext';
|
import SmartMarker from '../location/SmartMarker';
|
||||||
import Marker from '../location/Marker';
|
import { IBodyProps } from "./IBodyProps";
|
||||||
|
|
||||||
interface IState {
|
interface IState {
|
||||||
error: Error;
|
error: Error;
|
||||||
|
@ -45,61 +40,23 @@ interface IState {
|
||||||
export default class MLocationBody extends React.Component<IBodyProps, IState> {
|
export default class MLocationBody extends React.Component<IBodyProps, IState> {
|
||||||
public static contextType = MatrixClientContext;
|
public static contextType = MatrixClientContext;
|
||||||
public context!: React.ContextType<typeof MatrixClientContext>;
|
public context!: React.ContextType<typeof MatrixClientContext>;
|
||||||
private coords: GeolocationCoordinates;
|
private mapId: string;
|
||||||
private bodyId: string;
|
|
||||||
private markerId: string;
|
|
||||||
private map?: maplibregl.Map = null;
|
|
||||||
|
|
||||||
constructor(props: IBodyProps) {
|
constructor(props: IBodyProps) {
|
||||||
super(props);
|
super(props);
|
||||||
|
|
||||||
const randomString = Math.random().toString(16).slice(2, 10);
|
const randomString = Math.random().toString(16).slice(2, 10);
|
||||||
|
// multiple instances of same map might be in document
|
||||||
|
// eg thread and main timeline, reply
|
||||||
const idSuffix = `${props.mxEvent.getId()}_${randomString}`;
|
const idSuffix = `${props.mxEvent.getId()}_${randomString}`;
|
||||||
this.bodyId = `mx_MLocationBody_${idSuffix}`;
|
this.mapId = `mx_MLocationBody_${idSuffix}`;
|
||||||
this.markerId = `mx_MLocationBody_marker_${idSuffix}`;
|
|
||||||
this.coords = parseGeoUri(locationEventGeoUri(this.props.mxEvent));
|
|
||||||
|
|
||||||
this.state = {
|
this.state = {
|
||||||
error: undefined,
|
error: undefined,
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
|
|
||||||
componentDidMount() {
|
private onClick = () => {
|
||||||
if (this.state.error) {
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
this.context.on(ClientEvent.ClientWellKnown, this.updateStyleUrl);
|
|
||||||
|
|
||||||
this.map = createMapWithCoords(
|
|
||||||
this.coords,
|
|
||||||
false,
|
|
||||||
this.bodyId,
|
|
||||||
this.markerId,
|
|
||||||
(e: Error) => this.setState({ error: e }),
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
componentWillUnmount() {
|
|
||||||
this.context.off(ClientEvent.ClientWellKnown, this.updateStyleUrl);
|
|
||||||
}
|
|
||||||
|
|
||||||
private updateStyleUrl = (clientWellKnown: IClientWellKnown) => {
|
|
||||||
const style = tileServerFromWellKnown(clientWellKnown)?.["map_style_url"];
|
|
||||||
if (style) {
|
|
||||||
this.map?.setStyle(style);
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
private onClick = (
|
|
||||||
event: React.MouseEvent<HTMLDivElement, MouseEvent>,
|
|
||||||
) => {
|
|
||||||
// Don't open map if we clicked the attribution button
|
|
||||||
const target = event.target as Element;
|
|
||||||
if (target.classList.contains("maplibregl-ctrl-attrib-button")) {
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
Modal.createTrackedDialog(
|
Modal.createTrackedDialog(
|
||||||
'Location View',
|
'Location View',
|
||||||
'',
|
'',
|
||||||
|
@ -114,14 +71,17 @@ export default class MLocationBody extends React.Component<IBodyProps, IState> {
|
||||||
);
|
);
|
||||||
};
|
};
|
||||||
|
|
||||||
|
private onError = (error) => {
|
||||||
|
this.setState({ error });
|
||||||
|
};
|
||||||
|
|
||||||
render(): React.ReactElement<HTMLDivElement> {
|
render(): React.ReactElement<HTMLDivElement> {
|
||||||
return this.state.error ?
|
return this.state.error ?
|
||||||
<LocationBodyFallbackContent error={this.state.error} event={this.props.mxEvent} /> :
|
<LocationBodyFallbackContent error={this.state.error} event={this.props.mxEvent} /> :
|
||||||
<LocationBodyContent
|
<LocationBodyContent
|
||||||
mxEvent={this.props.mxEvent}
|
mxEvent={this.props.mxEvent}
|
||||||
bodyId={this.bodyId}
|
mapId={this.mapId}
|
||||||
markerId={this.markerId}
|
onError={this.onError}
|
||||||
error={this.state.error}
|
|
||||||
tooltip={_t("Expand map")}
|
tooltip={_t("Expand map")}
|
||||||
onClick={this.onClick}
|
onClick={this.onClick}
|
||||||
/>;
|
/>;
|
||||||
|
@ -147,68 +107,52 @@ export const LocationBodyFallbackContent: React.FC<{ event: MatrixEvent, error:
|
||||||
|
|
||||||
interface LocationBodyContentProps {
|
interface LocationBodyContentProps {
|
||||||
mxEvent: MatrixEvent;
|
mxEvent: MatrixEvent;
|
||||||
bodyId: string;
|
mapId: string;
|
||||||
markerId: string;
|
|
||||||
error: Error;
|
|
||||||
tooltip?: string;
|
tooltip?: string;
|
||||||
onClick?: (event: React.MouseEvent<HTMLDivElement, MouseEvent>) => void;
|
onError: (error: Error) => void;
|
||||||
zoomButtons?: boolean;
|
onClick?: () => void;
|
||||||
onZoomIn?: () => void;
|
|
||||||
onZoomOut?: () => void;
|
|
||||||
}
|
}
|
||||||
export const LocationBodyContent: React.FC<LocationBodyContentProps> = (props) => {
|
export const LocationBodyContent: React.FC<LocationBodyContentProps> = ({
|
||||||
const mapDiv = <div
|
mxEvent,
|
||||||
id={props.bodyId}
|
mapId,
|
||||||
onClick={props.onClick}
|
tooltip,
|
||||||
className="mx_MLocationBody_map"
|
onError,
|
||||||
/>;
|
onClick,
|
||||||
|
}) => {
|
||||||
// only pass member to marker when should render avatar marker
|
// only pass member to marker when should render avatar marker
|
||||||
const markerRoomMember = isSelfLocation(props.mxEvent.getContent()) ? props.mxEvent.sender : undefined;
|
const markerRoomMember = isSelfLocation(mxEvent.getContent()) ? mxEvent.sender : undefined;
|
||||||
|
const geoUri = locationEventGeoUri(mxEvent);
|
||||||
|
|
||||||
|
const mapElement = (<Map
|
||||||
|
id={mapId}
|
||||||
|
centerGeoUri={geoUri}
|
||||||
|
onClick={onClick}
|
||||||
|
onError={onError}
|
||||||
|
className="mx_MLocationBody_map"
|
||||||
|
>
|
||||||
|
{
|
||||||
|
({ map }) =>
|
||||||
|
<SmartMarker
|
||||||
|
map={map}
|
||||||
|
id={`${mapId}-marker`}
|
||||||
|
geoUri={geoUri}
|
||||||
|
roomMember={markerRoomMember}
|
||||||
|
/>
|
||||||
|
}
|
||||||
|
</Map>);
|
||||||
|
|
||||||
return <div className="mx_MLocationBody">
|
return <div className="mx_MLocationBody">
|
||||||
{
|
{
|
||||||
props.tooltip
|
tooltip
|
||||||
? <TooltipTarget
|
? <TooltipTarget
|
||||||
label={props.tooltip}
|
label={tooltip}
|
||||||
alignment={Alignment.InnerBottom}
|
alignment={Alignment.InnerBottom}
|
||||||
maxParentWidth={450}
|
maxParentWidth={450}
|
||||||
>
|
>
|
||||||
{ mapDiv }
|
{ mapElement }
|
||||||
</TooltipTarget>
|
</TooltipTarget>
|
||||||
: mapDiv
|
: mapElement
|
||||||
}
|
|
||||||
<Marker id={props.markerId} roomMember={markerRoomMember} />
|
|
||||||
{
|
|
||||||
props.zoomButtons
|
|
||||||
? <ZoomButtons
|
|
||||||
onZoomIn={props.onZoomIn}
|
|
||||||
onZoomOut={props.onZoomOut}
|
|
||||||
/>
|
|
||||||
: null
|
|
||||||
}
|
}
|
||||||
</div>;
|
</div>;
|
||||||
};
|
};
|
||||||
|
|
||||||
interface IZoomButtonsProps {
|
|
||||||
onZoomIn: () => void;
|
|
||||||
onZoomOut: () => void;
|
|
||||||
}
|
|
||||||
|
|
||||||
function ZoomButtons(props: IZoomButtonsProps): React.ReactElement<HTMLDivElement> {
|
|
||||||
return <div className="mx_MLocationBody_zoomButtons">
|
|
||||||
<AccessibleButton
|
|
||||||
onClick={props.onZoomIn}
|
|
||||||
title={_t("Zoom in")}
|
|
||||||
>
|
|
||||||
<div className="mx_MLocationBody_zoomButton mx_MLocationBody_plusButton" />
|
|
||||||
</AccessibleButton>
|
|
||||||
<AccessibleButton
|
|
||||||
onClick={props.onZoomOut}
|
|
||||||
title={_t("Zoom out")}
|
|
||||||
>
|
|
||||||
<div className="mx_MLocationBody_zoomButton mx_MLocationBody_minusButton" />
|
|
||||||
</AccessibleButton>
|
|
||||||
</div>;
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
|
@ -2118,8 +2118,6 @@
|
||||||
"Unable to load map": "Unable to load map",
|
"Unable to load map": "Unable to load map",
|
||||||
"Shared their location: ": "Shared their location: ",
|
"Shared their location: ": "Shared their location: ",
|
||||||
"Shared a location: ": "Shared a location: ",
|
"Shared a location: ": "Shared a location: ",
|
||||||
"Zoom in": "Zoom in",
|
|
||||||
"Zoom out": "Zoom out",
|
|
||||||
"Can't edit poll": "Can't edit poll",
|
"Can't edit poll": "Can't edit poll",
|
||||||
"Sorry, you can't edit a poll after votes have been cast.": "Sorry, you can't edit a poll after votes have been cast.",
|
"Sorry, you can't edit a poll after votes have been cast.": "Sorry, you can't edit a poll after votes have been cast.",
|
||||||
"Vote not registered": "Vote not registered",
|
"Vote not registered": "Vote not registered",
|
||||||
|
@ -2173,6 +2171,8 @@
|
||||||
"My live location": "My live location",
|
"My live location": "My live location",
|
||||||
"Drop a Pin": "Drop a Pin",
|
"Drop a Pin": "Drop a Pin",
|
||||||
"What location type do you want to share?": "What location type do you want to share?",
|
"What location type do you want to share?": "What location type do you want to share?",
|
||||||
|
"Zoom in": "Zoom in",
|
||||||
|
"Zoom out": "Zoom out",
|
||||||
"Frequently Used": "Frequently Used",
|
"Frequently Used": "Frequently Used",
|
||||||
"Smileys & People": "Smileys & People",
|
"Smileys & People": "Smileys & People",
|
||||||
"Animals & Nature": "Animals & Nature",
|
"Animals & Nature": "Animals & Nature",
|
||||||
|
|
62
src/utils/location/useMap.ts
Normal file
62
src/utils/location/useMap.ts
Normal file
|
@ -0,0 +1,62 @@
|
||||||
|
/*
|
||||||
|
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 { useEffect, useState } from 'react';
|
||||||
|
import { Map as MapLibreMap } from 'maplibre-gl';
|
||||||
|
|
||||||
|
import { createMap } from "./map";
|
||||||
|
|
||||||
|
interface UseMapProps {
|
||||||
|
bodyId: string;
|
||||||
|
onError: (error: Error) => void;
|
||||||
|
interactive?: boolean;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Create a map instance
|
||||||
|
* Add listeners for errors
|
||||||
|
* Make sure `onError` has a stable reference
|
||||||
|
* As map is recreated on changes to it
|
||||||
|
*/
|
||||||
|
export const useMap = ({
|
||||||
|
interactive,
|
||||||
|
bodyId,
|
||||||
|
onError,
|
||||||
|
}: UseMapProps): MapLibreMap => {
|
||||||
|
const [map, setMap] = useState<MapLibreMap>();
|
||||||
|
|
||||||
|
useEffect(
|
||||||
|
() => {
|
||||||
|
try {
|
||||||
|
setMap(createMap(interactive, bodyId, onError));
|
||||||
|
} catch (error) {
|
||||||
|
onError(error);
|
||||||
|
}
|
||||||
|
return () => {
|
||||||
|
if (map) {
|
||||||
|
map.remove();
|
||||||
|
setMap(undefined);
|
||||||
|
}
|
||||||
|
};
|
||||||
|
},
|
||||||
|
// map is excluded as a dependency
|
||||||
|
// eslint-disable-next-line react-hooks/exhaustive-deps
|
||||||
|
[interactive, bodyId, onError],
|
||||||
|
);
|
||||||
|
|
||||||
|
return map;
|
||||||
|
};
|
||||||
|
|
|
@ -283,7 +283,7 @@ describe("LocationPicker", () => {
|
||||||
});
|
});
|
||||||
|
|
||||||
// marker not added
|
// marker not added
|
||||||
expect(wrapper.find('.mx_MLocationBody_markerBorder').length).toBeFalsy();
|
expect(wrapper.find('Marker').length).toBeFalsy();
|
||||||
});
|
});
|
||||||
|
|
||||||
it('sets position on click event', () => {
|
it('sets position on click event', () => {
|
||||||
|
|
56
test/components/views/location/LocationViewDialog-test.tsx
Normal file
56
test/components/views/location/LocationViewDialog-test.tsx
Normal file
|
@ -0,0 +1,56 @@
|
||||||
|
/*
|
||||||
|
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 from 'react';
|
||||||
|
import { mount } from 'enzyme';
|
||||||
|
import { RoomMember } from 'matrix-js-sdk/src/matrix';
|
||||||
|
import { LocationAssetType } from 'matrix-js-sdk/src/@types/location';
|
||||||
|
|
||||||
|
import LocationViewDialog from '../../../../src/components/views/location/LocationViewDialog';
|
||||||
|
import { getMockClientWithEventEmitter, makeLocationEvent } from '../../../test-utils';
|
||||||
|
|
||||||
|
describe('<LocationViewDialog />', () => {
|
||||||
|
const roomId = '!room:server';
|
||||||
|
const userId = '@user:server';
|
||||||
|
const mockClient = getMockClientWithEventEmitter({
|
||||||
|
getClientWellKnown: jest.fn().mockReturnValue({
|
||||||
|
"m.tile_server": { map_style_url: 'maps.com' },
|
||||||
|
}),
|
||||||
|
isGuest: jest.fn().mockReturnValue(false),
|
||||||
|
});
|
||||||
|
const defaultEvent = makeLocationEvent("geo:51.5076,-0.1276", LocationAssetType.Pin);
|
||||||
|
const defaultProps = {
|
||||||
|
matrixClient: mockClient,
|
||||||
|
mxEvent: defaultEvent,
|
||||||
|
onFinished: jest.fn(),
|
||||||
|
};
|
||||||
|
const getComponent = (props = {}) =>
|
||||||
|
mount(<LocationViewDialog {...defaultProps} {...props} />);
|
||||||
|
|
||||||
|
it('renders map correctly', () => {
|
||||||
|
const component = getComponent();
|
||||||
|
expect(component.find('Map')).toMatchSnapshot();
|
||||||
|
});
|
||||||
|
|
||||||
|
it('renders marker correctly for self share', () => {
|
||||||
|
const selfShareEvent = makeLocationEvent("geo:51.5076,-0.1276", LocationAssetType.Self);
|
||||||
|
const member = new RoomMember(roomId, userId);
|
||||||
|
// @ts-ignore cheat assignment to property
|
||||||
|
selfShareEvent.sender = member;
|
||||||
|
const component = getComponent({ mxEvent: selfShareEvent });
|
||||||
|
expect(component.find('SmartMarker').props()['roomMember']).toEqual(member);
|
||||||
|
});
|
||||||
|
});
|
165
test/components/views/location/Map-test.tsx
Normal file
165
test/components/views/location/Map-test.tsx
Normal file
|
@ -0,0 +1,165 @@
|
||||||
|
/*
|
||||||
|
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 from 'react';
|
||||||
|
import { mount } from 'enzyme';
|
||||||
|
import { act } from 'react-dom/test-utils';
|
||||||
|
import maplibregl from 'maplibre-gl';
|
||||||
|
import { ClientEvent } from 'matrix-js-sdk/src/matrix';
|
||||||
|
import { logger } from 'matrix-js-sdk/src/logger';
|
||||||
|
|
||||||
|
import Map from '../../../../src/components/views/location/Map';
|
||||||
|
import { findByTestId, getMockClientWithEventEmitter } from '../../../test-utils';
|
||||||
|
import MatrixClientContext from '../../../../src/contexts/MatrixClientContext';
|
||||||
|
|
||||||
|
describe('<Map />', () => {
|
||||||
|
const defaultProps = {
|
||||||
|
centerGeoUri: 'geo:52,41',
|
||||||
|
id: 'test-123',
|
||||||
|
onError: jest.fn(),
|
||||||
|
onClick: jest.fn(),
|
||||||
|
};
|
||||||
|
const matrixClient = getMockClientWithEventEmitter({
|
||||||
|
getClientWellKnown: jest.fn().mockReturnValue({
|
||||||
|
"m.tile_server": { map_style_url: 'maps.com' },
|
||||||
|
}),
|
||||||
|
});
|
||||||
|
const getComponent = (props = {}) =>
|
||||||
|
mount(<Map {...defaultProps} {...props} />, {
|
||||||
|
wrappingComponent: MatrixClientContext.Provider,
|
||||||
|
wrappingComponentProps: { value: matrixClient },
|
||||||
|
});
|
||||||
|
|
||||||
|
beforeEach(() => {
|
||||||
|
jest.clearAllMocks();
|
||||||
|
matrixClient.getClientWellKnown.mockReturnValue({
|
||||||
|
"m.tile_server": { map_style_url: 'maps.com' },
|
||||||
|
});
|
||||||
|
|
||||||
|
jest.spyOn(logger, 'error').mockRestore();
|
||||||
|
});
|
||||||
|
|
||||||
|
const mockMap = new maplibregl.Map();
|
||||||
|
|
||||||
|
it('renders', () => {
|
||||||
|
const component = getComponent();
|
||||||
|
expect(component).toBeTruthy();
|
||||||
|
});
|
||||||
|
|
||||||
|
describe('onClientWellKnown emits', () => {
|
||||||
|
it('updates map style when style url is truthy', () => {
|
||||||
|
getComponent();
|
||||||
|
|
||||||
|
act(() => {
|
||||||
|
matrixClient.emit(ClientEvent.ClientWellKnown, {
|
||||||
|
"m.tile_server": { map_style_url: 'new.maps.com' },
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
expect(mockMap.setStyle).toHaveBeenCalledWith('new.maps.com');
|
||||||
|
});
|
||||||
|
|
||||||
|
it('does not update map style when style url is truthy', () => {
|
||||||
|
getComponent();
|
||||||
|
|
||||||
|
act(() => {
|
||||||
|
matrixClient.emit(ClientEvent.ClientWellKnown, {
|
||||||
|
"m.tile_server": { map_style_url: undefined },
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
expect(mockMap.setStyle).not.toHaveBeenCalledWith();
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
describe('map centering', () => {
|
||||||
|
it('does not try to center when no center uri provided', () => {
|
||||||
|
getComponent({ centerGeoUri: null });
|
||||||
|
expect(mockMap.setCenter).not.toHaveBeenCalled();
|
||||||
|
});
|
||||||
|
|
||||||
|
it('sets map center to centerGeoUri', () => {
|
||||||
|
getComponent({ centerGeoUri: 'geo:51,42' });
|
||||||
|
expect(mockMap.setCenter).toHaveBeenCalledWith({ lat: 51, lon: 42 });
|
||||||
|
});
|
||||||
|
|
||||||
|
it('handles invalid centerGeoUri', () => {
|
||||||
|
const logSpy = jest.spyOn(logger, 'error').mockImplementation();
|
||||||
|
getComponent({ centerGeoUri: '123 Sesame Street' });
|
||||||
|
expect(mockMap.setCenter).not.toHaveBeenCalled();
|
||||||
|
expect(logSpy).toHaveBeenCalledWith('Could not set map center', '123 Sesame Street');
|
||||||
|
});
|
||||||
|
|
||||||
|
it('updates map center when centerGeoUri prop changes', () => {
|
||||||
|
const component = getComponent({ centerGeoUri: 'geo:51,42' });
|
||||||
|
|
||||||
|
component.setProps({ centerGeoUri: 'geo:53,45' });
|
||||||
|
component.setProps({ centerGeoUri: 'geo:56,47' });
|
||||||
|
expect(mockMap.setCenter).toHaveBeenCalledWith({ lat: 51, lon: 42 });
|
||||||
|
expect(mockMap.setCenter).toHaveBeenCalledWith({ lat: 53, lon: 45 });
|
||||||
|
expect(mockMap.setCenter).toHaveBeenCalledWith({ lat: 56, lon: 47 });
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
describe('children', () => {
|
||||||
|
it('renders without children', () => {
|
||||||
|
const component = getComponent({ children: null });
|
||||||
|
|
||||||
|
component.setProps({});
|
||||||
|
|
||||||
|
// no error
|
||||||
|
expect(component).toBeTruthy();
|
||||||
|
});
|
||||||
|
|
||||||
|
it('renders children with map renderProp', () => {
|
||||||
|
const children = ({ map }) => <div data-test-id='test-child' data-map={map}>Hello, world</div>;
|
||||||
|
|
||||||
|
const component = getComponent({ children });
|
||||||
|
|
||||||
|
// renders child with map instance
|
||||||
|
expect(findByTestId(component, 'test-child').props()['data-map']).toEqual(mockMap);
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
describe('onClick', () => {
|
||||||
|
it('eats clicks to maplibre attribution button', () => {
|
||||||
|
const onClick = jest.fn();
|
||||||
|
const component = getComponent({ onClick });
|
||||||
|
|
||||||
|
act(() => {
|
||||||
|
// this is added to the dom by maplibregl
|
||||||
|
// which is mocked
|
||||||
|
// just fake the target
|
||||||
|
const fakeEl = document.createElement('div');
|
||||||
|
fakeEl.className = 'maplibregl-ctrl-attrib-button';
|
||||||
|
component.simulate('click', { target: fakeEl });
|
||||||
|
});
|
||||||
|
|
||||||
|
expect(onClick).not.toHaveBeenCalled();
|
||||||
|
});
|
||||||
|
|
||||||
|
it('calls onClick', () => {
|
||||||
|
const onClick = jest.fn();
|
||||||
|
const component = getComponent({ onClick });
|
||||||
|
|
||||||
|
act(() => {
|
||||||
|
component.simulate('click');
|
||||||
|
});
|
||||||
|
|
||||||
|
expect(onClick).toHaveBeenCalled();
|
||||||
|
});
|
||||||
|
});
|
||||||
|
});
|
|
@ -0,0 +1,156 @@
|
||||||
|
// Jest Snapshot v1, https://goo.gl/fbAQLP
|
||||||
|
|
||||||
|
exports[`<LocationViewDialog /> renders map correctly 1`] = `
|
||||||
|
<Map
|
||||||
|
centerGeoUri="geo:51.5076,-0.1276"
|
||||||
|
className="mx_LocationViewDialog_map"
|
||||||
|
id="mx_LocationViewDialog_$2"
|
||||||
|
interactive={true}
|
||||||
|
onError={[Function]}
|
||||||
|
>
|
||||||
|
<div
|
||||||
|
className="mx_Map mx_LocationViewDialog_map"
|
||||||
|
id="mx_Map_mx_LocationViewDialog_$2"
|
||||||
|
onClick={[Function]}
|
||||||
|
>
|
||||||
|
<SmartMarker
|
||||||
|
geoUri="geo:51.5076,-0.1276"
|
||||||
|
id="mx_LocationViewDialog_$2-marker"
|
||||||
|
map={
|
||||||
|
MockMap {
|
||||||
|
"_events": Object {
|
||||||
|
"error": [Function],
|
||||||
|
},
|
||||||
|
"_eventsCount": 1,
|
||||||
|
"_maxListeners": undefined,
|
||||||
|
"addControl": [MockFunction],
|
||||||
|
"removeControl": [MockFunction],
|
||||||
|
"setCenter": [MockFunction] {
|
||||||
|
"calls": Array [
|
||||||
|
Array [
|
||||||
|
Object {
|
||||||
|
"lat": 51.5076,
|
||||||
|
"lon": -0.1276,
|
||||||
|
},
|
||||||
|
],
|
||||||
|
],
|
||||||
|
"results": Array [
|
||||||
|
Object {
|
||||||
|
"type": "return",
|
||||||
|
"value": undefined,
|
||||||
|
},
|
||||||
|
],
|
||||||
|
},
|
||||||
|
"setStyle": [MockFunction],
|
||||||
|
"zoomIn": [MockFunction],
|
||||||
|
"zoomOut": [MockFunction],
|
||||||
|
Symbol(kCapture): false,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
>
|
||||||
|
<ForwardRef
|
||||||
|
id="mx_LocationViewDialog_$2-marker"
|
||||||
|
>
|
||||||
|
<div
|
||||||
|
className="mx_Marker mx_Marker_defaultColor"
|
||||||
|
id="mx_LocationViewDialog_$2-marker"
|
||||||
|
>
|
||||||
|
<div
|
||||||
|
className="mx_Marker_border"
|
||||||
|
>
|
||||||
|
<div
|
||||||
|
className="mx_Marker_icon"
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</ForwardRef>
|
||||||
|
</SmartMarker>
|
||||||
|
<ZoomButtons
|
||||||
|
map={
|
||||||
|
MockMap {
|
||||||
|
"_events": Object {
|
||||||
|
"error": [Function],
|
||||||
|
},
|
||||||
|
"_eventsCount": 1,
|
||||||
|
"_maxListeners": undefined,
|
||||||
|
"addControl": [MockFunction],
|
||||||
|
"removeControl": [MockFunction],
|
||||||
|
"setCenter": [MockFunction] {
|
||||||
|
"calls": Array [
|
||||||
|
Array [
|
||||||
|
Object {
|
||||||
|
"lat": 51.5076,
|
||||||
|
"lon": -0.1276,
|
||||||
|
},
|
||||||
|
],
|
||||||
|
],
|
||||||
|
"results": Array [
|
||||||
|
Object {
|
||||||
|
"type": "return",
|
||||||
|
"value": undefined,
|
||||||
|
},
|
||||||
|
],
|
||||||
|
},
|
||||||
|
"setStyle": [MockFunction],
|
||||||
|
"zoomIn": [MockFunction],
|
||||||
|
"zoomOut": [MockFunction],
|
||||||
|
Symbol(kCapture): false,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
>
|
||||||
|
<div
|
||||||
|
className="mx_ZoomButtons"
|
||||||
|
>
|
||||||
|
<AccessibleButton
|
||||||
|
className="mx_ZoomButtons_button"
|
||||||
|
data-test-id="map-zoom-in-button"
|
||||||
|
element="div"
|
||||||
|
onClick={[Function]}
|
||||||
|
role="button"
|
||||||
|
tabIndex={0}
|
||||||
|
title="Zoom in"
|
||||||
|
>
|
||||||
|
<div
|
||||||
|
className="mx_AccessibleButton mx_ZoomButtons_button"
|
||||||
|
data-test-id="map-zoom-in-button"
|
||||||
|
onClick={[Function]}
|
||||||
|
onKeyDown={[Function]}
|
||||||
|
onKeyUp={[Function]}
|
||||||
|
role="button"
|
||||||
|
tabIndex={0}
|
||||||
|
title="Zoom in"
|
||||||
|
>
|
||||||
|
<div
|
||||||
|
className="mx_ZoomButtons_icon"
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
</AccessibleButton>
|
||||||
|
<AccessibleButton
|
||||||
|
className="mx_ZoomButtons_button"
|
||||||
|
data-test-id="map-zoom-out-button"
|
||||||
|
element="div"
|
||||||
|
onClick={[Function]}
|
||||||
|
role="button"
|
||||||
|
tabIndex={0}
|
||||||
|
title="Zoom out"
|
||||||
|
>
|
||||||
|
<div
|
||||||
|
className="mx_AccessibleButton mx_ZoomButtons_button"
|
||||||
|
data-test-id="map-zoom-out-button"
|
||||||
|
onClick={[Function]}
|
||||||
|
onKeyDown={[Function]}
|
||||||
|
onKeyUp={[Function]}
|
||||||
|
role="button"
|
||||||
|
tabIndex={0}
|
||||||
|
title="Zoom out"
|
||||||
|
>
|
||||||
|
<div
|
||||||
|
className="mx_ZoomButtons_icon"
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
</AccessibleButton>
|
||||||
|
</div>
|
||||||
|
</ZoomButtons>
|
||||||
|
</div>
|
||||||
|
</Map>
|
||||||
|
`;
|
|
@ -10,6 +10,8 @@ exports[`<SmartMarker /> creates a marker on mount 1`] = `
|
||||||
"_maxListeners": undefined,
|
"_maxListeners": undefined,
|
||||||
"addControl": [MockFunction],
|
"addControl": [MockFunction],
|
||||||
"removeControl": [MockFunction],
|
"removeControl": [MockFunction],
|
||||||
|
"setCenter": [MockFunction],
|
||||||
|
"setStyle": [MockFunction],
|
||||||
"zoomIn": [MockFunction],
|
"zoomIn": [MockFunction],
|
||||||
"zoomOut": [MockFunction],
|
"zoomOut": [MockFunction],
|
||||||
Symbol(kCapture): false,
|
Symbol(kCapture): false,
|
||||||
|
@ -42,6 +44,8 @@ exports[`<SmartMarker /> removes marker on unmount 1`] = `
|
||||||
"_maxListeners": undefined,
|
"_maxListeners": undefined,
|
||||||
"addControl": [MockFunction],
|
"addControl": [MockFunction],
|
||||||
"removeControl": [MockFunction],
|
"removeControl": [MockFunction],
|
||||||
|
"setCenter": [MockFunction],
|
||||||
|
"setStyle": [MockFunction],
|
||||||
"zoomIn": [MockFunction],
|
"zoomIn": [MockFunction],
|
||||||
"zoomOut": [MockFunction],
|
"zoomOut": [MockFunction],
|
||||||
Symbol(kCapture): false,
|
Symbol(kCapture): false,
|
||||||
|
|
|
@ -9,6 +9,8 @@ exports[`<ZoomButtons /> renders buttons 1`] = `
|
||||||
"_maxListeners": undefined,
|
"_maxListeners": undefined,
|
||||||
"addControl": [MockFunction],
|
"addControl": [MockFunction],
|
||||||
"removeControl": [MockFunction],
|
"removeControl": [MockFunction],
|
||||||
|
"setCenter": [MockFunction],
|
||||||
|
"setStyle": [MockFunction],
|
||||||
"zoomIn": [MockFunction],
|
"zoomIn": [MockFunction],
|
||||||
"zoomOut": [MockFunction],
|
"zoomOut": [MockFunction],
|
||||||
Symbol(kCapture): false,
|
Symbol(kCapture): false,
|
||||||
|
|
|
@ -16,30 +16,31 @@ limitations under the License.
|
||||||
|
|
||||||
import React from 'react';
|
import React from 'react';
|
||||||
import { mount } from "enzyme";
|
import { mount } from "enzyme";
|
||||||
import { mocked } from 'jest-mock';
|
|
||||||
import { LocationAssetType } from "matrix-js-sdk/src/@types/location";
|
import { LocationAssetType } from "matrix-js-sdk/src/@types/location";
|
||||||
|
import { RoomMember } from 'matrix-js-sdk/src/matrix';
|
||||||
import maplibregl from 'maplibre-gl';
|
import maplibregl from 'maplibre-gl';
|
||||||
import { logger } from 'matrix-js-sdk/src/logger';
|
import { logger } from 'matrix-js-sdk/src/logger';
|
||||||
|
import { act } from 'react-dom/test-utils';
|
||||||
|
|
||||||
import MLocationBody from "../../../../src/components/views/messages/MLocationBody";
|
import MLocationBody from "../../../../src/components/views/messages/MLocationBody";
|
||||||
import MatrixClientContext from "../../../../src/contexts/MatrixClientContext";
|
import MatrixClientContext from "../../../../src/contexts/MatrixClientContext";
|
||||||
import { RoomPermalinkCreator } from "../../../../src/utils/permalinks/Permalinks";
|
import { RoomPermalinkCreator } from "../../../../src/utils/permalinks/Permalinks";
|
||||||
import { MediaEventHelper } from "../../../../src/utils/MediaEventHelper";
|
import { MediaEventHelper } from "../../../../src/utils/MediaEventHelper";
|
||||||
import { getTileServerWellKnown } from "../../../../src/utils/WellKnownUtils";
|
import Modal from '../../../../src/Modal';
|
||||||
import SdkConfig from "../../../../src/SdkConfig";
|
import SdkConfig from "../../../../src/SdkConfig";
|
||||||
import { makeLocationEvent } from "../../../test-utils/location";
|
import { makeLocationEvent } from "../../../test-utils/location";
|
||||||
|
import { getMockClientWithEventEmitter } from '../../../test-utils';
|
||||||
jest.mock("../../../../src/utils/WellKnownUtils", () => ({
|
|
||||||
getTileServerWellKnown: jest.fn(),
|
|
||||||
}));
|
|
||||||
|
|
||||||
describe("MLocationBody", () => {
|
describe("MLocationBody", () => {
|
||||||
describe('<MLocationBody>', () => {
|
describe('<MLocationBody>', () => {
|
||||||
describe('with error', () => {
|
const roomId = '!room:server';
|
||||||
const mockClient = {
|
const userId = '@user:server';
|
||||||
on: jest.fn(),
|
const mockClient = getMockClientWithEventEmitter({
|
||||||
off: jest.fn(),
|
getClientWellKnown: jest.fn().mockReturnValue({
|
||||||
};
|
"m.tile_server": { map_style_url: 'maps.com' },
|
||||||
|
}),
|
||||||
|
isGuest: jest.fn().mockReturnValue(false),
|
||||||
|
});
|
||||||
const defaultEvent = makeLocationEvent("geo:51.5076,-0.1276", LocationAssetType.Pin);
|
const defaultEvent = makeLocationEvent("geo:51.5076,-0.1276", LocationAssetType.Pin);
|
||||||
const defaultProps = {
|
const defaultProps = {
|
||||||
mxEvent: defaultEvent,
|
mxEvent: defaultEvent,
|
||||||
|
@ -54,12 +55,13 @@ describe("MLocationBody", () => {
|
||||||
wrappingComponent: MatrixClientContext.Provider,
|
wrappingComponent: MatrixClientContext.Provider,
|
||||||
wrappingComponentProps: { value: mockClient },
|
wrappingComponentProps: { value: mockClient },
|
||||||
});
|
});
|
||||||
|
describe('with error', () => {
|
||||||
let sdkConfigSpy;
|
let sdkConfigSpy;
|
||||||
|
|
||||||
beforeEach(() => {
|
beforeEach(() => {
|
||||||
// eat expected errors to keep console clean
|
// eat expected errors to keep console clean
|
||||||
jest.spyOn(logger, 'error').mockImplementation(() => { });
|
jest.spyOn(logger, 'error').mockImplementation(() => { });
|
||||||
mocked(getTileServerWellKnown).mockReturnValue({});
|
mockClient.getClientWellKnown.mockReturnValue({});
|
||||||
sdkConfigSpy = jest.spyOn(SdkConfig, 'get').mockReturnValue({});
|
sdkConfigSpy = jest.spyOn(SdkConfig, 'get').mockReturnValue({});
|
||||||
});
|
});
|
||||||
|
|
||||||
|
@ -75,7 +77,9 @@ describe("MLocationBody", () => {
|
||||||
|
|
||||||
it('displays correct fallback content when map_style_url is misconfigured', () => {
|
it('displays correct fallback content when map_style_url is misconfigured', () => {
|
||||||
const mockMap = new maplibregl.Map();
|
const mockMap = new maplibregl.Map();
|
||||||
mocked(getTileServerWellKnown).mockReturnValue({ map_style_url: 'bad-tile-server.com' });
|
mockClient.getClientWellKnown.mockReturnValue({
|
||||||
|
"m.tile_server": { map_style_url: 'bad-tile-server.com' },
|
||||||
|
});
|
||||||
const component = getComponent();
|
const component = getComponent();
|
||||||
|
|
||||||
// simulate error initialising map in maplibregl
|
// simulate error initialising map in maplibregl
|
||||||
|
@ -85,5 +89,69 @@ describe("MLocationBody", () => {
|
||||||
expect(component.find(".mx_EventTile_body")).toMatchSnapshot();
|
expect(component.find(".mx_EventTile_body")).toMatchSnapshot();
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
|
describe('without error', () => {
|
||||||
|
beforeEach(() => {
|
||||||
|
mockClient.getClientWellKnown.mockReturnValue({
|
||||||
|
"m.tile_server": { map_style_url: 'maps.com' },
|
||||||
|
});
|
||||||
|
|
||||||
|
// MLocationBody uses random number for map id
|
||||||
|
// stabilise for test
|
||||||
|
jest.spyOn(global.Math, 'random').mockReturnValue(0.123456);
|
||||||
|
});
|
||||||
|
|
||||||
|
afterAll(() => {
|
||||||
|
jest.spyOn(global.Math, 'random').mockRestore();
|
||||||
|
});
|
||||||
|
|
||||||
|
it('renders map correctly', () => {
|
||||||
|
const mockMap = new maplibregl.Map();
|
||||||
|
const component = getComponent();
|
||||||
|
|
||||||
|
expect(component).toMatchSnapshot();
|
||||||
|
// map was centered
|
||||||
|
expect(mockMap.setCenter).toHaveBeenCalledWith({
|
||||||
|
lat: 51.5076, lon: -0.1276,
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
it('opens map dialog on click', () => {
|
||||||
|
const modalSpy = jest.spyOn(Modal, 'createTrackedDialog').mockReturnValue(undefined);
|
||||||
|
const component = getComponent();
|
||||||
|
|
||||||
|
act(() => {
|
||||||
|
component.find('Map').at(0).simulate('click');
|
||||||
|
});
|
||||||
|
|
||||||
|
expect(modalSpy).toHaveBeenCalled();
|
||||||
|
});
|
||||||
|
|
||||||
|
it('renders marker correctly for a non-self share', () => {
|
||||||
|
const mockMap = new maplibregl.Map();
|
||||||
|
const component = getComponent();
|
||||||
|
|
||||||
|
expect(component.find('SmartMarker').at(0).props()).toEqual(
|
||||||
|
expect.objectContaining({
|
||||||
|
map: mockMap,
|
||||||
|
geoUri: 'geo:51.5076,-0.1276',
|
||||||
|
roomMember: undefined,
|
||||||
|
}),
|
||||||
|
);
|
||||||
|
});
|
||||||
|
|
||||||
|
it('renders marker correctly for a self share', () => {
|
||||||
|
const selfShareEvent = makeLocationEvent("geo:51.5076,-0.1276", LocationAssetType.Self);
|
||||||
|
const member = new RoomMember(roomId, userId);
|
||||||
|
// @ts-ignore cheat assignment to property
|
||||||
|
selfShareEvent.sender = member;
|
||||||
|
const component = getComponent({ mxEvent: selfShareEvent });
|
||||||
|
|
||||||
|
// render self locations with user avatars
|
||||||
|
expect(component.find('SmartMarker').at(0).props()['roomMember']).toEqual(
|
||||||
|
member,
|
||||||
|
);
|
||||||
|
});
|
||||||
|
});
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
|
@ -27,3 +27,159 @@ exports[`MLocationBody <MLocationBody> with error displays correct fallback cont
|
||||||
Shared a location: Found at geo:51.5076,-0.1276 at 2021-12-21T12:22+0000
|
Shared a location: Found at geo:51.5076,-0.1276 at 2021-12-21T12:22+0000
|
||||||
</div>
|
</div>
|
||||||
`;
|
`;
|
||||||
|
|
||||||
|
exports[`MLocationBody <MLocationBody> without error renders map correctly 1`] = `
|
||||||
|
<MLocationBody
|
||||||
|
highlightLink=""
|
||||||
|
highlights={Array []}
|
||||||
|
mediaEventHelper={Object {}}
|
||||||
|
mxEvent={
|
||||||
|
Object {
|
||||||
|
"content": Object {
|
||||||
|
"body": "Found at geo:51.5076,-0.1276 at 2021-12-21T12:22+0000",
|
||||||
|
"geo_uri": "geo:51.5076,-0.1276",
|
||||||
|
"msgtype": "m.location",
|
||||||
|
"org.matrix.msc1767.text": "Found at geo:51.5076,-0.1276 at 2021-12-21T12:22+0000",
|
||||||
|
"org.matrix.msc3488.asset": Object {
|
||||||
|
"type": "m.pin",
|
||||||
|
},
|
||||||
|
"org.matrix.msc3488.location": Object {
|
||||||
|
"description": "Human-readable label",
|
||||||
|
"uri": "geo:51.5076,-0.1276",
|
||||||
|
},
|
||||||
|
"org.matrix.msc3488.ts": 252523,
|
||||||
|
},
|
||||||
|
"event_id": "$2",
|
||||||
|
"type": "org.matrix.msc3488.location",
|
||||||
|
}
|
||||||
|
}
|
||||||
|
onHeightChanged={[MockFunction]}
|
||||||
|
onMessageAllowed={[MockFunction]}
|
||||||
|
permalinkCreator={Object {}}
|
||||||
|
>
|
||||||
|
<LocationBodyContent
|
||||||
|
mapId="mx_MLocationBody_$2_1f9acffa"
|
||||||
|
mxEvent={
|
||||||
|
Object {
|
||||||
|
"content": Object {
|
||||||
|
"body": "Found at geo:51.5076,-0.1276 at 2021-12-21T12:22+0000",
|
||||||
|
"geo_uri": "geo:51.5076,-0.1276",
|
||||||
|
"msgtype": "m.location",
|
||||||
|
"org.matrix.msc1767.text": "Found at geo:51.5076,-0.1276 at 2021-12-21T12:22+0000",
|
||||||
|
"org.matrix.msc3488.asset": Object {
|
||||||
|
"type": "m.pin",
|
||||||
|
},
|
||||||
|
"org.matrix.msc3488.location": Object {
|
||||||
|
"description": "Human-readable label",
|
||||||
|
"uri": "geo:51.5076,-0.1276",
|
||||||
|
},
|
||||||
|
"org.matrix.msc3488.ts": 252523,
|
||||||
|
},
|
||||||
|
"event_id": "$2",
|
||||||
|
"type": "org.matrix.msc3488.location",
|
||||||
|
}
|
||||||
|
}
|
||||||
|
onClick={[Function]}
|
||||||
|
onError={[Function]}
|
||||||
|
tooltip="Expand map"
|
||||||
|
>
|
||||||
|
<div
|
||||||
|
className="mx_MLocationBody"
|
||||||
|
>
|
||||||
|
<TooltipTarget
|
||||||
|
alignment={5}
|
||||||
|
label="Expand map"
|
||||||
|
maxParentWidth={450}
|
||||||
|
>
|
||||||
|
<div
|
||||||
|
onBlur={[Function]}
|
||||||
|
onFocus={[Function]}
|
||||||
|
onMouseLeave={[Function]}
|
||||||
|
onMouseOver={[Function]}
|
||||||
|
tabIndex={0}
|
||||||
|
>
|
||||||
|
<Map
|
||||||
|
centerGeoUri="geo:51.5076,-0.1276"
|
||||||
|
className="mx_MLocationBody_map"
|
||||||
|
id="mx_MLocationBody_$2_1f9acffa"
|
||||||
|
onClick={[Function]}
|
||||||
|
onError={[Function]}
|
||||||
|
>
|
||||||
|
<div
|
||||||
|
className="mx_Map mx_MLocationBody_map"
|
||||||
|
id="mx_Map_mx_MLocationBody_$2_1f9acffa"
|
||||||
|
onClick={[Function]}
|
||||||
|
>
|
||||||
|
<SmartMarker
|
||||||
|
geoUri="geo:51.5076,-0.1276"
|
||||||
|
id="mx_MLocationBody_$2_1f9acffa-marker"
|
||||||
|
map={
|
||||||
|
MockMap {
|
||||||
|
"_events": Object {
|
||||||
|
"error": Array [
|
||||||
|
[Function],
|
||||||
|
[Function],
|
||||||
|
],
|
||||||
|
},
|
||||||
|
"_eventsCount": 1,
|
||||||
|
"_maxListeners": undefined,
|
||||||
|
"addControl": [MockFunction],
|
||||||
|
"removeControl": [MockFunction],
|
||||||
|
"setCenter": [MockFunction] {
|
||||||
|
"calls": Array [
|
||||||
|
Array [
|
||||||
|
Object {
|
||||||
|
"lat": 51.5076,
|
||||||
|
"lon": -0.1276,
|
||||||
|
},
|
||||||
|
],
|
||||||
|
Array [
|
||||||
|
Object {
|
||||||
|
"lat": 51.5076,
|
||||||
|
"lon": -0.1276,
|
||||||
|
},
|
||||||
|
],
|
||||||
|
],
|
||||||
|
"results": Array [
|
||||||
|
Object {
|
||||||
|
"type": "return",
|
||||||
|
"value": undefined,
|
||||||
|
},
|
||||||
|
Object {
|
||||||
|
"type": "return",
|
||||||
|
"value": undefined,
|
||||||
|
},
|
||||||
|
],
|
||||||
|
},
|
||||||
|
"setStyle": [MockFunction],
|
||||||
|
"zoomIn": [MockFunction],
|
||||||
|
"zoomOut": [MockFunction],
|
||||||
|
Symbol(kCapture): false,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
>
|
||||||
|
<ForwardRef
|
||||||
|
id="mx_MLocationBody_$2_1f9acffa-marker"
|
||||||
|
>
|
||||||
|
<div
|
||||||
|
className="mx_Marker mx_Marker_defaultColor"
|
||||||
|
id="mx_MLocationBody_$2_1f9acffa-marker"
|
||||||
|
>
|
||||||
|
<div
|
||||||
|
className="mx_Marker_border"
|
||||||
|
>
|
||||||
|
<div
|
||||||
|
className="mx_Marker_icon"
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</ForwardRef>
|
||||||
|
</SmartMarker>
|
||||||
|
</div>
|
||||||
|
</Map>
|
||||||
|
</div>
|
||||||
|
</TooltipTarget>
|
||||||
|
</div>
|
||||||
|
</LocationBodyContent>
|
||||||
|
</MLocationBody>
|
||||||
|
`;
|
||||||
|
|
Loading…
Reference in a new issue