* display live share warning only when geolocation is happening Signed-off-by: Kerry Archibald <kerrya@element.io> * kill beacons when geolocation is unavailable or permissions denied Signed-off-by: Kerry Archibald <kerrya@element.io> * polish and comments Signed-off-by: Kerry Archibald <kerrya@element.io>
169 lines
5.4 KiB
TypeScript
169 lines
5.4 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, { useCallback, useEffect, useState } from 'react';
|
|
import classNames from 'classnames';
|
|
import { Room, Beacon } from 'matrix-js-sdk/src/matrix';
|
|
|
|
import { _t } from '../../../languageHandler';
|
|
import { useEventEmitterState } from '../../../hooks/useEventEmitter';
|
|
import { OwnBeaconStore, OwnBeaconStoreEvent } from '../../../stores/OwnBeaconStore';
|
|
import AccessibleButton from '../elements/AccessibleButton';
|
|
import StyledLiveBeaconIcon from './StyledLiveBeaconIcon';
|
|
import { formatDuration } from '../../../DateUtils';
|
|
import { getBeaconMsUntilExpiry, sortBeaconsByLatestExpiry } from '../../../utils/beacon';
|
|
import Spinner from '../elements/Spinner';
|
|
import { useInterval } from '../../../hooks/useTimeout';
|
|
|
|
interface Props {
|
|
roomId: Room['roomId'];
|
|
}
|
|
|
|
const MINUTE_MS = 60000;
|
|
const HOUR_MS = MINUTE_MS * 60;
|
|
|
|
const getUpdateInterval = (ms: number) => {
|
|
// every 10 mins when more than an hour
|
|
if (ms > HOUR_MS) {
|
|
return MINUTE_MS * 10;
|
|
}
|
|
// every minute when more than a minute
|
|
if (ms > MINUTE_MS) {
|
|
return MINUTE_MS;
|
|
}
|
|
// otherwise every second
|
|
return 1000;
|
|
};
|
|
const useMsRemaining = (beacon: Beacon): number => {
|
|
const [msRemaining, setMsRemaining] = useState(() => getBeaconMsUntilExpiry(beacon));
|
|
|
|
useEffect(() => {
|
|
setMsRemaining(getBeaconMsUntilExpiry(beacon));
|
|
}, [beacon]);
|
|
|
|
const updateMsRemaining = useCallback(() => {
|
|
const ms = getBeaconMsUntilExpiry(beacon);
|
|
setMsRemaining(ms);
|
|
}, [beacon]);
|
|
|
|
useInterval(updateMsRemaining, getUpdateInterval(msRemaining));
|
|
|
|
return msRemaining;
|
|
};
|
|
|
|
/**
|
|
* It's technically possible to have multiple live beacons in one room
|
|
* Select the latest expiry to display,
|
|
* and kill all beacons on stop sharing
|
|
*/
|
|
type LiveBeaconsState = {
|
|
beacon?: Beacon;
|
|
onStopSharing?: () => void;
|
|
stoppingInProgress?: boolean;
|
|
};
|
|
const useLiveBeacons = (roomId: Room['roomId']): LiveBeaconsState => {
|
|
const [stoppingInProgress, setStoppingInProgress] = useState(false);
|
|
|
|
// do we have an active geolocation.watchPosition
|
|
const isMonitoringLiveLocation = useEventEmitterState(
|
|
OwnBeaconStore.instance,
|
|
OwnBeaconStoreEvent.MonitoringLivePosition,
|
|
() => OwnBeaconStore.instance.isMonitoringLiveLocation,
|
|
);
|
|
|
|
const liveBeaconIds = useEventEmitterState(
|
|
OwnBeaconStore.instance,
|
|
OwnBeaconStoreEvent.LivenessChange,
|
|
() => OwnBeaconStore.instance.getLiveBeaconIds(roomId),
|
|
);
|
|
|
|
// reset stopping in progress on change in live ids
|
|
useEffect(() => {
|
|
setStoppingInProgress(false);
|
|
}, [liveBeaconIds]);
|
|
|
|
if (!isMonitoringLiveLocation || !liveBeaconIds?.length) {
|
|
return {};
|
|
}
|
|
|
|
// select the beacon with latest expiry to display expiry time
|
|
const beacon = liveBeaconIds.map(beaconId => OwnBeaconStore.instance.getBeaconById(beaconId))
|
|
.sort(sortBeaconsByLatestExpiry)
|
|
.shift();
|
|
|
|
const onStopSharing = async () => {
|
|
setStoppingInProgress(true);
|
|
try {
|
|
await Promise.all(liveBeaconIds.map(beaconId => OwnBeaconStore.instance.stopBeacon(beaconId)));
|
|
} catch (error) {
|
|
// only clear loading in case of error
|
|
// to avoid flash of not-loading state
|
|
// after beacons have been stopped but we wait for sync
|
|
setStoppingInProgress(false);
|
|
}
|
|
};
|
|
|
|
return { onStopSharing, beacon, stoppingInProgress };
|
|
};
|
|
|
|
const LiveTimeRemaining: React.FC<{ beacon: Beacon }> = ({ beacon }) => {
|
|
const msRemaining = useMsRemaining(beacon);
|
|
|
|
const timeRemaining = formatDuration(msRemaining);
|
|
const liveTimeRemaining = _t(`%(timeRemaining)s left`, { timeRemaining });
|
|
|
|
return <span
|
|
data-test-id='room-live-share-expiry'
|
|
className="mx_RoomLiveShareWarning_expiry"
|
|
>{ liveTimeRemaining }</span>;
|
|
};
|
|
|
|
const RoomLiveShareWarning: React.FC<Props> = ({ roomId }) => {
|
|
const {
|
|
onStopSharing,
|
|
beacon,
|
|
stoppingInProgress,
|
|
} = useLiveBeacons(roomId);
|
|
|
|
if (!beacon) {
|
|
return null;
|
|
}
|
|
|
|
return <div
|
|
className={classNames('mx_RoomLiveShareWarning')}
|
|
>
|
|
<StyledLiveBeaconIcon className="mx_RoomLiveShareWarning_icon" />
|
|
<span className="mx_RoomLiveShareWarning_label">
|
|
{ _t('You are sharing your live location') }
|
|
</span>
|
|
|
|
{ stoppingInProgress ?
|
|
<span className='mx_RoomLiveShareWarning_spinner'><Spinner h={16} w={16} /></span> :
|
|
<LiveTimeRemaining beacon={beacon} />
|
|
}
|
|
<AccessibleButton
|
|
data-test-id='room-live-share-stop-sharing'
|
|
onClick={onStopSharing}
|
|
kind='danger'
|
|
element='button'
|
|
disabled={stoppingInProgress}
|
|
>
|
|
{ _t('Stop sharing') }
|
|
</AccessibleButton>
|
|
</div>;
|
|
};
|
|
|
|
export default RoomLiveShareWarning;
|