mirror of https://github.com/vector-im/riot-web
180 lines
5.4 KiB
TypeScript
180 lines
5.4 KiB
TypeScript
/*
|
|
Copyright 2021 Šimon Brandner <simon.bra.ag@gmail.com>
|
|
|
|
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 { _t } from '../../../languageHandler';
|
|
import BaseDialog from "..//dialogs/BaseDialog";
|
|
import DialogButtons from "./DialogButtons";
|
|
import classNames from 'classnames';
|
|
import AccessibleButton from './AccessibleButton';
|
|
import { getDesktopCapturerSources } from "matrix-js-sdk/src/webrtc/call";
|
|
import { replaceableComponent } from "../../../utils/replaceableComponent";
|
|
import TabbedView, { Tab, TabLocation } from '../../structures/TabbedView';
|
|
|
|
export interface DesktopCapturerSource {
|
|
id: string;
|
|
name: string;
|
|
thumbnailURL;
|
|
}
|
|
|
|
export enum Tabs {
|
|
Screens = "screen",
|
|
Windows = "window",
|
|
}
|
|
|
|
export interface ExistingSourceIProps {
|
|
source: DesktopCapturerSource;
|
|
onSelect(source: DesktopCapturerSource): void;
|
|
selected: boolean;
|
|
}
|
|
|
|
export class ExistingSource extends React.Component<ExistingSourceIProps> {
|
|
constructor(props: ExistingSourceIProps) {
|
|
super(props);
|
|
}
|
|
|
|
private onClick = (): void => {
|
|
this.props.onSelect(this.props.source);
|
|
};
|
|
|
|
render() {
|
|
const thumbnailClasses = classNames({
|
|
mx_desktopCapturerSourcePicker_source_thumbnail: true,
|
|
mx_desktopCapturerSourcePicker_source_thumbnail_selected: this.props.selected,
|
|
});
|
|
|
|
return (
|
|
<AccessibleButton
|
|
className="mx_desktopCapturerSourcePicker_source"
|
|
title={this.props.source.name}
|
|
onClick={this.onClick}
|
|
>
|
|
<img
|
|
className={thumbnailClasses}
|
|
src={this.props.source.thumbnailURL}
|
|
/>
|
|
<span className="mx_desktopCapturerSourcePicker_source_name">{ this.props.source.name }</span>
|
|
</AccessibleButton>
|
|
);
|
|
}
|
|
}
|
|
|
|
export interface PickerIState {
|
|
selectedTab: Tabs;
|
|
sources: Array<DesktopCapturerSource>;
|
|
selectedSource: DesktopCapturerSource | null;
|
|
}
|
|
export interface PickerIProps {
|
|
onFinished(source: DesktopCapturerSource): void;
|
|
}
|
|
|
|
@replaceableComponent("views.elements.DesktopCapturerSourcePicker")
|
|
export default class DesktopCapturerSourcePicker extends React.Component<
|
|
PickerIProps,
|
|
PickerIState
|
|
> {
|
|
interval: number;
|
|
|
|
constructor(props: PickerIProps) {
|
|
super(props);
|
|
|
|
this.state = {
|
|
selectedTab: Tabs.Screens,
|
|
sources: [],
|
|
selectedSource: null,
|
|
};
|
|
}
|
|
|
|
async componentDidMount() {
|
|
// setInterval() first waits and then executes, therefore
|
|
// we call getDesktopCapturerSources() here without any delay.
|
|
// Otherwise the dialog would be left empty for some time.
|
|
this.setState({
|
|
sources: await getDesktopCapturerSources(),
|
|
});
|
|
|
|
// We update the sources every 500ms to get newer thumbnails
|
|
this.interval = setInterval(async () => {
|
|
this.setState({
|
|
sources: await getDesktopCapturerSources(),
|
|
});
|
|
}, 500);
|
|
}
|
|
|
|
componentWillUnmount() {
|
|
clearInterval(this.interval);
|
|
}
|
|
|
|
private onSelect = (source: DesktopCapturerSource): void => {
|
|
this.setState({ selectedSource: source });
|
|
};
|
|
|
|
private onShare = (): void => {
|
|
this.props.onFinished(this.state.selectedSource);
|
|
};
|
|
|
|
private onTabChange = (): void => {
|
|
this.setState({ selectedSource: null });
|
|
};
|
|
|
|
private onCloseClick = (): void => {
|
|
this.props.onFinished(null);
|
|
};
|
|
|
|
private getTab(type: "screen" | "window", label: string): Tab {
|
|
const sources = this.state.sources.filter((source) => source.id.startsWith(type)).map((source) => {
|
|
return (
|
|
<ExistingSource
|
|
selected={this.state.selectedSource?.id === source.id}
|
|
source={source}
|
|
onSelect={this.onSelect}
|
|
key={source.id}
|
|
/>
|
|
);
|
|
});
|
|
|
|
return new Tab(type, label, null, (
|
|
<div className="mx_desktopCapturerSourcePicker_tab">
|
|
{ sources }
|
|
</div>
|
|
));
|
|
}
|
|
|
|
render() {
|
|
const tabs = [
|
|
this.getTab("screen", _t("Share entire screen")),
|
|
this.getTab("window", _t("Application window")),
|
|
];
|
|
|
|
return (
|
|
<BaseDialog
|
|
className="mx_desktopCapturerSourcePicker"
|
|
onFinished={this.onCloseClick}
|
|
title={_t("Share content")}
|
|
>
|
|
<TabbedView tabs={tabs} tabLocation={TabLocation.TOP} onChange={this.onTabChange} />
|
|
<DialogButtons
|
|
primaryButton={_t("Share")}
|
|
hasCancel={true}
|
|
onCancel={this.onCloseClick}
|
|
onPrimaryButtonClick={this.onShare}
|
|
primaryDisabled={!this.state.selectedSource}
|
|
/>
|
|
</BaseDialog>
|
|
);
|
|
}
|
|
}
|