From b568a53c9f882ac4c35f7cc11561193f247eba9d Mon Sep 17 00:00:00 2001 From: Kerry Archibald Date: Fri, 18 Mar 2022 16:13:44 +0100 Subject: [PATCH 01/12] add duration dropdown to live location picker Signed-off-by: Kerry Archibald --- test/components/views/location/LocationPicker-test.tsx | 1 + 1 file changed, 1 insertion(+) diff --git a/test/components/views/location/LocationPicker-test.tsx b/test/components/views/location/LocationPicker-test.tsx index 914821a53de..93845dc0d35 100644 --- a/test/components/views/location/LocationPicker-test.tsx +++ b/test/components/views/location/LocationPicker-test.tsx @@ -146,6 +146,7 @@ describe("LocationPicker", () => { jest.spyOn(MatrixClientPeg, 'get').mockReturnValue(mockClient as unknown as MatrixClient); jest.clearAllMocks(); mocked(mockMap).addControl.mockReset(); + console.log('MM', maplibregl.Marker.mock.calls); mocked(findMapStyleUrl).mockReturnValue('tileserver.com'); }); From fc00cb89e0d60527e8bb5e45dee806e5edaa5581 Mon Sep 17 00:00:00 2001 From: Kerry Archibald Date: Fri, 18 Mar 2022 16:27:26 +0100 Subject: [PATCH 02/12] tidy comments Signed-off-by: Kerry Archibald --- test/components/views/location/LocationPicker-test.tsx | 1 - 1 file changed, 1 deletion(-) diff --git a/test/components/views/location/LocationPicker-test.tsx b/test/components/views/location/LocationPicker-test.tsx index 93845dc0d35..914821a53de 100644 --- a/test/components/views/location/LocationPicker-test.tsx +++ b/test/components/views/location/LocationPicker-test.tsx @@ -146,7 +146,6 @@ describe("LocationPicker", () => { jest.spyOn(MatrixClientPeg, 'get').mockReturnValue(mockClient as unknown as MatrixClient); jest.clearAllMocks(); mocked(mockMap).addControl.mockReset(); - console.log('MM', maplibregl.Marker.mock.calls); mocked(findMapStyleUrl).mockReturnValue('tileserver.com'); }); From 2638ffb8984b57ba3a25c94f244365584afe9e18 Mon Sep 17 00:00:00 2001 From: Kerry Archibald Date: Mon, 21 Mar 2022 10:25:29 +0100 Subject: [PATCH 03/12] setup component Signed-off-by: Kerry Archibald --- .../views/beacon/_RoomLiveShareWarning.scss | 4 + .../views/beacon/RoomLiveShareWarning.tsx | 49 ++++++++++++ .../beacon/RoomLiveShareWarning-test.tsx | 75 +++++++++++++++++++ .../RoomLiveShareWarning-test.tsx.snap | 12 +++ 4 files changed, 140 insertions(+) create mode 100644 res/css/components/views/beacon/_RoomLiveShareWarning.scss create mode 100644 src/components/views/beacon/RoomLiveShareWarning.tsx create mode 100644 test/components/views/beacon/RoomLiveShareWarning-test.tsx create mode 100644 test/components/views/beacon/__snapshots__/RoomLiveShareWarning-test.tsx.snap diff --git a/res/css/components/views/beacon/_RoomLiveShareWarning.scss b/res/css/components/views/beacon/_RoomLiveShareWarning.scss new file mode 100644 index 00000000000..37cbeda4b7c --- /dev/null +++ b/res/css/components/views/beacon/_RoomLiveShareWarning.scss @@ -0,0 +1,4 @@ + +.mx_RoomLiveShareWarning { + +} diff --git a/src/components/views/beacon/RoomLiveShareWarning.tsx b/src/components/views/beacon/RoomLiveShareWarning.tsx new file mode 100644 index 00000000000..f8fd564fa76 --- /dev/null +++ b/src/components/views/beacon/RoomLiveShareWarning.tsx @@ -0,0 +1,49 @@ +/* +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 classNames from 'classnames'; +import React from 'react'; +import { Room } from 'matrix-js-sdk/src/matrix'; + +import { useEventEmitterState } from '../../../hooks/useEventEmitter'; +import { _t } from '../../../languageHandler'; +import { OwnBeaconStore, OwnBeaconStoreEvent } from '../../../stores/OwnBeaconStore'; +import { Icon as LiveLocationIcon } from '../../../../res/img/location/live-location.svg'; + +interface Props { + roomId: Room['roomId']; +} + +const RoomLiveShareWarning: React.FC = ({ roomId }) => { + const hasLiveBeacons = useEventEmitterState( + OwnBeaconStore.instance, + OwnBeaconStoreEvent.LivenessChange, + () => OwnBeaconStore.instance.hasLiveBeacons(roomId), + ); + + if (!hasLiveBeacons) { + return null; + } + + return
+ + { _t('You are sharing your live location') } +
; +}; + +export default RoomLiveShareWarning; diff --git a/test/components/views/beacon/RoomLiveShareWarning-test.tsx b/test/components/views/beacon/RoomLiveShareWarning-test.tsx new file mode 100644 index 00000000000..8cb72bccbc1 --- /dev/null +++ b/test/components/views/beacon/RoomLiveShareWarning-test.tsx @@ -0,0 +1,75 @@ +/* +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 { mocked } from 'jest-mock'; +import { mount } from 'enzyme'; + +import '../../../skinned-sdk'; +import RoomLiveShareWarning from '../../../../src/components/views/beacon/RoomLiveShareWarning'; +import { OwnBeaconStore, OwnBeaconStoreEvent } from '../../../../src/stores/OwnBeaconStore'; +import { flushPromises } from '../../../test-utils'; + +jest.mock('../../../../src/stores/OwnBeaconStore', () => { + // eslint-disable-next-line @typescript-eslint/no-var-requires + const EventEmitter = require("events"); + class MockOwnBeaconStore extends EventEmitter { + public hasLiveBeacons = jest.fn().mockReturnValue(false); + } + return { + // @ts-ignore + ...jest.requireActual('../../../../src/stores/OwnBeaconStore'), + OwnBeaconStore: { + instance: new MockOwnBeaconStore() as unknown as OwnBeaconStore, + }, + }; +}, +); + +describe('', () => { + const defaultProps = {}; + const getComponent = (props = {}) => + mount(); + + it('renders nothing when user has no live beacons', () => { + const component = getComponent(); + expect(component.html()).toBe(null); + }); + + describe('when user has live beacons', () => { + beforeEach(() => { + mocked(OwnBeaconStore.instance).hasLiveBeacons.mockReturnValue(true); + }); + it('renders correctly when not minimized', () => { + const component = getComponent(); + expect(component).toMatchSnapshot(); + }); + + it('removes itself when user stops having live beacons', async () => { + const component = getComponent(); + // started out rendered + expect(component.html()).toBeTruthy(); + + mocked(OwnBeaconStore.instance).hasLiveBeacons.mockReturnValue(false); + OwnBeaconStore.instance.emit(OwnBeaconStoreEvent.LivenessChange, false); + + await flushPromises(); + component.setProps({}); + + expect(component.html()).toBe(null); + }); + }); +}); diff --git a/test/components/views/beacon/__snapshots__/RoomLiveShareWarning-test.tsx.snap b/test/components/views/beacon/__snapshots__/RoomLiveShareWarning-test.tsx.snap new file mode 100644 index 00000000000..ab5008a8ee5 --- /dev/null +++ b/test/components/views/beacon/__snapshots__/RoomLiveShareWarning-test.tsx.snap @@ -0,0 +1,12 @@ +// Jest Snapshot v1, https://goo.gl/fbAQLP + +exports[` when user has live beacons renders correctly when not minimized 1`] = ` + +
+
+ You are sharing your live location +
+ +`; From 08c733ef7e87ae23969a360473fc340ce839157a Mon Sep 17 00:00:00 2001 From: Kerry Archibald Date: Mon, 21 Mar 2022 11:37:36 +0100 Subject: [PATCH 04/12] replace references to beaconInfoId with beacon.identifier Signed-off-by: Kerry Archibald --- res/css/_components.scss | 1 + .../views/beacon/_RoomLiveShareWarning.scss | 22 +++++++ .../views/beacon/RoomLiveShareWarning.tsx | 39 +++++++++++-- src/components/views/rooms/RoomHeader.tsx | 2 + src/i18n/strings/en_EN.json | 1 + src/stores/OwnBeaconStore.ts | 27 +++++---- .../beacon/RoomLiveShareWarning-test.tsx | 8 ++- test/stores/OwnBeaconStore-test.ts | 57 +++++++++++++------ test/test-utils/beacon.ts | 3 +- 9 files changed, 126 insertions(+), 34 deletions(-) diff --git a/res/css/_components.scss b/res/css/_components.scss index f0d78df5bc1..e607deffe81 100644 --- a/res/css/_components.scss +++ b/res/css/_components.scss @@ -5,6 +5,7 @@ @import "./_font-weights.scss"; @import "./_spacing.scss"; @import "./components/views/beacon/_LeftPanelLiveShareWarning.scss"; +@import "./components/views/beacon/_RoomLiveShareWarning.scss"; @import "./components/views/beacon/_StyledLiveBeaconIcon.scss"; @import "./components/views/location/_LiveDurationDropdown.scss"; @import "./components/views/location/_LocationShareMenu.scss"; diff --git a/res/css/components/views/beacon/_RoomLiveShareWarning.scss b/res/css/components/views/beacon/_RoomLiveShareWarning.scss index 37cbeda4b7c..05ad445729a 100644 --- a/res/css/components/views/beacon/_RoomLiveShareWarning.scss +++ b/res/css/components/views/beacon/_RoomLiveShareWarning.scss @@ -1,4 +1,26 @@ .mx_RoomLiveShareWarning { + width: 100%; + display: flex; + flex-direction: row; + align-items: center; + + box-sizing: border-box; + padding: $spacing-12 $spacing-16; + + color: $primary-content; + background-color: $system; +} + +.mx_RoomLiveShareWarning_label { + flex: 1; + font-size: $font-15px; } + + +.mx_RoomLiveShareWarning_expiry { + color: $secondary-content; + font-size: $font-12px; + margin-right: $spacing-16; +} \ No newline at end of file diff --git a/src/components/views/beacon/RoomLiveShareWarning.tsx b/src/components/views/beacon/RoomLiveShareWarning.tsx index f8fd564fa76..ef86ed6310b 100644 --- a/src/components/views/beacon/RoomLiveShareWarning.tsx +++ b/src/components/views/beacon/RoomLiveShareWarning.tsx @@ -22,27 +22,58 @@ import { useEventEmitterState } from '../../../hooks/useEventEmitter'; import { _t } from '../../../languageHandler'; import { OwnBeaconStore, OwnBeaconStoreEvent } from '../../../stores/OwnBeaconStore'; import { Icon as LiveLocationIcon } from '../../../../res/img/location/live-location.svg'; +import AccessibleButton from '../elements/AccessibleButton'; interface Props { roomId: Room['roomId']; } const RoomLiveShareWarning: React.FC = ({ roomId }) => { - const hasLiveBeacons = useEventEmitterState( + const liveBeaconIds = useEventEmitterState( OwnBeaconStore.instance, OwnBeaconStoreEvent.LivenessChange, - () => OwnBeaconStore.instance.hasLiveBeacons(roomId), + () => OwnBeaconStore.instance.getLiveBeaconIds(roomId), ); - if (!hasLiveBeacons) { + if (!liveBeaconIds?.length) { return null; } + if (liveBeaconIds.length > 1) { + throw new Error('not handled yet'); + } + + const beaconId = liveBeaconIds[0]; + + const beacon = OwnBeaconStore.instance.getBeaconById(liveBeaconIds[0]); + const liveTimeRemaining = `${beacon.beaconInfo.timeout}`; + + + const onStopSharing = () => { + OwnBeaconStore.instance.stopBeacon(beaconId) + } + + return
- + {/* */ } + + { _t('You are sharing your live location') } + + { liveTimeRemaining } + + { _t('Stop sharing') } +
; }; diff --git a/src/components/views/rooms/RoomHeader.tsx b/src/components/views/rooms/RoomHeader.tsx index c0a07181114..08e6d488efb 100644 --- a/src/components/views/rooms/RoomHeader.tsx +++ b/src/components/views/rooms/RoomHeader.tsx @@ -41,6 +41,7 @@ import { RoomNotificationStateStore } from '../../../stores/notifications/RoomNo import { RightPanelPhases } from '../../../stores/right-panel/RightPanelStorePhases'; import { NotificationStateEvents } from '../../../stores/notifications/NotificationState'; import RoomContext from "../../../contexts/RoomContext"; +import RoomLiveShareWarning from '../beacon/RoomLiveShareWarning'; export interface ISearchInfo { searchTerm: string; @@ -273,6 +274,7 @@ export default class RoomHeader extends React.Component { { rightRow }
+ ); } diff --git a/src/i18n/strings/en_EN.json b/src/i18n/strings/en_EN.json index 6b94b7a85df..33c14fa99b0 100644 --- a/src/i18n/strings/en_EN.json +++ b/src/i18n/strings/en_EN.json @@ -2965,6 +2965,7 @@ "Leave the beta": "Leave the beta", "Join the beta": "Join the beta", "You are sharing your live location": "You are sharing your live location", + "Stop sharing": "Stop sharing", "Avatar": "Avatar", "This room is public": "This room is public", "Away": "Away", diff --git a/src/stores/OwnBeaconStore.ts b/src/stores/OwnBeaconStore.ts index 6bd256be2cf..2a0163c5c73 100644 --- a/src/stores/OwnBeaconStore.ts +++ b/src/stores/OwnBeaconStore.ts @@ -41,6 +41,7 @@ type OwnBeaconStoreState = { }; export class OwnBeaconStore extends AsyncStoreWithClient { private static internalInstance = new OwnBeaconStore(); + // users beacons, keyed by event type public readonly beacons = new Map(); public readonly beaconsByRoomId = new Map>(); private liveBeaconIds = []; @@ -86,8 +87,12 @@ export class OwnBeaconStore extends AsyncStoreWithClient { return this.liveBeaconIds.filter(beaconId => this.beaconsByRoomId.get(roomId)?.has(beaconId)); } - public stopBeacon = async (beaconInfoId: string): Promise => { - const beacon = this.beacons.get(beaconInfoId); + public getBeaconById(beaconId: string): Beacon | undefined { + return this.beacons.get(beaconId); + } + + public stopBeacon = async (beaconInfoType: string): Promise => { + const beacon = this.beacons.get(beaconInfoType); // if no beacon, or beacon is already explicitly set isLive: false // do nothing if (!beacon?.beaconInfo?.live) { @@ -107,22 +112,22 @@ export class OwnBeaconStore extends AsyncStoreWithClient { private onBeaconLiveness = (isLive: boolean, beacon: Beacon): void => { // check if we care about this beacon - if (!this.beacons.has(beacon.beaconInfoId)) { + if (!this.beacons.has(beacon.identifier)) { return; } - if (!isLive && this.liveBeaconIds.includes(beacon.beaconInfoId)) { + if (!isLive && this.liveBeaconIds.includes(beacon.identifier)) { this.liveBeaconIds = - this.liveBeaconIds.filter(beaconId => beaconId !== beacon.beaconInfoId); + this.liveBeaconIds.filter(beaconId => beaconId !== beacon.identifier); } - if (isLive && !this.liveBeaconIds.includes(beacon.beaconInfoId)) { - this.liveBeaconIds.push(beacon.beaconInfoId); + if (isLive && !this.liveBeaconIds.includes(beacon.identifier)) { + this.liveBeaconIds.push(beacon.identifier); } // beacon expired, update beacon to un-alive state if (!isLive) { - this.stopBeacon(beacon.beaconInfoId); + this.stopBeacon(beacon.identifier); } // TODO start location polling here @@ -146,13 +151,13 @@ export class OwnBeaconStore extends AsyncStoreWithClient { }; private addBeacon = (beacon: Beacon): void => { - this.beacons.set(beacon.beaconInfoId, beacon); + this.beacons.set(beacon.identifier, beacon); if (!this.beaconsByRoomId.has(beacon.roomId)) { this.beaconsByRoomId.set(beacon.roomId, new Set()); } - this.beaconsByRoomId.get(beacon.roomId).add(beacon.beaconInfoId); + this.beaconsByRoomId.get(beacon.roomId).add(beacon.identifier); beacon.monitorLiveness(); }; @@ -161,7 +166,7 @@ export class OwnBeaconStore extends AsyncStoreWithClient { const prevLiveness = this.hasLiveBeacons(); this.liveBeaconIds = [...this.beacons.values()] .filter(beacon => beacon.isLive) - .map(beacon => beacon.beaconInfoId); + .map(beacon => beacon.identifier); const newLiveness = this.hasLiveBeacons(); diff --git a/test/components/views/beacon/RoomLiveShareWarning-test.tsx b/test/components/views/beacon/RoomLiveShareWarning-test.tsx index 8cb72bccbc1..def8c288507 100644 --- a/test/components/views/beacon/RoomLiveShareWarning-test.tsx +++ b/test/components/views/beacon/RoomLiveShareWarning-test.tsx @@ -28,6 +28,7 @@ jest.mock('../../../../src/stores/OwnBeaconStore', () => { const EventEmitter = require("events"); class MockOwnBeaconStore extends EventEmitter { public hasLiveBeacons = jest.fn().mockReturnValue(false); + public getLiveBeaconIds = jest.fn().mockReturnValue([]); } return { // @ts-ignore @@ -40,7 +41,9 @@ jest.mock('../../../../src/stores/OwnBeaconStore', () => { ); describe('', () => { - const defaultProps = {}; + const defaultProps = { + roomId: '!room:server.org', + }; const getComponent = (props = {}) => mount(); @@ -49,9 +52,10 @@ describe('', () => { expect(component.html()).toBe(null); }); - describe('when user has live beacons', () => { + xdescribe('when user has live beacons', () => { beforeEach(() => { mocked(OwnBeaconStore.instance).hasLiveBeacons.mockReturnValue(true); + mocked(OwnBeaconStore.instance).getLiveBeaconIds.mockReturnValue([]); }); it('renders correctly when not minimized', () => { const component = getComponent(); diff --git a/test/stores/OwnBeaconStore-test.ts b/test/stores/OwnBeaconStore-test.ts index 6c8cacd3df9..a4055bc6186 100644 --- a/test/stores/OwnBeaconStore-test.ts +++ b/test/stores/OwnBeaconStore-test.ts @@ -46,11 +46,36 @@ xdescribe('OwnBeaconStore', () => { // event creation sets timestamp to Date.now() jest.spyOn(global.Date, 'now').mockReturnValue(now - HOUR_MS); - const alicesRoom1BeaconInfo = makeBeaconInfoEvent(aliceId, room1Id, { isLive: true }, '$alice-room1-1'); - const alicesRoom2BeaconInfo = makeBeaconInfoEvent(aliceId, room2Id, { isLive: true }, '$alice-room2-1'); - const alicesOldRoomIdBeaconInfo = makeBeaconInfoEvent(aliceId, room1Id, { isLive: false }, '$alice-room1-2'); - const bobsRoom1BeaconInfo = makeBeaconInfoEvent(bobId, room1Id, { isLive: true }, '$bob-room1-1'); - const bobsOldRoom1BeaconInfo = makeBeaconInfoEvent(bobId, room1Id, { isLive: false }, '$bob-room1-2'); + const alicesRoom1BeaconInfo = makeBeaconInfoEvent(aliceId, + room1Id, + { isLive: true }, + '$alice-room1-1' + , '$alice-room1-1', + ); + const alicesRoom2BeaconInfo = makeBeaconInfoEvent(aliceId, + room2Id, + { isLive: true }, + '$alice-room2-1' + , '$alice-room2-1', + ); + const alicesOldRoomIdBeaconInfo = makeBeaconInfoEvent(aliceId, + room1Id, + { isLive: false }, + '$alice-room1-2' + , '$alice-room1-2', + ); + const bobsRoom1BeaconInfo = makeBeaconInfoEvent(bobId, + room1Id, + { isLive: true }, + '$bob-room1-1' + , '$bob-room1-1', + ); + const bobsOldRoom1BeaconInfo = makeBeaconInfoEvent(bobId, + room1Id, + { isLive: false }, + '$bob-room1-2' + , '$bob-room1-2', + ); // make fresh rooms every time // as we update room state @@ -121,8 +146,8 @@ xdescribe('OwnBeaconStore', () => { const store = await makeOwnBeaconStore(); expect(store.hasLiveBeacons()).toBe(true); expect(store.getLiveBeaconIds()).toEqual([ - alicesRoom1BeaconInfo.getId(), - alicesRoom2BeaconInfo.getId(), + alicesRoom1BeaconInfo.getType(), + alicesRoom2BeaconInfo.getType(), ]); }); }); @@ -143,7 +168,7 @@ xdescribe('OwnBeaconStore', () => { alicesRoom1BeaconInfo, ]); const store = await makeOwnBeaconStore(); - const beacon = room1.currentState.beacons.get(alicesRoom1BeaconInfo.getId()); + const beacon = room1.currentState.beacons.get(alicesRoom1BeaconInfo.getType()); const destroySpy = jest.spyOn(beacon, 'destroy'); // @ts-ignore store.onNotReady(); @@ -226,7 +251,7 @@ xdescribe('OwnBeaconStore', () => { ]); const store = await makeOwnBeaconStore(); expect(store.getLiveBeaconIds()).toEqual([ - alicesRoom1BeaconInfo.getId(), + alicesRoom1BeaconInfo.getType(), ]); }); @@ -249,10 +274,10 @@ xdescribe('OwnBeaconStore', () => { ]); const store = await makeOwnBeaconStore(); expect(store.getLiveBeaconIds(room1Id)).toEqual([ - alicesRoom1BeaconInfo.getId(), + alicesRoom1BeaconInfo.getType(), ]); expect(store.getLiveBeaconIds(room2Id)).toEqual([ - alicesRoom2BeaconInfo.getId(), + alicesRoom2BeaconInfo.getType(), ]); }); @@ -400,7 +425,7 @@ xdescribe('OwnBeaconStore', () => { const emitSpy = jest.spyOn(store, 'emit'); const alicesBeacon = new Beacon(alicesOldRoomIdBeaconInfo); const liveUpdate = makeBeaconInfoEvent( - aliceId, room1Id, { isLive: true }, alicesOldRoomIdBeaconInfo.getId(), + aliceId, room1Id, { isLive: true }, alicesOldRoomIdBeaconInfo.getId(), '$alice-room1-2', ); // bring the beacon back to life @@ -437,10 +462,10 @@ xdescribe('OwnBeaconStore', () => { it('updates beacon to live:false when it is unexpired', async () => { const store = await makeOwnBeaconStore(); - await store.stopBeacon(alicesOldRoomIdBeaconInfo.getId()); + await store.stopBeacon(alicesOldRoomIdBeaconInfo.getType()); const prevEventContent = alicesRoom1BeaconInfo.getContent(); - await store.stopBeacon(alicesRoom1BeaconInfo.getId()); + await store.stopBeacon(alicesRoom1BeaconInfo.getType()); // matches original state of event content // except for live property @@ -461,13 +486,13 @@ xdescribe('OwnBeaconStore', () => { it('updates beacon to live:false when it is expired but live property is true', async () => { const store = await makeOwnBeaconStore(); - await store.stopBeacon(alicesOldRoomIdBeaconInfo.getId()); + await store.stopBeacon(alicesOldRoomIdBeaconInfo.getType()); const prevEventContent = alicesRoom1BeaconInfo.getContent(); // time travel until beacon is expired advanceDateAndTime(HOUR_MS * 3); - await store.stopBeacon(alicesRoom1BeaconInfo.getId()); + await store.stopBeacon(alicesRoom1BeaconInfo.getType()); // matches original state of event content // except for live property diff --git a/test/test-utils/beacon.ts b/test/test-utils/beacon.ts index d0e63e6b4b6..d156baaa5e6 100644 --- a/test/test-utils/beacon.ts +++ b/test/test-utils/beacon.ts @@ -42,6 +42,7 @@ export const makeBeaconInfoEvent = ( roomId: string, contentProps: Partial = {}, eventId?: string, + eventTypeSuffix?: string, ): MatrixEvent => { const { timeout, @@ -54,7 +55,7 @@ export const makeBeaconInfoEvent = ( ...contentProps, }; const event = new MatrixEvent({ - type: `${M_BEACON_INFO.name}.${sender}.${++count}`, + type: `${M_BEACON_INFO.name}.${sender}.${eventTypeSuffix || ++count}`, room_id: roomId, state_key: sender, content: makeBeaconInfoContent(timeout, isLive, description, assetType, timestamp), From 1bbed141c6fba9d9d0eac5009fed82c522c2937d Mon Sep 17 00:00:00 2001 From: Kerry Archibald Date: Mon, 21 Mar 2022 11:38:31 +0100 Subject: [PATCH 05/12] icon Signed-off-by: Kerry Archibald --- src/components/views/beacon/RoomLiveShareWarning.tsx | 10 ++++------ src/components/views/rooms/RoomHeader.tsx | 2 +- 2 files changed, 5 insertions(+), 7 deletions(-) diff --git a/src/components/views/beacon/RoomLiveShareWarning.tsx b/src/components/views/beacon/RoomLiveShareWarning.tsx index ef86ed6310b..5e2fecece22 100644 --- a/src/components/views/beacon/RoomLiveShareWarning.tsx +++ b/src/components/views/beacon/RoomLiveShareWarning.tsx @@ -48,19 +48,17 @@ const RoomLiveShareWarning: React.FC = ({ roomId }) => { const beacon = OwnBeaconStore.instance.getBeaconById(liveBeaconIds[0]); const liveTimeRemaining = `${beacon.beaconInfo.timeout}`; - const onStopSharing = () => { - OwnBeaconStore.instance.stopBeacon(beaconId) - } - + OwnBeaconStore.instance.stopBeacon(beaconId); + }; return
- {/* */ } + - { _t('You are sharing your live location') } + { _t('You are sharing your live location') } { { rightRow }
- + ); } From c34317f4ecd6633cad737d3bfc087c626a4f0a41 Mon Sep 17 00:00:00 2001 From: Kerry Archibald Date: Mon, 21 Mar 2022 12:37:29 +0100 Subject: [PATCH 06/12] component for styled live beacon icon Signed-off-by: Kerry Archibald --- .../views/beacon/StyledLiveBeaconIcon-test.tsx | 17 +++++++++++++++++ 1 file changed, 17 insertions(+) create mode 100644 test/components/views/beacon/StyledLiveBeaconIcon-test.tsx diff --git a/test/components/views/beacon/StyledLiveBeaconIcon-test.tsx b/test/components/views/beacon/StyledLiveBeaconIcon-test.tsx new file mode 100644 index 00000000000..81cbdbd28b4 --- /dev/null +++ b/test/components/views/beacon/StyledLiveBeaconIcon-test.tsx @@ -0,0 +1,17 @@ + +import React from 'react'; +import { mount } from 'enzyme'; + +import '../../../skinned-sdk'; +import StyledLiveBeaconIcon from '../../../../src/components/views/beacon/StyledLiveBeaconIcon'; + +describe('', () => { + const defaultProps = {}; + const getComponent = (props = {}) => + mount(); + + it('renders', () => { + const component = getComponent(); + expect(component).toBeTruthy(); + }); +}); From 1afc8bd28b02cd9ca28a56258d734ae6a7d4b7ba Mon Sep 17 00:00:00 2001 From: Kerry Archibald Date: Mon, 21 Mar 2022 12:40:25 +0100 Subject: [PATCH 07/12] emit liveness change whenever livebeaconIds changes Signed-off-by: Kerry Archibald --- src/stores/OwnBeaconStore.ts | 13 ++++++------- test/stores/OwnBeaconStore-test.ts | 13 ++++++++----- 2 files changed, 14 insertions(+), 12 deletions(-) diff --git a/src/stores/OwnBeaconStore.ts b/src/stores/OwnBeaconStore.ts index 2a0163c5c73..5fcbfa9a903 100644 --- a/src/stores/OwnBeaconStore.ts +++ b/src/stores/OwnBeaconStore.ts @@ -27,11 +27,12 @@ import { import defaultDispatcher from "../dispatcher/dispatcher"; import { ActionPayload } from "../dispatcher/payloads"; import { AsyncStoreWithClient } from "./AsyncStoreWithClient"; +import { arrayHasDiff } from "../utils/arrays"; const isOwnBeacon = (beacon: Beacon, userId: string): boolean => beacon.beaconInfoOwner === userId; export enum OwnBeaconStoreEvent { - LivenessChange = 'OwnBeaconStore.LivenessChange' + LivenessChange = 'OwnBeaconStore.LivenessChange', } type OwnBeaconStoreState = { @@ -132,7 +133,7 @@ export class OwnBeaconStore extends AsyncStoreWithClient { // TODO start location polling here - this.emit(OwnBeaconStoreEvent.LivenessChange, this.hasLiveBeacons()); + this.emit(OwnBeaconStoreEvent.LivenessChange, this.getLiveBeaconIds()); }; private initialiseBeaconState = () => { @@ -163,15 +164,13 @@ export class OwnBeaconStore extends AsyncStoreWithClient { }; private checkLiveness = (): void => { - const prevLiveness = this.hasLiveBeacons(); + const prevLiveness = this.getLiveBeaconIds(); this.liveBeaconIds = [...this.beacons.values()] .filter(beacon => beacon.isLive) .map(beacon => beacon.identifier); - const newLiveness = this.hasLiveBeacons(); - - if (prevLiveness !== newLiveness) { - this.emit(OwnBeaconStoreEvent.LivenessChange, newLiveness); + if (arrayHasDiff(prevLiveness, this.liveBeaconIds)) { + this.emit(OwnBeaconStoreEvent.LivenessChange, this.liveBeaconIds); } }; diff --git a/test/stores/OwnBeaconStore-test.ts b/test/stores/OwnBeaconStore-test.ts index a4055bc6186..a9b1f839a1f 100644 --- a/test/stores/OwnBeaconStore-test.ts +++ b/test/stores/OwnBeaconStore-test.ts @@ -328,10 +328,10 @@ xdescribe('OwnBeaconStore', () => { mockClient.emit(BeaconEvent.New, alicesRoom1BeaconInfo, alicesLiveBeacon); - expect(emitSpy).toHaveBeenCalledWith(OwnBeaconStoreEvent.LivenessChange, true); + expect(emitSpy).toHaveBeenCalledWith(OwnBeaconStoreEvent.LivenessChange, [alicesRoom1BeaconInfo.getType()]); }); - it('does not emit a liveness change event when new beacons do not change live state', async () => { + it('emits a liveness change event when new beacons do not change live state', async () => { makeRoomsWithStateEvents([ alicesRoom2BeaconInfo, ]); @@ -343,7 +343,7 @@ xdescribe('OwnBeaconStore', () => { mockClient.emit(BeaconEvent.New, alicesRoom1BeaconInfo, alicesLiveBeacon); - expect(emitSpy).not.toHaveBeenCalled(); + expect(emitSpy).toHaveBeenCalled(); }); }); @@ -382,7 +382,7 @@ xdescribe('OwnBeaconStore', () => { expect(store.hasLiveBeacons()).toBe(false); expect(store.hasLiveBeacons(room1Id)).toBe(false); - expect(emitSpy).toHaveBeenCalledWith(OwnBeaconStoreEvent.LivenessChange, false); + expect(emitSpy).toHaveBeenCalledWith(OwnBeaconStoreEvent.LivenessChange, []); }); it('stops beacon when liveness changes from true to false and beacon is expired', async () => { @@ -435,7 +435,10 @@ xdescribe('OwnBeaconStore', () => { expect(store.hasLiveBeacons()).toBe(true); expect(store.hasLiveBeacons(room1Id)).toBe(true); - expect(emitSpy).toHaveBeenCalledWith(OwnBeaconStoreEvent.LivenessChange, true); + expect(emitSpy).toHaveBeenCalledWith( + OwnBeaconStoreEvent.LivenessChange, + [alicesOldRoomIdBeaconInfo.getType()] + ); }); }); From 549fa37b399d41e966a2d9c92ebdc94f77535093 Mon Sep 17 00:00:00 2001 From: Kerry Archibald Date: Mon, 21 Mar 2022 16:40:36 +0100 Subject: [PATCH 08/12] Handle multiple live beacons in room share warning, test Signed-off-by: Kerry Archibald --- .../views/beacon/_RoomLiveShareWarning.scss | 14 +- .../views/beacon/RoomLiveShareWarning.tsx | 92 ++++++-- src/i18n/strings/en_EN.json | 3 + .../beacon/RoomLiveShareWarning-test.tsx | 202 ++++++++++++++---- .../RoomLiveShareWarning-test.tsx.snap | 97 ++++++++- test/stores/OwnBeaconStore-test.ts | 2 +- 6 files changed, 344 insertions(+), 66 deletions(-) diff --git a/res/css/components/views/beacon/_RoomLiveShareWarning.scss b/res/css/components/views/beacon/_RoomLiveShareWarning.scss index 05ad445729a..b18a00ffd9c 100644 --- a/res/css/components/views/beacon/_RoomLiveShareWarning.scss +++ b/res/css/components/views/beacon/_RoomLiveShareWarning.scss @@ -1,4 +1,3 @@ - .mx_RoomLiveShareWarning { width: 100%; @@ -13,14 +12,23 @@ background-color: $system; } +.mx_RoomLiveShareWarning_icon { + height: 32px; + width: 32px; + margin-right: $spacing-8; +} + .mx_RoomLiveShareWarning_label { flex: 1; font-size: $font-15px; } - .mx_RoomLiveShareWarning_expiry { color: $secondary-content; font-size: $font-12px; margin-right: $spacing-16; -} \ No newline at end of file +} + +.mx_RoomLiveShareWarning_spinner { + margin-right: $spacing-16; +} diff --git a/src/components/views/beacon/RoomLiveShareWarning.tsx b/src/components/views/beacon/RoomLiveShareWarning.tsx index 5e2fecece22..5958263af8f 100644 --- a/src/components/views/beacon/RoomLiveShareWarning.tsx +++ b/src/components/views/beacon/RoomLiveShareWarning.tsx @@ -14,61 +14,109 @@ See the License for the specific language governing permissions and limitations under the License. */ +import React, { useEffect, useState } from 'react'; import classNames from 'classnames'; -import React from 'react'; import { Room } from 'matrix-js-sdk/src/matrix'; -import { useEventEmitterState } from '../../../hooks/useEventEmitter'; import { _t } from '../../../languageHandler'; +import { useEventEmitterState } from '../../../hooks/useEventEmitter'; import { OwnBeaconStore, OwnBeaconStoreEvent } from '../../../stores/OwnBeaconStore'; -import { Icon as LiveLocationIcon } from '../../../../res/img/location/live-location.svg'; import AccessibleButton from '../elements/AccessibleButton'; +import StyledLiveBeaconIcon from './StyledLiveBeaconIcon'; +import { formatDuration } from '../../../DateUtils'; +import { getBeaconMsUntilExpiry, sortBeaconsByLatestExpiry } from '../../../utils/beacon'; +import Spinner from '../elements/Spinner'; interface Props { roomId: Room['roomId']; } -const RoomLiveShareWarning: React.FC = ({ roomId }) => { +/** + * 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 = { + liveBeaconIds: string[]; + msRemaining?: number; + onStopSharing?: () => void; + stoppingInProgress?: boolean; +}; +const useLiveBeacons = (roomId: Room['roomId']): LiveBeaconsState => { + const [stoppingInProgress, setStoppingInProgress] = useState(false); const liveBeaconIds = useEventEmitterState( OwnBeaconStore.instance, OwnBeaconStoreEvent.LivenessChange, () => OwnBeaconStore.instance.getLiveBeaconIds(roomId), ); + // reset stopping in progress on change in live ids + useEffect(() => { + setStoppingInProgress(false); + }, [liveBeaconIds, setStoppingInProgress]); + if (!liveBeaconIds?.length) { - return null; + return { liveBeaconIds }; } - if (liveBeaconIds.length > 1) { - throw new Error('not handled yet'); - } + // select the beacon with latest expiry to display expiry time + const beacon = liveBeaconIds.map(beaconId => OwnBeaconStore.instance.getBeaconById(beaconId)) + .sort(sortBeaconsByLatestExpiry) + .shift(); - const beaconId = liveBeaconIds[0]; + 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); + } + }; - const beacon = OwnBeaconStore.instance.getBeaconById(liveBeaconIds[0]); - const liveTimeRemaining = `${beacon.beaconInfo.timeout}`; + const msRemaining = getBeaconMsUntilExpiry(beacon); - const onStopSharing = () => { - OwnBeaconStore.instance.stopBeacon(beaconId); - }; + return { liveBeaconIds, onStopSharing, msRemaining, stoppingInProgress }; +}; + +const RoomLiveShareWarning: React.FC = ({ roomId }) => { + const { + liveBeaconIds, + onStopSharing, + msRemaining, + stoppingInProgress, + } = useLiveBeacons(roomId); + + if (!liveBeaconIds?.length) { + return null; + } + + const timeRemaining = formatDuration(msRemaining); + const liveTimeRemaining = _t(`%(timeRemaining)s left`, { timeRemaining }); return
- + - - { _t('You are sharing your live location') } + { _t('You are sharing %(count)s live locations', { count: liveBeaconIds.length }) } - { liveTimeRemaining } + + { stoppingInProgress ? + : + { liveTimeRemaining } + } { _t('Stop sharing') } diff --git a/src/i18n/strings/en_EN.json b/src/i18n/strings/en_EN.json index 33c14fa99b0..9d777293d08 100644 --- a/src/i18n/strings/en_EN.json +++ b/src/i18n/strings/en_EN.json @@ -2965,6 +2965,9 @@ "Leave the beta": "Leave the beta", "Join the beta": "Join the beta", "You are sharing your live location": "You are sharing your live location", + "%(timeRemaining)s left": "%(timeRemaining)s left", + "You are sharing %(count)s live locations|one": "You are sharing your live location", + "You are sharing %(count)s live locations|other": "You are sharing %(count)s live locations", "Stop sharing": "Stop sharing", "Avatar": "Avatar", "This room is public": "This room is public", diff --git a/test/components/views/beacon/RoomLiveShareWarning-test.tsx b/test/components/views/beacon/RoomLiveShareWarning-test.tsx index def8c288507..57095bba4ed 100644 --- a/test/components/views/beacon/RoomLiveShareWarning-test.tsx +++ b/test/components/views/beacon/RoomLiveShareWarning-test.tsx @@ -15,65 +15,195 @@ limitations under the License. */ import React from 'react'; -import { mocked } from 'jest-mock'; +import { act } from 'react-dom/test-utils'; import { mount } from 'enzyme'; +import { Room, Beacon, BeaconEvent } from 'matrix-js-sdk/src/matrix'; import '../../../skinned-sdk'; import RoomLiveShareWarning from '../../../../src/components/views/beacon/RoomLiveShareWarning'; -import { OwnBeaconStore, OwnBeaconStoreEvent } from '../../../../src/stores/OwnBeaconStore'; -import { flushPromises } from '../../../test-utils'; - -jest.mock('../../../../src/stores/OwnBeaconStore', () => { - // eslint-disable-next-line @typescript-eslint/no-var-requires - const EventEmitter = require("events"); - class MockOwnBeaconStore extends EventEmitter { - public hasLiveBeacons = jest.fn().mockReturnValue(false); - public getLiveBeaconIds = jest.fn().mockReturnValue([]); - } - return { - // @ts-ignore - ...jest.requireActual('../../../../src/stores/OwnBeaconStore'), - OwnBeaconStore: { - instance: new MockOwnBeaconStore() as unknown as OwnBeaconStore, - }, - }; -}, -); +import { OwnBeaconStore } from '../../../../src/stores/OwnBeaconStore'; +import { + findByTestId, + getMockClientWithEventEmitter, + makeBeaconInfoEvent, + resetAsyncStoreWithClient, + setupAsyncStoreWithClient, +} from '../../../test-utils'; +jest.useFakeTimers(); describe('', () => { + const aliceId = '@alice:server.org'; + const room1Id = '$room1:server.org'; + const room2Id = '$room2:server.org'; + const room3Id = '$room3:server.org'; + const mockClient = getMockClientWithEventEmitter({ + getVisibleRooms: jest.fn().mockReturnValue([]), + getUserId: jest.fn().mockReturnValue(aliceId), + unstable_setLiveBeacon: jest.fn().mockResolvedValue({ event_id: '1' }), + }); + + // 14.03.2022 16:15 + const now = 1647270879403; + const HOUR_MS = 3600000; + // mock the date so events are stable for snapshots etc + jest.spyOn(global.Date, 'now').mockReturnValue(now); + const room1Beacon1 = makeBeaconInfoEvent(aliceId, room1Id, { isLive: true, timeout: HOUR_MS }); + const room2Beacon1 = makeBeaconInfoEvent(aliceId, room2Id, { isLive: true, timeout: HOUR_MS }); + const room2Beacon2 = makeBeaconInfoEvent(aliceId, room2Id, { isLive: true, timeout: HOUR_MS * 12 }); + const room3Beacon1 = makeBeaconInfoEvent(aliceId, room3Id, { isLive: true, timeout: HOUR_MS }); + + // make fresh rooms every time + // as we update room state + const makeRoomsWithStateEvents = (stateEvents = []): [Room, Room] => { + const room1 = new Room(room1Id, mockClient, aliceId); + const room2 = new Room(room2Id, mockClient, aliceId); + + room1.currentState.setStateEvents(stateEvents); + room2.currentState.setStateEvents(stateEvents); + mockClient.getVisibleRooms.mockReturnValue([room1, room2]); + + return [room1, room2]; + }; + + const advanceDateAndTime = (ms: number) => { + // bc liveness check uses Date.now we have to advance this mock + jest.spyOn(global.Date, 'now').mockReturnValue(now + ms); + // then advance time for the interval by the same amount + jest.advanceTimersByTime(ms); + }; + + const makeOwnBeaconStore = async () => { + const store = OwnBeaconStore.instance; + + await setupAsyncStoreWithClient(store, mockClient); + return store; + }; + const defaultProps = { - roomId: '!room:server.org', + roomId: room1Id, + }; + const getComponent = (props = {}) => { + let component; + // component updates on render + // wrap in act + act(() => { + component = mount(); + }); + return component; }; - const getComponent = (props = {}) => - mount(); - it('renders nothing when user has no live beacons', () => { + beforeEach(() => { + jest.spyOn(global.Date, 'now').mockReturnValue(now); + mockClient.unstable_setLiveBeacon.mockClear(); + }); + + afterEach(async () => { + await resetAsyncStoreWithClient(OwnBeaconStore.instance); + }); + + afterAll(() => { + jest.spyOn(global.Date, 'now').mockRestore(); + }); + + it('renders nothing when user has no live beacons at all', async () => { + await makeOwnBeaconStore(); const component = getComponent(); expect(component.html()).toBe(null); }); - xdescribe('when user has live beacons', () => { - beforeEach(() => { - mocked(OwnBeaconStore.instance).hasLiveBeacons.mockReturnValue(true); - mocked(OwnBeaconStore.instance).getLiveBeaconIds.mockReturnValue([]); + it('renders nothing when user has no live beacons in room', async () => { + await act(async () => { + await makeRoomsWithStateEvents([room2Beacon1]); + await makeOwnBeaconStore(); }); - it('renders correctly when not minimized', () => { - const component = getComponent(); + const component = getComponent({ roomId: room1Id }); + expect(component.html()).toBe(null); + }); + + describe('when user has live beacons', () => { + beforeEach(async () => { + await act(async () => { + await makeRoomsWithStateEvents([room1Beacon1, room2Beacon1, room2Beacon2]); + await makeOwnBeaconStore(); + }); + }); + + it('renders correctly with one live beacon in room', () => { + const component = getComponent({ roomId: room1Id }); expect(component).toMatchSnapshot(); }); + it('renders correctly with two live beacons in room', () => { + const component = getComponent({ roomId: room2Id }); + expect(component).toMatchSnapshot(); + // later expiry displayed + expect(findByTestId(component, 'room-live-share-expiry').text()).toEqual('12h left'); + }); + it('removes itself when user stops having live beacons', async () => { - const component = getComponent(); + const component = getComponent({ roomId: room1Id }); // started out rendered expect(component.html()).toBeTruthy(); - mocked(OwnBeaconStore.instance).hasLiveBeacons.mockReturnValue(false); - OwnBeaconStore.instance.emit(OwnBeaconStoreEvent.LivenessChange, false); - - await flushPromises(); - component.setProps({}); + // time travel until room1Beacon1 is expired + advanceDateAndTime(HOUR_MS + 1); + act(() => { + mockClient.emit(BeaconEvent.LivenessChange, false, new Beacon(room1Beacon1)); + component.setProps({}); + }); expect(component.html()).toBe(null); }); + + it('renders when user adds a live beacon', async () => { + const component = getComponent({ roomId: room3Id }); + // started out not rendered + expect(component.html()).toBeFalsy(); + + act(() => { + mockClient.emit(BeaconEvent.New, room3Beacon1, new Beacon(room3Beacon1)); + component.setProps({}); + }); + + expect(component.html()).toBeTruthy(); + }); + + describe('stopping beacons', () => { + it('stops beacon on stop sharing click', () => { + const component = getComponent({ roomId: room2Id }); + + act(() => { + findByTestId(component, 'room-live-share-stop-sharing').at(0).simulate('click'); + component.setProps({}); + }); + + expect(mockClient.unstable_setLiveBeacon).toHaveBeenCalledTimes(2); + expect(component.find('Spinner').length).toBeTruthy(); + expect(findByTestId(component, 'room-live-share-stop-sharing').at(0).props().disabled).toBeTruthy(); + }); + + it('displays again with correct state after stopping a beacon', () => { + // make sure the loading state is reset correctly after removing a beacon + const component = getComponent({ roomId: room2Id }); + + act(() => { + findByTestId(component, 'room-live-share-stop-sharing').at(0).simulate('click'); + }); + // time travel until room1Beacon1 is expired + advanceDateAndTime(HOUR_MS + 1); + act(() => { + mockClient.emit(BeaconEvent.LivenessChange, false, new Beacon(room1Beacon1)); + }); + + const newLiveBeacon = makeBeaconInfoEvent(aliceId, room2Id, { isLive: true }); + act(() => { + mockClient.emit(BeaconEvent.New, newLiveBeacon, new Beacon(newLiveBeacon)); + }); + + // button not disabled and expiry time shown + expect(findByTestId(component, 'room-live-share-stop-sharing').at(0).props().disabled).toBeFalsy(); + expect(findByTestId(component, 'room-live-share-expiry').text()).toEqual('11h left'); + }); + }); }); }); diff --git a/test/components/views/beacon/__snapshots__/RoomLiveShareWarning-test.tsx.snap b/test/components/views/beacon/__snapshots__/RoomLiveShareWarning-test.tsx.snap index ab5008a8ee5..3079670fc55 100644 --- a/test/components/views/beacon/__snapshots__/RoomLiveShareWarning-test.tsx.snap +++ b/test/components/views/beacon/__snapshots__/RoomLiveShareWarning-test.tsx.snap @@ -1,12 +1,101 @@ // Jest Snapshot v1, https://goo.gl/fbAQLP -exports[` when user has live beacons renders correctly when not minimized 1`] = ` - +exports[` when user has live beacons renders correctly with one live beacon in room 1`] = ` +
-
- You are sharing your live location + +
+ + + You are sharing your live location + + + 1h left + + + + +
+ +`; + +exports[` when user has live beacons renders correctly with two live beacons in room 1`] = ` + +
+ +
+ + + You are sharing 2 live locations + + + 12h left + + + +
`; diff --git a/test/stores/OwnBeaconStore-test.ts b/test/stores/OwnBeaconStore-test.ts index a9b1f839a1f..d562c1ca115 100644 --- a/test/stores/OwnBeaconStore-test.ts +++ b/test/stores/OwnBeaconStore-test.ts @@ -437,7 +437,7 @@ xdescribe('OwnBeaconStore', () => { expect(store.hasLiveBeacons(room1Id)).toBe(true); expect(emitSpy).toHaveBeenCalledWith( OwnBeaconStoreEvent.LivenessChange, - [alicesOldRoomIdBeaconInfo.getType()] + [alicesOldRoomIdBeaconInfo.getType()], ); }); }); From c2de95a1826df7022947f8d812152925e2d0a8d1 Mon Sep 17 00:00:00 2001 From: Kerry Archibald Date: Tue, 22 Mar 2022 11:53:05 +0100 Subject: [PATCH 09/12] un xdescribe beaconstore tests Signed-off-by: Kerry Archibald --- test/stores/OwnBeaconStore-test.ts | 3 +-- 1 file changed, 1 insertion(+), 2 deletions(-) diff --git a/test/stores/OwnBeaconStore-test.ts b/test/stores/OwnBeaconStore-test.ts index d562c1ca115..14b87d822d1 100644 --- a/test/stores/OwnBeaconStore-test.ts +++ b/test/stores/OwnBeaconStore-test.ts @@ -24,8 +24,7 @@ import { getMockClientWithEventEmitter } from "../test-utils/client"; jest.useFakeTimers(); -// xdescribing while mismatch with matrix-js-sdk -xdescribe('OwnBeaconStore', () => { +describe('OwnBeaconStore', () => { // 14.03.2022 16:15 const now = 1647270879403; const HOUR_MS = 3600000; From 508c07efc2b78ce662bd3d1b440867a1bac611b4 Mon Sep 17 00:00:00 2001 From: Kerry Archibald Date: Tue, 22 Mar 2022 11:56:25 +0100 Subject: [PATCH 10/12] missed copyrights Signed-off-by: Kerry Archibald --- .../views/beacon/_RoomLiveShareWarning.scss | 16 ++++++++++++++++ .../views/beacon/StyledLiveBeaconIcon-test.tsx | 15 +++++++++++++++ 2 files changed, 31 insertions(+) diff --git a/res/css/components/views/beacon/_RoomLiveShareWarning.scss b/res/css/components/views/beacon/_RoomLiveShareWarning.scss index b18a00ffd9c..c0d5ea47fe1 100644 --- a/res/css/components/views/beacon/_RoomLiveShareWarning.scss +++ b/res/css/components/views/beacon/_RoomLiveShareWarning.scss @@ -1,3 +1,19 @@ +/* +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. +*/ + .mx_RoomLiveShareWarning { width: 100%; diff --git a/test/components/views/beacon/StyledLiveBeaconIcon-test.tsx b/test/components/views/beacon/StyledLiveBeaconIcon-test.tsx index 81cbdbd28b4..38fc8391d58 100644 --- a/test/components/views/beacon/StyledLiveBeaconIcon-test.tsx +++ b/test/components/views/beacon/StyledLiveBeaconIcon-test.tsx @@ -1,3 +1,18 @@ +/* +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'; From fe689b12df76dbfa82129eabbe5cc02ffba18f3f Mon Sep 17 00:00:00 2001 From: Kerry Archibald Date: Tue, 22 Mar 2022 11:59:36 +0100 Subject: [PATCH 11/12] i18n Signed-off-by: Kerry Archibald --- src/i18n/strings/en_EN.json | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/src/i18n/strings/en_EN.json b/src/i18n/strings/en_EN.json index 9d777293d08..ddda953227f 100644 --- a/src/i18n/strings/en_EN.json +++ b/src/i18n/strings/en_EN.json @@ -2966,8 +2966,8 @@ "Join the beta": "Join the beta", "You are sharing your live location": "You are sharing your live location", "%(timeRemaining)s left": "%(timeRemaining)s left", - "You are sharing %(count)s live locations|one": "You are sharing your live location", "You are sharing %(count)s live locations|other": "You are sharing %(count)s live locations", + "You are sharing %(count)s live locations|one": "You are sharing your live location", "Stop sharing": "Stop sharing", "Avatar": "Avatar", "This room is public": "This room is public", From bbe6e320e160004a823d06c5f067ef485268b17b Mon Sep 17 00:00:00 2001 From: Kerry Archibald Date: Tue, 22 Mar 2022 14:50:05 +0100 Subject: [PATCH 12/12] tidy Signed-off-by: Kerry Archibald --- src/components/views/beacon/RoomLiveShareWarning.tsx | 3 ++- src/stores/OwnBeaconStore.ts | 4 ++-- 2 files changed, 4 insertions(+), 3 deletions(-) diff --git a/src/components/views/beacon/RoomLiveShareWarning.tsx b/src/components/views/beacon/RoomLiveShareWarning.tsx index 5958263af8f..5eb85e71b45 100644 --- a/src/components/views/beacon/RoomLiveShareWarning.tsx +++ b/src/components/views/beacon/RoomLiveShareWarning.tsx @@ -42,6 +42,7 @@ type LiveBeaconsState = { onStopSharing?: () => void; stoppingInProgress?: boolean; }; + const useLiveBeacons = (roomId: Room['roomId']): LiveBeaconsState => { const [stoppingInProgress, setStoppingInProgress] = useState(false); const liveBeaconIds = useEventEmitterState( @@ -53,7 +54,7 @@ const useLiveBeacons = (roomId: Room['roomId']): LiveBeaconsState => { // reset stopping in progress on change in live ids useEffect(() => { setStoppingInProgress(false); - }, [liveBeaconIds, setStoppingInProgress]); + }, [liveBeaconIds]); if (!liveBeaconIds?.length) { return { liveBeaconIds }; diff --git a/src/stores/OwnBeaconStore.ts b/src/stores/OwnBeaconStore.ts index 5fcbfa9a903..4ac2bcaddea 100644 --- a/src/stores/OwnBeaconStore.ts +++ b/src/stores/OwnBeaconStore.ts @@ -164,12 +164,12 @@ export class OwnBeaconStore extends AsyncStoreWithClient { }; private checkLiveness = (): void => { - const prevLiveness = this.getLiveBeaconIds(); + const prevLiveBeaconIds = this.getLiveBeaconIds(); this.liveBeaconIds = [...this.beacons.values()] .filter(beacon => beacon.isLive) .map(beacon => beacon.identifier); - if (arrayHasDiff(prevLiveness, this.liveBeaconIds)) { + if (arrayHasDiff(prevLiveBeaconIds, this.liveBeaconIds)) { this.emit(OwnBeaconStoreEvent.LivenessChange, this.liveBeaconIds); } };