signal-desktop/ts/util/desktopCapturer.ts

230 lines
6.2 KiB
TypeScript
Raw Normal View History

2024-09-20 01:03:44 +00:00
// Copyright 2024 Signal Messenger, LLC
// SPDX-License-Identifier: AGPL-3.0-only
import { ipcRenderer, type DesktopCapturerSource } from 'electron';
import * as log from '../logging/log';
import * as Errors from '../types/errors';
import type { PresentableSource } from '../types/Calling';
import type { LocalizerType } from '../types/Util';
import {
REQUESTED_VIDEO_WIDTH,
REQUESTED_VIDEO_HEIGHT,
REQUESTED_VIDEO_FRAMERATE,
} from '../calling/constants';
import { strictAssert } from './assert';
import { explodePromise } from './explodePromise';
import { isNotNil } from './isNotNil';
enum Step {
RequestingMedia = 'RequestingMedia',
Done = 'Done',
Error = 'Error',
// Skipped on macOS Sequoia
SelectingSource = 'SelectingSource',
SelectedSource = 'SelectedSource',
}
type State = Readonly<
| {
step: Step.RequestingMedia;
promise: Promise<void>;
}
| {
step: Step.SelectingSource;
promise: Promise<void>;
sources: ReadonlyArray<DesktopCapturerSource>;
onSource: (source: DesktopCapturerSource | undefined) => void;
}
| {
step: Step.SelectedSource;
promise: Promise<void>;
}
| {
step: Step.Done;
}
| {
step: Step.Error;
}
>;
export const liveCapturers = new Set<DesktopCapturer>();
export type IpcResponseType = Readonly<{
id: string;
sources: ReadonlyArray<DesktopCapturerSource>;
}>;
export type DesktopCapturerOptionsType = Readonly<{
i18n: LocalizerType;
onPresentableSources(sources: ReadonlyArray<PresentableSource>): void;
onMediaStream(stream: MediaStream): void;
onError(error: Error): void;
}>;
export type DesktopCapturerBaton = Readonly<{
__desktop_capturer_baton: never;
}>;
export class DesktopCapturer {
private state: State;
private static isInitialized = false;
// For use as a key in weak maps
public readonly baton = {} as DesktopCapturerBaton;
constructor(private readonly options: DesktopCapturerOptionsType) {
if (!DesktopCapturer.isInitialized) {
DesktopCapturer.initialize();
}
this.state = { step: Step.RequestingMedia, promise: this.getStream() };
}
public selectSource(id: string | undefined): void {
strictAssert(
this.state.step === Step.SelectingSource,
`Invalid state in "selectSource" ${this.state.step}`
);
const { promise, sources, onSource } = this.state;
const source = id == null ? undefined : sources.find(s => s.id === id);
this.state = { step: Step.SelectedSource, promise };
onSource(source);
}
/** @internal */
private onSources(
sources: ReadonlyArray<DesktopCapturerSource>
): Promise<DesktopCapturerSource | undefined> {
strictAssert(
this.state.step === Step.RequestingMedia,
`Invalid state in "onSources" ${this.state.step}`
);
const presentableSources = sources
.map(source => {
// If electron can't retrieve a thumbnail then it won't be able to
// present this source so we filter these out.
if (source.thumbnail.isEmpty()) {
return undefined;
}
return {
appIcon:
source.appIcon && !source.appIcon.isEmpty()
? source.appIcon.toDataURL()
: undefined,
id: source.id,
name: this.translateSourceName(source),
isScreen: isScreenSource(source),
thumbnail: source.thumbnail.toDataURL(),
};
})
.filter(isNotNil);
const { promise } = this.state;
const { promise: source, resolve: onSource } = explodePromise<
DesktopCapturerSource | undefined
>();
this.state = { step: Step.SelectingSource, promise, sources, onSource };
this.options.onPresentableSources(presentableSources);
return source;
}
private async getStream(): Promise<void> {
liveCapturers.add(this);
try {
const stream = await navigator.mediaDevices.getDisplayMedia({
video: {
width: {
max: REQUESTED_VIDEO_WIDTH,
ideal: REQUESTED_VIDEO_WIDTH,
},
height: {
max: REQUESTED_VIDEO_HEIGHT,
ideal: REQUESTED_VIDEO_HEIGHT,
},
frameRate: {
max: REQUESTED_VIDEO_FRAMERATE,
ideal: REQUESTED_VIDEO_FRAMERATE,
},
},
});
strictAssert(
this.state.step === Step.RequestingMedia ||
this.state.step === Step.SelectedSource,
`Invalid state in "getStream.success" ${this.state.step}`
);
this.options.onMediaStream(stream);
this.state = { step: Step.Done };
} catch (error) {
strictAssert(
this.state.step === Step.RequestingMedia ||
this.state.step === Step.SelectedSource,
`Invalid state in "getStream.error" ${this.state.step}`
);
this.options.onError(error);
this.state = { step: Step.Error };
} finally {
liveCapturers.delete(this);
}
}
private translateSourceName(source: DesktopCapturerSource): string {
const { i18n } = this.options;
const { name } = source;
if (!isScreenSource(source)) {
return name;
}
if (name === 'Entire Screen') {
return i18n('icu:calling__SelectPresentingSourcesModal--entireScreen');
}
const match = name.match(/^Screen (\d+)$/);
if (match) {
return i18n('icu:calling__SelectPresentingSourcesModal--screen', {
id: match[1],
});
}
return name;
}
private static initialize(): void {
DesktopCapturer.isInitialized = true;
ipcRenderer.on(
'select-capture-sources',
async (_, { id, sources }: IpcResponseType) => {
let selected: DesktopCapturerSource | undefined;
try {
const { value: capturer, done } = liveCapturers.values().next();
strictAssert(!done, 'No capturer available for incoming sources');
liveCapturers.delete(capturer);
selected = await capturer.onSources(sources);
} catch (error) {
log.error(
'desktopCapturer: failed to get the source',
Errors.toLogFormat(error)
);
}
ipcRenderer.send(`select-capture-sources:${id}:response`, selected);
}
);
}
}
function isScreenSource(source: DesktopCapturerSource): boolean {
return source.id.startsWith('screen');
}