Add voice broadcast recording stores (#9319)
* Add voice broadcast recording stores * Refactor startNewVoiceBroadcastRecording
This commit is contained in:
parent
c14191bfb6
commit
d775e403c4
14 changed files with 769 additions and 119 deletions
|
@ -55,9 +55,8 @@ import { isLocalRoom } from '../../../utils/localRoom/isLocalRoom';
|
||||||
import { Features } from '../../../settings/Settings';
|
import { Features } from '../../../settings/Settings';
|
||||||
import { VoiceMessageRecording } from '../../../audio/VoiceMessageRecording';
|
import { VoiceMessageRecording } from '../../../audio/VoiceMessageRecording';
|
||||||
import {
|
import {
|
||||||
VoiceBroadcastInfoEventContent,
|
startNewVoiceBroadcastRecording,
|
||||||
VoiceBroadcastInfoEventType,
|
VoiceBroadcastRecordingsStore,
|
||||||
VoiceBroadcastInfoState,
|
|
||||||
} from '../../../voice-broadcast';
|
} from '../../../voice-broadcast';
|
||||||
|
|
||||||
let instanceCount = 0;
|
let instanceCount = 0;
|
||||||
|
@ -508,16 +507,11 @@ export default class MessageComposer extends React.Component<IProps, IState> {
|
||||||
showStickersButton={this.showStickersButton}
|
showStickersButton={this.showStickersButton}
|
||||||
toggleButtonMenu={this.toggleButtonMenu}
|
toggleButtonMenu={this.toggleButtonMenu}
|
||||||
showVoiceBroadcastButton={this.showVoiceBroadcastButton}
|
showVoiceBroadcastButton={this.showVoiceBroadcastButton}
|
||||||
onStartVoiceBroadcastClick={async () => {
|
onStartVoiceBroadcastClick={() => {
|
||||||
const client = MatrixClientPeg.get();
|
startNewVoiceBroadcastRecording(
|
||||||
client.sendStateEvent(
|
|
||||||
this.props.room.roomId,
|
this.props.room.roomId,
|
||||||
VoiceBroadcastInfoEventType,
|
MatrixClientPeg.get(),
|
||||||
{
|
VoiceBroadcastRecordingsStore.instance(),
|
||||||
state: VoiceBroadcastInfoState.Started,
|
|
||||||
chunk_length: 300,
|
|
||||||
} as VoiceBroadcastInfoEventContent,
|
|
||||||
client.getUserId(),
|
|
||||||
);
|
);
|
||||||
this.toggleButtonMenu();
|
this.toggleButtonMenu();
|
||||||
}}
|
}}
|
||||||
|
|
|
@ -14,55 +14,43 @@ See the License for the specific language governing permissions and
|
||||||
limitations under the License.
|
limitations under the License.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
import React from "react";
|
import React, { useState } from "react";
|
||||||
import { MatrixEvent, RelationType } from "matrix-js-sdk/src/matrix";
|
|
||||||
|
|
||||||
import { VoiceBroadcastInfoEventType, VoiceBroadcastInfoState, VoiceBroadcastRecordingBody } from "..";
|
import {
|
||||||
|
VoiceBroadcastInfoState,
|
||||||
|
VoiceBroadcastRecordingBody,
|
||||||
|
VoiceBroadcastRecordingsStore,
|
||||||
|
VoiceBroadcastRecording,
|
||||||
|
VoiceBroadcastRecordingEvent,
|
||||||
|
} from "..";
|
||||||
import { IBodyProps } from "../../components/views/messages/IBodyProps";
|
import { IBodyProps } from "../../components/views/messages/IBodyProps";
|
||||||
import { MatrixClientPeg } from "../../MatrixClientPeg";
|
import { MatrixClientPeg } from "../../MatrixClientPeg";
|
||||||
|
import { useTypedEventEmitter } from "../../hooks/useEventEmitter";
|
||||||
|
|
||||||
/**
|
export const VoiceBroadcastBody: React.FC<IBodyProps> = ({ mxEvent }) => {
|
||||||
* Temporary component to display voice broadcasts.
|
|
||||||
* XXX: To be refactored to some fancy store/hook/controller architecture.
|
|
||||||
*/
|
|
||||||
export const VoiceBroadcastBody: React.FC<IBodyProps> = ({
|
|
||||||
getRelationsForEvent,
|
|
||||||
mxEvent,
|
|
||||||
}) => {
|
|
||||||
const client = MatrixClientPeg.get();
|
const client = MatrixClientPeg.get();
|
||||||
const relations = getRelationsForEvent?.(
|
const room = client.getRoom(mxEvent.getRoomId());
|
||||||
mxEvent.getId(),
|
const recording = VoiceBroadcastRecordingsStore.instance().getByInfoEvent(mxEvent, client);
|
||||||
RelationType.Reference,
|
const [recordingState, setRecordingState] = useState(recording.state);
|
||||||
VoiceBroadcastInfoEventType,
|
|
||||||
|
useTypedEventEmitter(
|
||||||
|
recording,
|
||||||
|
VoiceBroadcastRecordingEvent.StateChanged,
|
||||||
|
(state: VoiceBroadcastInfoState, _recording: VoiceBroadcastRecording) => {
|
||||||
|
setRecordingState(state);
|
||||||
|
},
|
||||||
);
|
);
|
||||||
const relatedEvents = relations?.getRelations();
|
|
||||||
const live = !relatedEvents?.find((event: MatrixEvent) => {
|
|
||||||
return event.getContent()?.state === VoiceBroadcastInfoState.Stopped;
|
|
||||||
});
|
|
||||||
|
|
||||||
const stopVoiceBroadcast = () => {
|
const stopVoiceBroadcast = () => {
|
||||||
if (!live) return;
|
if (recordingState !== VoiceBroadcastInfoState.Started) return;
|
||||||
|
recording.stop();
|
||||||
client.sendStateEvent(
|
|
||||||
mxEvent.getRoomId(),
|
|
||||||
VoiceBroadcastInfoEventType,
|
|
||||||
{
|
|
||||||
state: VoiceBroadcastInfoState.Stopped,
|
|
||||||
["m.relates_to"]: {
|
|
||||||
rel_type: RelationType.Reference,
|
|
||||||
event_id: mxEvent.getId(),
|
|
||||||
},
|
|
||||||
},
|
|
||||||
client.getUserId(),
|
|
||||||
);
|
|
||||||
};
|
};
|
||||||
|
|
||||||
const room = client.getRoom(mxEvent.getRoomId());
|
|
||||||
const senderId = mxEvent.getSender();
|
const senderId = mxEvent.getSender();
|
||||||
const sender = mxEvent.sender;
|
const sender = mxEvent.sender;
|
||||||
return <VoiceBroadcastRecordingBody
|
return <VoiceBroadcastRecordingBody
|
||||||
onClick={stopVoiceBroadcast}
|
onClick={stopVoiceBroadcast}
|
||||||
live={live}
|
live={recordingState === VoiceBroadcastInfoState.Started}
|
||||||
member={sender}
|
member={sender}
|
||||||
userId={senderId}
|
userId={senderId}
|
||||||
title={`${sender?.name ?? senderId} • ${room.name}`}
|
title={`${sender?.name ?? senderId} • ${room.name}`}
|
||||||
|
|
|
@ -22,7 +22,9 @@ limitations under the License.
|
||||||
import { RelationType } from "matrix-js-sdk/src/matrix";
|
import { RelationType } from "matrix-js-sdk/src/matrix";
|
||||||
|
|
||||||
export * from "./components";
|
export * from "./components";
|
||||||
|
export * from "./models";
|
||||||
export * from "./utils";
|
export * from "./utils";
|
||||||
|
export * from "./stores";
|
||||||
|
|
||||||
export const VoiceBroadcastInfoEventType = "io.element.voice_broadcast_info";
|
export const VoiceBroadcastInfoEventType = "io.element.voice_broadcast_info";
|
||||||
|
|
||||||
|
@ -35,7 +37,7 @@ export enum VoiceBroadcastInfoState {
|
||||||
|
|
||||||
export interface VoiceBroadcastInfoEventContent {
|
export interface VoiceBroadcastInfoEventContent {
|
||||||
state: VoiceBroadcastInfoState;
|
state: VoiceBroadcastInfoState;
|
||||||
chunk_length: number;
|
chunk_length?: number;
|
||||||
["m.relates_to"]?: {
|
["m.relates_to"]?: {
|
||||||
rel_type: RelationType;
|
rel_type: RelationType;
|
||||||
event_id: string;
|
event_id: string;
|
||||||
|
|
78
src/voice-broadcast/models/VoiceBroadcastRecording.ts
Normal file
78
src/voice-broadcast/models/VoiceBroadcastRecording.ts
Normal file
|
@ -0,0 +1,78 @@
|
||||||
|
/*
|
||||||
|
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 { MatrixClient, MatrixEvent, RelationType } from "matrix-js-sdk/src/matrix";
|
||||||
|
import { TypedEventEmitter } from "matrix-js-sdk/src/models/typed-event-emitter";
|
||||||
|
|
||||||
|
import { VoiceBroadcastInfoEventType, VoiceBroadcastInfoState } from "..";
|
||||||
|
|
||||||
|
export enum VoiceBroadcastRecordingEvent {
|
||||||
|
StateChanged = "liveness_changed",
|
||||||
|
}
|
||||||
|
|
||||||
|
interface EventMap {
|
||||||
|
[VoiceBroadcastRecordingEvent.StateChanged]: (state: VoiceBroadcastInfoState) => void;
|
||||||
|
}
|
||||||
|
|
||||||
|
export class VoiceBroadcastRecording extends TypedEventEmitter<VoiceBroadcastRecordingEvent, EventMap> {
|
||||||
|
private _state: VoiceBroadcastInfoState;
|
||||||
|
|
||||||
|
public constructor(
|
||||||
|
public readonly infoEvent: MatrixEvent,
|
||||||
|
private client: MatrixClient,
|
||||||
|
) {
|
||||||
|
super();
|
||||||
|
|
||||||
|
const room = this.client.getRoom(this.infoEvent.getRoomId());
|
||||||
|
const relations = room?.getUnfilteredTimelineSet()?.relations?.getChildEventsForEvent(
|
||||||
|
this.infoEvent.getId(),
|
||||||
|
RelationType.Reference,
|
||||||
|
VoiceBroadcastInfoEventType,
|
||||||
|
);
|
||||||
|
const relatedEvents = relations?.getRelations();
|
||||||
|
this._state = !relatedEvents?.find((event: MatrixEvent) => {
|
||||||
|
return event.getContent()?.state === VoiceBroadcastInfoState.Stopped;
|
||||||
|
}) ? VoiceBroadcastInfoState.Started : VoiceBroadcastInfoState.Stopped;
|
||||||
|
|
||||||
|
// TODO Michael W: add listening for updates
|
||||||
|
}
|
||||||
|
|
||||||
|
private setState(state: VoiceBroadcastInfoState): void {
|
||||||
|
this._state = state;
|
||||||
|
this.emit(VoiceBroadcastRecordingEvent.StateChanged, this.state);
|
||||||
|
}
|
||||||
|
|
||||||
|
public async stop() {
|
||||||
|
this.setState(VoiceBroadcastInfoState.Stopped);
|
||||||
|
// TODO Michael W: add error handling
|
||||||
|
await this.client.sendStateEvent(
|
||||||
|
this.infoEvent.getRoomId(),
|
||||||
|
VoiceBroadcastInfoEventType,
|
||||||
|
{
|
||||||
|
state: VoiceBroadcastInfoState.Stopped,
|
||||||
|
["m.relates_to"]: {
|
||||||
|
rel_type: RelationType.Reference,
|
||||||
|
event_id: this.infoEvent.getId(),
|
||||||
|
},
|
||||||
|
},
|
||||||
|
this.client.getUserId(),
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
public get state(): VoiceBroadcastInfoState {
|
||||||
|
return this._state;
|
||||||
|
}
|
||||||
|
}
|
17
src/voice-broadcast/models/index.ts
Normal file
17
src/voice-broadcast/models/index.ts
Normal file
|
@ -0,0 +1,17 @@
|
||||||
|
/*
|
||||||
|
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.
|
||||||
|
*/
|
||||||
|
|
||||||
|
export * from "./VoiceBroadcastRecording";
|
71
src/voice-broadcast/stores/VoiceBroadcastRecordingsStore.ts
Normal file
71
src/voice-broadcast/stores/VoiceBroadcastRecordingsStore.ts
Normal file
|
@ -0,0 +1,71 @@
|
||||||
|
/*
|
||||||
|
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 { MatrixClient, MatrixEvent } from "matrix-js-sdk/src/matrix";
|
||||||
|
import { TypedEventEmitter } from "matrix-js-sdk/src/models/typed-event-emitter";
|
||||||
|
|
||||||
|
import { VoiceBroadcastRecording } from "..";
|
||||||
|
|
||||||
|
export enum VoiceBroadcastRecordingsStoreEvent {
|
||||||
|
CurrentChanged = "current_changed",
|
||||||
|
}
|
||||||
|
|
||||||
|
interface EventMap {
|
||||||
|
[VoiceBroadcastRecordingsStoreEvent.CurrentChanged]: (recording: VoiceBroadcastRecording) => void;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* This store provides access to the current and specific Voice Broadcast recordings.
|
||||||
|
*/
|
||||||
|
export class VoiceBroadcastRecordingsStore extends TypedEventEmitter<VoiceBroadcastRecordingsStoreEvent, EventMap> {
|
||||||
|
private _current: VoiceBroadcastRecording | null;
|
||||||
|
private recordings = new Map<string, VoiceBroadcastRecording>();
|
||||||
|
|
||||||
|
public constructor() {
|
||||||
|
super();
|
||||||
|
}
|
||||||
|
|
||||||
|
public setCurrent(current: VoiceBroadcastRecording): void {
|
||||||
|
if (this._current === current) return;
|
||||||
|
|
||||||
|
this._current = current;
|
||||||
|
this.recordings.set(current.infoEvent.getId(), current);
|
||||||
|
this.emit(VoiceBroadcastRecordingsStoreEvent.CurrentChanged, current);
|
||||||
|
}
|
||||||
|
|
||||||
|
public get current(): VoiceBroadcastRecording {
|
||||||
|
return this._current;
|
||||||
|
}
|
||||||
|
|
||||||
|
public getByInfoEvent(infoEvent: MatrixEvent, client: MatrixClient): VoiceBroadcastRecording {
|
||||||
|
const infoEventId = infoEvent.getId();
|
||||||
|
|
||||||
|
if (!this.recordings.has(infoEventId)) {
|
||||||
|
this.recordings.set(infoEventId, new VoiceBroadcastRecording(infoEvent, client));
|
||||||
|
}
|
||||||
|
|
||||||
|
return this.recordings.get(infoEventId);
|
||||||
|
}
|
||||||
|
|
||||||
|
public static readonly _instance = new VoiceBroadcastRecordingsStore();
|
||||||
|
|
||||||
|
/**
|
||||||
|
* TODO Michael W: replace when https://github.com/matrix-org/matrix-react-sdk/pull/9293 has been merged
|
||||||
|
*/
|
||||||
|
public static instance() {
|
||||||
|
return VoiceBroadcastRecordingsStore._instance;
|
||||||
|
}
|
||||||
|
}
|
17
src/voice-broadcast/stores/index.ts
Normal file
17
src/voice-broadcast/stores/index.ts
Normal file
|
@ -0,0 +1,17 @@
|
||||||
|
/*
|
||||||
|
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.
|
||||||
|
*/
|
||||||
|
|
||||||
|
export * from "./VoiceBroadcastRecordingsStore";
|
|
@ -15,3 +15,4 @@ limitations under the License.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
export * from "./shouldDisplayAsVoiceBroadcastTile";
|
export * from "./shouldDisplayAsVoiceBroadcastTile";
|
||||||
|
export * from "./startNewVoiceBroadcastRecording";
|
||||||
|
|
74
src/voice-broadcast/utils/startNewVoiceBroadcastRecording.ts
Normal file
74
src/voice-broadcast/utils/startNewVoiceBroadcastRecording.ts
Normal file
|
@ -0,0 +1,74 @@
|
||||||
|
/*
|
||||||
|
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 { ISendEventResponse, MatrixClient, RoomStateEvent } from "matrix-js-sdk/src/matrix";
|
||||||
|
import { defer } from "matrix-js-sdk/src/utils";
|
||||||
|
|
||||||
|
import {
|
||||||
|
VoiceBroadcastInfoEventContent,
|
||||||
|
VoiceBroadcastInfoEventType,
|
||||||
|
VoiceBroadcastInfoState,
|
||||||
|
VoiceBroadcastRecordingsStore,
|
||||||
|
VoiceBroadcastRecording,
|
||||||
|
} from "..";
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Starts a new Voice Broadcast Recording.
|
||||||
|
* Sends a voice_broadcast_info state event and waits for the event to actually appear in the room state.
|
||||||
|
*/
|
||||||
|
export const startNewVoiceBroadcastRecording = async (
|
||||||
|
roomId: string,
|
||||||
|
client: MatrixClient,
|
||||||
|
recordingsStore: VoiceBroadcastRecordingsStore,
|
||||||
|
): Promise<VoiceBroadcastRecording> => {
|
||||||
|
const room = client.getRoom(roomId);
|
||||||
|
const { promise, resolve } = defer<VoiceBroadcastRecording>();
|
||||||
|
let result: ISendEventResponse = null;
|
||||||
|
|
||||||
|
const onRoomStateEvents = () => {
|
||||||
|
if (!result) return;
|
||||||
|
|
||||||
|
const voiceBroadcastEvent = room.currentState.getStateEvents(
|
||||||
|
VoiceBroadcastInfoEventType,
|
||||||
|
client.getUserId(),
|
||||||
|
);
|
||||||
|
|
||||||
|
if (voiceBroadcastEvent?.getId() === result.event_id) {
|
||||||
|
room.off(RoomStateEvent.Events, onRoomStateEvents);
|
||||||
|
const recording = new VoiceBroadcastRecording(
|
||||||
|
voiceBroadcastEvent,
|
||||||
|
client,
|
||||||
|
);
|
||||||
|
recordingsStore.setCurrent(recording);
|
||||||
|
resolve(recording);
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
room.on(RoomStateEvent.Events, onRoomStateEvents);
|
||||||
|
|
||||||
|
// XXX Michael W: refactor to live event
|
||||||
|
result = await client.sendStateEvent(
|
||||||
|
roomId,
|
||||||
|
VoiceBroadcastInfoEventType,
|
||||||
|
{
|
||||||
|
state: VoiceBroadcastInfoState.Started,
|
||||||
|
chunk_length: 300,
|
||||||
|
} as VoiceBroadcastInfoEventContent,
|
||||||
|
client.getUserId(),
|
||||||
|
);
|
||||||
|
|
||||||
|
return promise;
|
||||||
|
};
|
|
@ -401,7 +401,7 @@ export function mkStubRoom(roomId: string = null, name: string, client: MatrixCl
|
||||||
getMembers: jest.fn().mockReturnValue([]),
|
getMembers: jest.fn().mockReturnValue([]),
|
||||||
getPendingEvents: () => [],
|
getPendingEvents: () => [],
|
||||||
getLiveTimeline: jest.fn().mockReturnValue(stubTimeline),
|
getLiveTimeline: jest.fn().mockReturnValue(stubTimeline),
|
||||||
getUnfilteredTimelineSet: () => null,
|
getUnfilteredTimelineSet: jest.fn(),
|
||||||
findEventById: () => null,
|
findEventById: () => null,
|
||||||
getAccountData: () => null,
|
getAccountData: () => null,
|
||||||
hasMembershipState: () => null,
|
hasMembershipState: () => null,
|
||||||
|
|
|
@ -15,10 +15,9 @@ limitations under the License.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
import React from "react";
|
import React from "react";
|
||||||
import { render } from "@testing-library/react";
|
import { render, screen } from "@testing-library/react";
|
||||||
import userEvent from "@testing-library/user-event";
|
import userEvent from "@testing-library/user-event";
|
||||||
import { MatrixClient, MatrixEvent, RelationType } from "matrix-js-sdk/src/matrix";
|
import { MatrixClient, MatrixEvent, Room } from "matrix-js-sdk/src/matrix";
|
||||||
import { Relations } from "matrix-js-sdk/src/models/relations";
|
|
||||||
import { mocked } from "jest-mock";
|
import { mocked } from "jest-mock";
|
||||||
|
|
||||||
import {
|
import {
|
||||||
|
@ -26,8 +25,11 @@ import {
|
||||||
VoiceBroadcastInfoEventType,
|
VoiceBroadcastInfoEventType,
|
||||||
VoiceBroadcastInfoState,
|
VoiceBroadcastInfoState,
|
||||||
VoiceBroadcastRecordingBody,
|
VoiceBroadcastRecordingBody,
|
||||||
|
VoiceBroadcastRecordingsStore,
|
||||||
|
VoiceBroadcastRecording,
|
||||||
|
VoiceBroadcastRecordingEvent,
|
||||||
} from "../../../src/voice-broadcast";
|
} from "../../../src/voice-broadcast";
|
||||||
import { mkEvent, stubClient } from "../../test-utils";
|
import { mkEvent, mkStubRoom, stubClient } from "../../test-utils";
|
||||||
import { IBodyProps } from "../../../src/components/views/messages/IBodyProps";
|
import { IBodyProps } from "../../../src/components/views/messages/IBodyProps";
|
||||||
|
|
||||||
jest.mock("../../../src/voice-broadcast/components/molecules/VoiceBroadcastRecordingBody", () => ({
|
jest.mock("../../../src/voice-broadcast/components/molecules/VoiceBroadcastRecordingBody", () => ({
|
||||||
|
@ -38,10 +40,10 @@ describe("VoiceBroadcastBody", () => {
|
||||||
const roomId = "!room:example.com";
|
const roomId = "!room:example.com";
|
||||||
const recordingTestid = "voice-recording";
|
const recordingTestid = "voice-recording";
|
||||||
let client: MatrixClient;
|
let client: MatrixClient;
|
||||||
let getRelationsForEvent: (eventId: string, relationType: string, eventType: string) => Relations;
|
let room: Room;
|
||||||
let event: MatrixEvent;
|
let infoEvent: MatrixEvent;
|
||||||
let relatedEvent: MatrixEvent;
|
let recording: VoiceBroadcastRecording;
|
||||||
let recordingElement: HTMLElement;
|
let onRecordingStateChanged: (state: VoiceBroadcastInfoState) => void;
|
||||||
|
|
||||||
const mkVoiceBroadcastInfoEvent = (state: VoiceBroadcastInfoState) => {
|
const mkVoiceBroadcastInfoEvent = (state: VoiceBroadcastInfoState) => {
|
||||||
return mkEvent({
|
return mkEvent({
|
||||||
|
@ -55,13 +57,13 @@ describe("VoiceBroadcastBody", () => {
|
||||||
});
|
});
|
||||||
};
|
};
|
||||||
|
|
||||||
const renderVoiceBroadcast = async () => {
|
const renderVoiceBroadcast = () => {
|
||||||
const props: IBodyProps = {
|
const props: IBodyProps = {
|
||||||
getRelationsForEvent,
|
mxEvent: infoEvent,
|
||||||
mxEvent: event,
|
|
||||||
} as unknown as IBodyProps;
|
} as unknown as IBodyProps;
|
||||||
const result = render(<VoiceBroadcastBody {...props} />);
|
render(<VoiceBroadcastBody {...props} />);
|
||||||
recordingElement = await result.findByTestId(recordingTestid);
|
recording = VoiceBroadcastRecordingsStore.instance().getByInfoEvent(infoEvent, client);
|
||||||
|
recording.on(VoiceBroadcastRecordingEvent.StateChanged, onRecordingStateChanged);
|
||||||
};
|
};
|
||||||
|
|
||||||
const itShouldRenderALiveVoiceBroadcast = () => {
|
const itShouldRenderALiveVoiceBroadcast = () => {
|
||||||
|
@ -70,12 +72,14 @@ describe("VoiceBroadcastBody", () => {
|
||||||
{
|
{
|
||||||
onClick: expect.any(Function),
|
onClick: expect.any(Function),
|
||||||
live: true,
|
live: true,
|
||||||
member: event.sender,
|
member: infoEvent.sender,
|
||||||
userId: client.getUserId(),
|
userId: client.getUserId(),
|
||||||
title: "@userId:matrix.org • My room",
|
title: "@userId:matrix.org • test room",
|
||||||
},
|
},
|
||||||
{},
|
{},
|
||||||
);
|
);
|
||||||
|
screen.getByTestId(recordingTestid);
|
||||||
|
screen.getByText("Live");
|
||||||
});
|
});
|
||||||
};
|
};
|
||||||
|
|
||||||
|
@ -85,12 +89,15 @@ describe("VoiceBroadcastBody", () => {
|
||||||
{
|
{
|
||||||
onClick: expect.any(Function),
|
onClick: expect.any(Function),
|
||||||
live: false,
|
live: false,
|
||||||
member: event.sender,
|
member: infoEvent.sender,
|
||||||
userId: client.getUserId(),
|
userId: client.getUserId(),
|
||||||
title: "@userId:matrix.org • My room",
|
title: "@userId:matrix.org • test room",
|
||||||
},
|
},
|
||||||
{},
|
{},
|
||||||
);
|
);
|
||||||
|
expect(screen.getByTestId(recordingTestid)).not.toBeNull();
|
||||||
|
screen.getByTestId(recordingTestid);
|
||||||
|
expect(screen.queryByText("live")).toBeNull();
|
||||||
});
|
});
|
||||||
};
|
};
|
||||||
|
|
||||||
|
@ -108,74 +115,48 @@ describe("VoiceBroadcastBody", () => {
|
||||||
data-testid={recordingTestid}
|
data-testid={recordingTestid}
|
||||||
onClick={onClick}
|
onClick={onClick}
|
||||||
>
|
>
|
||||||
{ title }
|
<div>{ title }</div>
|
||||||
{ live && "Live" }
|
<div>{ live && "Live" }</div>
|
||||||
</div>
|
</div>
|
||||||
);
|
);
|
||||||
},
|
},
|
||||||
);
|
);
|
||||||
client = stubClient();
|
client = stubClient();
|
||||||
event = mkVoiceBroadcastInfoEvent(VoiceBroadcastInfoState.Started);
|
room = mkStubRoom(roomId, "test room", client);
|
||||||
|
mocked(client.getRoom).mockImplementation((getRoomId: string) => {
|
||||||
|
if (getRoomId === roomId) {
|
||||||
|
return room;
|
||||||
|
}
|
||||||
|
});
|
||||||
|
infoEvent = mkVoiceBroadcastInfoEvent(VoiceBroadcastInfoState.Started);
|
||||||
|
onRecordingStateChanged = jest.fn();
|
||||||
});
|
});
|
||||||
|
|
||||||
describe("when getRelationsForEvent is undefined", () => {
|
afterEach(() => {
|
||||||
beforeEach(async () => {
|
if (recording && onRecordingStateChanged) {
|
||||||
await renderVoiceBroadcast();
|
recording.off(VoiceBroadcastRecordingEvent.StateChanged, onRecordingStateChanged);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
describe("when there is a Started Voice Broadcast info event", () => {
|
||||||
|
beforeEach(() => {
|
||||||
|
renderVoiceBroadcast();
|
||||||
});
|
});
|
||||||
|
|
||||||
itShouldRenderALiveVoiceBroadcast();
|
itShouldRenderALiveVoiceBroadcast();
|
||||||
|
|
||||||
describe("and the Voice Broadcast tile has been clicked", () => {
|
describe("and it is clicked", () => {
|
||||||
beforeEach(async () => {
|
beforeEach(async () => {
|
||||||
await userEvent.click(recordingElement);
|
mocked(VoiceBroadcastRecordingBody).mockClear();
|
||||||
|
mocked(onRecordingStateChanged).mockClear();
|
||||||
|
await userEvent.click(screen.getByTestId(recordingTestid));
|
||||||
});
|
});
|
||||||
|
|
||||||
it("should emit a Voice Broadcast stop state event", () => {
|
itShouldRenderANonLiveVoiceBroadcast();
|
||||||
expect(mocked(client.sendStateEvent)).toHaveBeenCalledWith(
|
|
||||||
roomId,
|
|
||||||
VoiceBroadcastInfoEventType,
|
|
||||||
{
|
|
||||||
state: VoiceBroadcastInfoState.Stopped,
|
|
||||||
["m.relates_to"]: {
|
|
||||||
rel_type: RelationType.Reference,
|
|
||||||
event_id: event.getId(),
|
|
||||||
},
|
|
||||||
},
|
|
||||||
client.getUserId(),
|
|
||||||
);
|
|
||||||
});
|
|
||||||
});
|
|
||||||
});
|
|
||||||
|
|
||||||
describe("when getRelationsForEvent returns null", () => {
|
it("should call stop on the recording", () => {
|
||||||
beforeEach(async () => {
|
expect(recording.state).toBe(VoiceBroadcastInfoState.Stopped);
|
||||||
getRelationsForEvent = jest.fn().mockReturnValue(null);
|
expect(onRecordingStateChanged).toHaveBeenCalledWith(VoiceBroadcastInfoState.Stopped);
|
||||||
await renderVoiceBroadcast();
|
|
||||||
});
|
|
||||||
|
|
||||||
itShouldRenderALiveVoiceBroadcast();
|
|
||||||
});
|
|
||||||
|
|
||||||
describe("when getRelationsForEvent returns a stopped Voice Broadcast info", () => {
|
|
||||||
beforeEach(async () => {
|
|
||||||
relatedEvent = mkVoiceBroadcastInfoEvent(VoiceBroadcastInfoState.Stopped);
|
|
||||||
getRelationsForEvent = jest.fn().mockReturnValue({
|
|
||||||
getRelations: jest.fn().mockReturnValue([
|
|
||||||
relatedEvent,
|
|
||||||
]),
|
|
||||||
});
|
|
||||||
await renderVoiceBroadcast();
|
|
||||||
});
|
|
||||||
|
|
||||||
itShouldRenderANonLiveVoiceBroadcast();
|
|
||||||
|
|
||||||
describe("and the Voice Broadcast tile has been clicked", () => {
|
|
||||||
beforeEach(async () => {
|
|
||||||
await userEvent.click(recordingElement);
|
|
||||||
});
|
|
||||||
|
|
||||||
it("should not emit a voice broadcast stop state event", () => {
|
|
||||||
expect(mocked(client.sendStateEvent)).not.toHaveBeenCalled();
|
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
158
test/voice-broadcast/models/VoiceBroadcastRecording-test.ts
Normal file
158
test/voice-broadcast/models/VoiceBroadcastRecording-test.ts
Normal file
|
@ -0,0 +1,158 @@
|
||||||
|
/*
|
||||||
|
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 { mocked } from "jest-mock";
|
||||||
|
import { EventTimelineSet, EventType, MatrixClient, MatrixEvent, RelationType, Room } from "matrix-js-sdk/src/matrix";
|
||||||
|
import { Relations } from "matrix-js-sdk/src/models/relations";
|
||||||
|
|
||||||
|
import {
|
||||||
|
VoiceBroadcastInfoEventContent,
|
||||||
|
VoiceBroadcastInfoEventType,
|
||||||
|
VoiceBroadcastInfoState,
|
||||||
|
VoiceBroadcastRecording,
|
||||||
|
VoiceBroadcastRecordingEvent,
|
||||||
|
} from "../../../src/voice-broadcast";
|
||||||
|
import { mkEvent, mkStubRoom, stubClient } from "../../test-utils";
|
||||||
|
|
||||||
|
describe("VoiceBroadcastRecording", () => {
|
||||||
|
const roomId = "!room:example.com";
|
||||||
|
let room: Room;
|
||||||
|
let client: MatrixClient;
|
||||||
|
let infoEvent: MatrixEvent;
|
||||||
|
let voiceBroadcastRecording: VoiceBroadcastRecording;
|
||||||
|
let onStateChanged: (state: VoiceBroadcastInfoState) => void;
|
||||||
|
|
||||||
|
const mkVoiceBroadcastInfoEvent = (content: VoiceBroadcastInfoEventContent) => {
|
||||||
|
return mkEvent({
|
||||||
|
event: true,
|
||||||
|
type: VoiceBroadcastInfoEventType,
|
||||||
|
user: client.getUserId(),
|
||||||
|
room: roomId,
|
||||||
|
content,
|
||||||
|
});
|
||||||
|
};
|
||||||
|
|
||||||
|
const setUpVoiceBroadcastRecording = () => {
|
||||||
|
voiceBroadcastRecording = new VoiceBroadcastRecording(infoEvent, client);
|
||||||
|
voiceBroadcastRecording.on(VoiceBroadcastRecordingEvent.StateChanged, onStateChanged);
|
||||||
|
};
|
||||||
|
|
||||||
|
beforeEach(() => {
|
||||||
|
client = stubClient();
|
||||||
|
room = mkStubRoom(roomId, "Test Room", client);
|
||||||
|
mocked(client.getRoom).mockImplementation((getRoomId: string) => {
|
||||||
|
if (getRoomId === roomId) {
|
||||||
|
return room;
|
||||||
|
}
|
||||||
|
});
|
||||||
|
onStateChanged = jest.fn();
|
||||||
|
});
|
||||||
|
|
||||||
|
afterEach(() => {
|
||||||
|
voiceBroadcastRecording.off(VoiceBroadcastRecordingEvent.StateChanged, onStateChanged);
|
||||||
|
});
|
||||||
|
|
||||||
|
describe("when created for a Voice Broadcast Info without relations", () => {
|
||||||
|
beforeEach(() => {
|
||||||
|
infoEvent = mkVoiceBroadcastInfoEvent({
|
||||||
|
state: VoiceBroadcastInfoState.Started,
|
||||||
|
});
|
||||||
|
setUpVoiceBroadcastRecording();
|
||||||
|
});
|
||||||
|
|
||||||
|
it("should be in Started state", () => {
|
||||||
|
expect(voiceBroadcastRecording.state).toBe(VoiceBroadcastInfoState.Started);
|
||||||
|
});
|
||||||
|
|
||||||
|
describe("and calling stop()", () => {
|
||||||
|
beforeEach(() => {
|
||||||
|
voiceBroadcastRecording.stop();
|
||||||
|
});
|
||||||
|
|
||||||
|
it("should send a stopped Voice Broadcast Info event", () => {
|
||||||
|
expect(mocked(client.sendStateEvent)).toHaveBeenCalledWith(
|
||||||
|
roomId,
|
||||||
|
VoiceBroadcastInfoEventType,
|
||||||
|
{
|
||||||
|
state: VoiceBroadcastInfoState.Stopped,
|
||||||
|
["m.relates_to"]: {
|
||||||
|
rel_type: RelationType.Reference,
|
||||||
|
event_id: infoEvent.getId(),
|
||||||
|
},
|
||||||
|
},
|
||||||
|
client.getUserId(),
|
||||||
|
);
|
||||||
|
});
|
||||||
|
|
||||||
|
it("should be in state stopped", () => {
|
||||||
|
expect(voiceBroadcastRecording.state).toBe(VoiceBroadcastInfoState.Stopped);
|
||||||
|
});
|
||||||
|
|
||||||
|
it("should emit a stopped state changed event", () => {
|
||||||
|
expect(onStateChanged).toHaveBeenCalledWith(VoiceBroadcastInfoState.Stopped);
|
||||||
|
});
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
describe("when created for a Voice Broadcast Info with a Stopped relation", () => {
|
||||||
|
beforeEach(() => {
|
||||||
|
infoEvent = mkVoiceBroadcastInfoEvent({
|
||||||
|
state: VoiceBroadcastInfoState.Started,
|
||||||
|
chunk_length: 300,
|
||||||
|
});
|
||||||
|
|
||||||
|
const relationsContainer = {
|
||||||
|
getRelations: jest.fn(),
|
||||||
|
} as unknown as Relations;
|
||||||
|
mocked(relationsContainer.getRelations).mockReturnValue([
|
||||||
|
mkVoiceBroadcastInfoEvent({
|
||||||
|
state: VoiceBroadcastInfoState.Stopped,
|
||||||
|
["m.relates_to"]: {
|
||||||
|
rel_type: RelationType.Reference,
|
||||||
|
event_id: infoEvent.getId(),
|
||||||
|
},
|
||||||
|
}),
|
||||||
|
]);
|
||||||
|
|
||||||
|
const timelineSet = {
|
||||||
|
relations: {
|
||||||
|
getChildEventsForEvent: jest.fn().mockImplementation(
|
||||||
|
(
|
||||||
|
eventId: string,
|
||||||
|
relationType: RelationType | string,
|
||||||
|
eventType: EventType | string,
|
||||||
|
) => {
|
||||||
|
if (
|
||||||
|
eventId === infoEvent.getId()
|
||||||
|
&& relationType === RelationType.Reference
|
||||||
|
&& eventType === VoiceBroadcastInfoEventType
|
||||||
|
) {
|
||||||
|
return relationsContainer;
|
||||||
|
}
|
||||||
|
},
|
||||||
|
),
|
||||||
|
},
|
||||||
|
} as unknown as EventTimelineSet;
|
||||||
|
mocked(room.getUnfilteredTimelineSet).mockReturnValue(timelineSet);
|
||||||
|
|
||||||
|
setUpVoiceBroadcastRecording();
|
||||||
|
});
|
||||||
|
|
||||||
|
it("should be in Stopped state", () => {
|
||||||
|
expect(voiceBroadcastRecording.state).toBe(VoiceBroadcastInfoState.Stopped);
|
||||||
|
});
|
||||||
|
});
|
||||||
|
});
|
|
@ -0,0 +1,129 @@
|
||||||
|
/*
|
||||||
|
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 { mocked } from "jest-mock";
|
||||||
|
import { MatrixClient, MatrixEvent, Room } from "matrix-js-sdk/src/matrix";
|
||||||
|
|
||||||
|
import {
|
||||||
|
VoiceBroadcastInfoEventType,
|
||||||
|
VoiceBroadcastRecordingsStore,
|
||||||
|
VoiceBroadcastRecordingsStoreEvent,
|
||||||
|
VoiceBroadcastRecording,
|
||||||
|
} from "../../../src/voice-broadcast";
|
||||||
|
import { mkEvent, mkStubRoom, stubClient } from "../../test-utils";
|
||||||
|
|
||||||
|
jest.mock("../../../src/voice-broadcast/models/VoiceBroadcastRecording.ts", () => ({
|
||||||
|
VoiceBroadcastRecording: jest.fn().mockImplementation(
|
||||||
|
(
|
||||||
|
infoEvent: MatrixEvent,
|
||||||
|
client: MatrixClient,
|
||||||
|
) => ({ infoEvent, client }),
|
||||||
|
),
|
||||||
|
}));
|
||||||
|
|
||||||
|
describe("VoiceBroadcastRecordingsStore", () => {
|
||||||
|
const roomId = "!room:example.com";
|
||||||
|
let client: MatrixClient;
|
||||||
|
let room: Room;
|
||||||
|
let infoEvent: MatrixEvent;
|
||||||
|
let recording: VoiceBroadcastRecording;
|
||||||
|
let recordings: VoiceBroadcastRecordingsStore;
|
||||||
|
let onCurrentChanged: (recording: VoiceBroadcastRecording) => void;
|
||||||
|
|
||||||
|
beforeEach(() => {
|
||||||
|
client = stubClient();
|
||||||
|
room = mkStubRoom(roomId, "test room", client);
|
||||||
|
mocked(client.getRoom).mockImplementation((roomId: string) => {
|
||||||
|
if (roomId === room.roomId) {
|
||||||
|
return room;
|
||||||
|
}
|
||||||
|
});
|
||||||
|
infoEvent = mkEvent({
|
||||||
|
event: true,
|
||||||
|
type: VoiceBroadcastInfoEventType,
|
||||||
|
user: client.getUserId(),
|
||||||
|
room: roomId,
|
||||||
|
content: {},
|
||||||
|
});
|
||||||
|
recording = {
|
||||||
|
infoEvent,
|
||||||
|
} as unknown as VoiceBroadcastRecording;
|
||||||
|
recordings = new VoiceBroadcastRecordingsStore();
|
||||||
|
onCurrentChanged = jest.fn();
|
||||||
|
recordings.on(VoiceBroadcastRecordingsStoreEvent.CurrentChanged, onCurrentChanged);
|
||||||
|
});
|
||||||
|
|
||||||
|
afterEach(() => {
|
||||||
|
recordings.off(VoiceBroadcastRecordingsStoreEvent.CurrentChanged, onCurrentChanged);
|
||||||
|
});
|
||||||
|
|
||||||
|
describe("when setting a current Voice Broadcast recording", () => {
|
||||||
|
beforeEach(() => {
|
||||||
|
recordings.setCurrent(recording);
|
||||||
|
});
|
||||||
|
|
||||||
|
it("should return it as current", () => {
|
||||||
|
expect(recordings.current).toBe(recording);
|
||||||
|
});
|
||||||
|
|
||||||
|
it("should return it by id", () => {
|
||||||
|
expect(recordings.getByInfoEvent(infoEvent, client)).toBe(recording);
|
||||||
|
});
|
||||||
|
|
||||||
|
it("should emit a CurrentChanged event", () => {
|
||||||
|
expect(onCurrentChanged).toHaveBeenCalledWith(recording);
|
||||||
|
});
|
||||||
|
|
||||||
|
describe("and setting the same again", () => {
|
||||||
|
beforeEach(() => {
|
||||||
|
mocked(onCurrentChanged).mockClear();
|
||||||
|
recordings.setCurrent(recording);
|
||||||
|
});
|
||||||
|
|
||||||
|
it("should not emit a CurrentChanged event", () => {
|
||||||
|
expect(onCurrentChanged).not.toHaveBeenCalled();
|
||||||
|
});
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
describe("getByInfoEventId", () => {
|
||||||
|
let returnedRecording: VoiceBroadcastRecording;
|
||||||
|
|
||||||
|
describe("when retrieving a known recording", () => {
|
||||||
|
beforeEach(() => {
|
||||||
|
recordings.setCurrent(recording);
|
||||||
|
returnedRecording = recordings.getByInfoEvent(infoEvent, client);
|
||||||
|
});
|
||||||
|
|
||||||
|
it("should return the recording", () => {
|
||||||
|
expect(returnedRecording).toBe(recording);
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
describe("when retrieving an unknown recording", () => {
|
||||||
|
beforeEach(() => {
|
||||||
|
returnedRecording = recordings.getByInfoEvent(infoEvent, client);
|
||||||
|
});
|
||||||
|
|
||||||
|
it("should return the recording", () => {
|
||||||
|
expect(returnedRecording).toEqual({
|
||||||
|
infoEvent,
|
||||||
|
client,
|
||||||
|
});
|
||||||
|
});
|
||||||
|
});
|
||||||
|
});
|
||||||
|
});
|
|
@ -0,0 +1,140 @@
|
||||||
|
/*
|
||||||
|
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 { mocked } from "jest-mock";
|
||||||
|
import { EventType, MatrixClient, MatrixEvent, Room, RoomStateEvent } from "matrix-js-sdk/src/matrix";
|
||||||
|
|
||||||
|
import {
|
||||||
|
startNewVoiceBroadcastRecording,
|
||||||
|
VoiceBroadcastInfoEventType,
|
||||||
|
VoiceBroadcastInfoState,
|
||||||
|
VoiceBroadcastRecordingsStore,
|
||||||
|
VoiceBroadcastRecording,
|
||||||
|
} from "../../../src/voice-broadcast";
|
||||||
|
import { mkEvent, stubClient } from "../../test-utils";
|
||||||
|
|
||||||
|
jest.mock("../../../src/voice-broadcast/models/VoiceBroadcastRecording", () => ({
|
||||||
|
VoiceBroadcastRecording: jest.fn(),
|
||||||
|
}));
|
||||||
|
|
||||||
|
describe("startNewVoiceBroadcastRecording", () => {
|
||||||
|
const roomId = "!room:example.com";
|
||||||
|
let client: MatrixClient;
|
||||||
|
let recordingsStore: VoiceBroadcastRecordingsStore;
|
||||||
|
let room: Room;
|
||||||
|
let roomOnStateEventsCallbackRegistered: Promise<void>;
|
||||||
|
let roomOnStateEventsCallbackRegisteredResolver: Function;
|
||||||
|
let roomOnStateEventsCallback: () => void;
|
||||||
|
let infoEvent: MatrixEvent;
|
||||||
|
let otherEvent: MatrixEvent;
|
||||||
|
let stateEvent: MatrixEvent;
|
||||||
|
|
||||||
|
beforeEach(() => {
|
||||||
|
roomOnStateEventsCallbackRegistered = new Promise((resolve) => {
|
||||||
|
roomOnStateEventsCallbackRegisteredResolver = resolve;
|
||||||
|
});
|
||||||
|
|
||||||
|
room = {
|
||||||
|
currentState: {
|
||||||
|
getStateEvents: jest.fn().mockImplementation((type, userId) => {
|
||||||
|
if (type === VoiceBroadcastInfoEventType && userId === client.getUserId()) {
|
||||||
|
return stateEvent;
|
||||||
|
}
|
||||||
|
}),
|
||||||
|
},
|
||||||
|
on: jest.fn().mockImplementation((eventType, callback) => {
|
||||||
|
if (eventType === RoomStateEvent.Events) {
|
||||||
|
roomOnStateEventsCallback = callback;
|
||||||
|
roomOnStateEventsCallbackRegisteredResolver();
|
||||||
|
}
|
||||||
|
}),
|
||||||
|
off: jest.fn(),
|
||||||
|
} as unknown as Room;
|
||||||
|
|
||||||
|
client = stubClient();
|
||||||
|
mocked(client.getRoom).mockImplementation((getRoomId: string) => {
|
||||||
|
if (getRoomId === roomId) {
|
||||||
|
return room;
|
||||||
|
}
|
||||||
|
});
|
||||||
|
mocked(client.sendStateEvent).mockImplementation((
|
||||||
|
sendRoomId: string,
|
||||||
|
eventType: string,
|
||||||
|
_content: any,
|
||||||
|
_stateKey: string,
|
||||||
|
) => {
|
||||||
|
if (sendRoomId === roomId && eventType === VoiceBroadcastInfoEventType) {
|
||||||
|
return Promise.resolve({ event_id: infoEvent.getId() });
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
recordingsStore = {
|
||||||
|
setCurrent: jest.fn(),
|
||||||
|
} as unknown as VoiceBroadcastRecordingsStore;
|
||||||
|
|
||||||
|
infoEvent = mkEvent({
|
||||||
|
event: true,
|
||||||
|
type: VoiceBroadcastInfoEventType,
|
||||||
|
content: {
|
||||||
|
state: VoiceBroadcastInfoState.Started,
|
||||||
|
},
|
||||||
|
user: client.getUserId(),
|
||||||
|
room: roomId,
|
||||||
|
});
|
||||||
|
otherEvent = mkEvent({
|
||||||
|
event: true,
|
||||||
|
type: EventType.RoomMember,
|
||||||
|
content: {},
|
||||||
|
user: client.getUserId(),
|
||||||
|
room: roomId,
|
||||||
|
});
|
||||||
|
|
||||||
|
mocked(VoiceBroadcastRecording).mockImplementation((
|
||||||
|
infoEvent: MatrixEvent,
|
||||||
|
client: MatrixClient,
|
||||||
|
) => {
|
||||||
|
return {
|
||||||
|
infoEvent,
|
||||||
|
client,
|
||||||
|
} as unknown as VoiceBroadcastRecording;
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
it("should create a new Voice Broadcast", (done) => {
|
||||||
|
let ok = false;
|
||||||
|
|
||||||
|
startNewVoiceBroadcastRecording(roomId, client, recordingsStore).then((recording) => {
|
||||||
|
expect(ok).toBe(true);
|
||||||
|
expect(mocked(room.off)).toHaveBeenCalledWith(RoomStateEvent.Events, roomOnStateEventsCallback);
|
||||||
|
expect(recording.infoEvent).toBe(infoEvent);
|
||||||
|
done();
|
||||||
|
});
|
||||||
|
|
||||||
|
roomOnStateEventsCallbackRegistered.then(() => {
|
||||||
|
// no state event, yet
|
||||||
|
roomOnStateEventsCallback();
|
||||||
|
|
||||||
|
// other state event
|
||||||
|
stateEvent = otherEvent;
|
||||||
|
roomOnStateEventsCallback();
|
||||||
|
|
||||||
|
// the expected Voice Broadcast Info event
|
||||||
|
stateEvent = infoEvent;
|
||||||
|
ok = true;
|
||||||
|
roomOnStateEventsCallback();
|
||||||
|
});
|
||||||
|
});
|
||||||
|
});
|
Loading…
Add table
Add a link
Reference in a new issue