mirror of
https://github.com/vector-im/element-call.git
synced 2026-03-07 05:47:03 +00:00
0e0fba6575 added the ability to send call notification events when starting a call, but I forgot to give the widget the right capabilities to do this. The effect was that notifications just wouldn't send in widget mode.
196 lines
6.6 KiB
TypeScript
196 lines
6.6 KiB
TypeScript
/*
|
|
Copyright 2022-2024 New Vector Ltd.
|
|
|
|
SPDX-License-Identifier: AGPL-3.0-only OR LicenseRef-Element-Commercial
|
|
Please see LICENSE in the repository root for full details.
|
|
*/
|
|
|
|
import { logger } from "matrix-js-sdk/lib/logger";
|
|
import { EventType, createRoomWidgetClient } from "matrix-js-sdk";
|
|
import {
|
|
WidgetApi,
|
|
MatrixCapabilities,
|
|
WidgetApiToWidgetAction,
|
|
} from "matrix-widget-api";
|
|
|
|
import type { MatrixClient } from "matrix-js-sdk";
|
|
import type { IWidgetApiRequest } from "matrix-widget-api";
|
|
import { LazyEventEmitter } from "./LazyEventEmitter";
|
|
import { getUrlParams } from "./UrlParams";
|
|
import { Config } from "./config/Config";
|
|
import { ElementCallReactionEventType } from "./reactions";
|
|
|
|
// Subset of the actions in element-web
|
|
export enum ElementWidgetActions {
|
|
JoinCall = "io.element.join",
|
|
HangupCall = "im.vector.hangup",
|
|
Close = "io.element.close",
|
|
TileLayout = "io.element.tile_layout",
|
|
SpotlightLayout = "io.element.spotlight_layout",
|
|
// This can be sent as from or to widget
|
|
// fromWidget: updates the client about the current device mute state
|
|
// toWidget: the client requests a specific device mute configuration
|
|
// The reply will always be the resulting configuration
|
|
// It is possible to sent an empty configuration to retrieve the current values or
|
|
// just one of the fields to update that particular value
|
|
// An undefined field means that EC will keep the mute state as is.
|
|
// -> this will allow the client to only get the current state
|
|
//
|
|
// The data of the widget action request and the response are:
|
|
// {
|
|
// audio_enabled?: boolean,
|
|
// video_enabled?: boolean
|
|
// }
|
|
DeviceMute = "io.element.device_mute",
|
|
}
|
|
|
|
export interface JoinCallData {
|
|
audioInput: string | null;
|
|
videoInput: string | null;
|
|
}
|
|
|
|
export interface WidgetHelpers {
|
|
api: WidgetApi;
|
|
lazyActions: LazyEventEmitter;
|
|
client: Promise<MatrixClient>;
|
|
}
|
|
|
|
/**
|
|
* A point of access to the widget API, if the app is running as a widget. This
|
|
* is declared and initialized on the top level because the widget messaging
|
|
* needs to be set up ASAP on load to ensure it doesn't miss any requests.
|
|
*/
|
|
export const widget = ((): WidgetHelpers | null => {
|
|
try {
|
|
const { widgetId, parentUrl } = getUrlParams();
|
|
|
|
if (widgetId && parentUrl) {
|
|
const parentOrigin = new URL(parentUrl).origin;
|
|
logger.info("Widget API is available");
|
|
const api = new WidgetApi(widgetId, parentOrigin);
|
|
api.requestCapability(MatrixCapabilities.AlwaysOnScreen);
|
|
|
|
// Set up the lazy action emitter, but only for select actions that we
|
|
// intend for the app to handle
|
|
const lazyActions = new LazyEventEmitter();
|
|
[
|
|
WidgetApiToWidgetAction.ThemeChange,
|
|
ElementWidgetActions.JoinCall,
|
|
ElementWidgetActions.HangupCall,
|
|
ElementWidgetActions.TileLayout,
|
|
ElementWidgetActions.SpotlightLayout,
|
|
ElementWidgetActions.DeviceMute,
|
|
].forEach((action) => {
|
|
api.on(`action:${action}`, (ev: CustomEvent<IWidgetApiRequest>) => {
|
|
ev.preventDefault();
|
|
lazyActions.emit(action, ev);
|
|
});
|
|
});
|
|
|
|
// Now, initialize the matryoshka MatrixClient (so named because it routes
|
|
// all requests through the host client via the widget API)
|
|
// We need to do this now rather than later because it has capabilities to
|
|
// request, and is responsible for starting the transport (should it be?)
|
|
|
|
const {
|
|
roomId,
|
|
userId,
|
|
deviceId,
|
|
baseUrl,
|
|
e2eEnabled,
|
|
allowIceFallback,
|
|
} = getUrlParams();
|
|
if (!roomId) throw new Error("Room ID must be supplied");
|
|
if (!userId) throw new Error("User ID must be supplied");
|
|
if (!deviceId) throw new Error("Device ID must be supplied");
|
|
if (!baseUrl) throw new Error("Base URL must be supplied");
|
|
|
|
// These are all the event types the app uses
|
|
const sendEvent = [
|
|
EventType.CallNotify, // Sent as a deprecated fallback
|
|
EventType.RTCNotification,
|
|
];
|
|
const sendRecvEvent = [
|
|
"org.matrix.rageshake_request",
|
|
EventType.CallEncryptionKeysPrefix,
|
|
EventType.Reaction,
|
|
EventType.RoomRedaction,
|
|
ElementCallReactionEventType,
|
|
];
|
|
|
|
const sendState = [
|
|
userId, // Legacy call membership events
|
|
`_${userId}_${deviceId}`, // Session membership events
|
|
`${userId}_${deviceId}`, // The above with no leading underscore, for room versions whose auth rules allow it
|
|
].map((stateKey) => ({
|
|
eventType: EventType.GroupCallMemberPrefix,
|
|
stateKey,
|
|
}));
|
|
const receiveState = [
|
|
{ eventType: EventType.RoomCreate },
|
|
{ eventType: EventType.RoomName },
|
|
{ eventType: EventType.RoomMember },
|
|
{ eventType: EventType.RoomEncryption },
|
|
{ eventType: EventType.GroupCallMemberPrefix },
|
|
];
|
|
|
|
const sendRecvToDevice = [
|
|
EventType.CallInvite,
|
|
EventType.CallCandidates,
|
|
EventType.CallAnswer,
|
|
EventType.CallHangup,
|
|
EventType.CallReject,
|
|
EventType.CallSelectAnswer,
|
|
EventType.CallNegotiate,
|
|
EventType.CallSDPStreamMetadataChanged,
|
|
EventType.CallSDPStreamMetadataChangedPrefix,
|
|
EventType.CallReplaces,
|
|
EventType.CallEncryptionKeysPrefix,
|
|
];
|
|
|
|
const client = createRoomWidgetClient(
|
|
api,
|
|
{
|
|
sendEvent: [...sendEvent, ...sendRecvEvent],
|
|
receiveEvent: sendRecvEvent,
|
|
sendState,
|
|
receiveState,
|
|
sendToDevice: sendRecvToDevice,
|
|
receiveToDevice: sendRecvToDevice,
|
|
turnServers: false,
|
|
sendDelayedEvents: true,
|
|
updateDelayedEvents: true,
|
|
},
|
|
roomId,
|
|
{
|
|
baseUrl,
|
|
userId,
|
|
deviceId,
|
|
timelineSupport: true,
|
|
useE2eForGroupCall: e2eEnabled,
|
|
fallbackICEServerAllowed: allowIceFallback,
|
|
},
|
|
// ContentLoaded event will be sent as soon as the theme is set (see useTheme.ts)
|
|
false,
|
|
);
|
|
|
|
const clientPromise = async (): Promise<MatrixClient> => {
|
|
// Wait for the config file to be ready (we load very early on so it might not
|
|
// be otherwise)
|
|
await Config.init();
|
|
await client.startClient({ clientWellKnownPollPeriod: 60 * 10 });
|
|
return client;
|
|
};
|
|
|
|
return { api, lazyActions, client: clientPromise() };
|
|
} else {
|
|
if (import.meta.env.MODE !== "test")
|
|
logger.info("No widget API available");
|
|
return null;
|
|
}
|
|
} catch (e) {
|
|
logger.warn("Continuing without the widget API", e);
|
|
return null;
|
|
}
|
|
})();
|