|
@@ -11,6 +11,7 @@ import { AbstractDialogTab } from '../../../base/dialog';
|
11
|
11
|
import type { Props as AbstractDialogTabProps } from '../../../base/dialog';
|
12
|
12
|
import { translate } from '../../../base/i18n';
|
13
|
13
|
import TouchmoveHack from '../../../chat/components/web/TouchmoveHack';
|
|
14
|
+import { SS_DEFAULT_FRAME_RATE } from '../../constants';
|
14
|
15
|
|
15
|
16
|
/**
|
16
|
17
|
* The type of the React {@code Component} props of {@link MoreTab}.
|
|
@@ -18,12 +19,22 @@ import TouchmoveHack from '../../../chat/components/web/TouchmoveHack';
|
18
|
19
|
export type Props = {
|
19
|
20
|
...$Exact<AbstractDialogTabProps>,
|
20
|
21
|
|
|
22
|
+ /**
|
|
23
|
+ * The currently selected desktop share frame rate in the frame rate select dropdown.
|
|
24
|
+ */
|
|
25
|
+ currentFramerate: string,
|
|
26
|
+
|
21
|
27
|
/**
|
22
|
28
|
* The currently selected language to display in the language select
|
23
|
29
|
* dropdown.
|
24
|
30
|
*/
|
25
|
31
|
currentLanguage: string,
|
26
|
32
|
|
|
33
|
+ /**
|
|
34
|
+ * All available desktop capture frame rates.
|
|
35
|
+ */
|
|
36
|
+ desktopShareFramerates: Array<number>,
|
|
37
|
+
|
27
|
38
|
/**
|
28
|
39
|
* Whether or not follow me is currently active (enabled by some other participant).
|
29
|
40
|
*/
|
|
@@ -59,7 +70,6 @@ export type Props = {
|
59
|
70
|
*/
|
60
|
71
|
showPrejoinPage: boolean,
|
61
|
72
|
|
62
|
|
-
|
63
|
73
|
/**
|
64
|
74
|
* Whether or not the user has selected the Start Audio Muted feature to be
|
65
|
75
|
* enabled.
|
|
@@ -83,6 +93,11 @@ export type Props = {
|
83
|
93
|
*/
|
84
|
94
|
type State = {
|
85
|
95
|
|
|
96
|
+ /**
|
|
97
|
+ * Whether or not the desktop share frame rate select dropdown is open.
|
|
98
|
+ */
|
|
99
|
+ isFramerateSelectOpen: boolean,
|
|
100
|
+
|
86
|
101
|
/**
|
87
|
102
|
* Whether or not the language select dropdown is open.
|
88
|
103
|
*/
|
|
@@ -105,12 +120,14 @@ class MoreTab extends AbstractDialogTab<Props, State> {
|
105
|
120
|
super(props);
|
106
|
121
|
|
107
|
122
|
this.state = {
|
|
123
|
+ isFramerateSelectOpen: false,
|
108
|
124
|
isLanguageSelectOpen: false
|
109
|
125
|
};
|
110
|
126
|
|
111
|
127
|
// Bind event handler so it is only bound once for every instance.
|
112
|
|
- this._onLanguageDropdownOpenChange
|
113
|
|
- = this._onLanguageDropdownOpenChange.bind(this);
|
|
128
|
+ this._onFramerateDropdownOpenChange = this._onFramerateDropdownOpenChange.bind(this);
|
|
129
|
+ this._onFramerateItemSelect = this._onFramerateItemSelect.bind(this);
|
|
130
|
+ this._onLanguageDropdownOpenChange = this._onLanguageDropdownOpenChange.bind(this);
|
114
|
131
|
this._onLanguageItemSelect = this._onLanguageItemSelect.bind(this);
|
115
|
132
|
this._onStartAudioMutedChanged = this._onStartAudioMutedChanged.bind(this);
|
116
|
133
|
this._onStartVideoMutedChanged = this._onStartVideoMutedChanged.bind(this);
|
|
@@ -126,25 +143,40 @@ class MoreTab extends AbstractDialogTab<Props, State> {
|
126
|
143
|
* @returns {ReactElement}
|
127
|
144
|
*/
|
128
|
145
|
render() {
|
129
|
|
- const { showModeratorSettings, showLanguageSettings, showPrejoinSettings } = this.props;
|
130
|
146
|
const content = [];
|
131
|
147
|
|
132
|
|
- if (showPrejoinSettings) {
|
133
|
|
- content.push(this._renderPrejoinScreenSettings());
|
134
|
|
- }
|
|
148
|
+ content.push(this._renderSettingsLeft());
|
|
149
|
+ content.push(this._renderSettingsRight());
|
135
|
150
|
|
136
|
|
- content.push(this._renderKeyboardShortcutCheckbox());
|
|
151
|
+ return <div className = 'more-tab box'>{ content }</div>;
|
|
152
|
+ }
|
137
|
153
|
|
|
154
|
+ _onFramerateDropdownOpenChange: (Object) => void;
|
138
|
155
|
|
139
|
|
- if (showModeratorSettings) {
|
140
|
|
- content.push(this._renderModeratorSettings());
|
141
|
|
- }
|
|
156
|
+ /**
|
|
157
|
+ * Callback invoked to toggle display of the desktop share framerate select dropdown.
|
|
158
|
+ *
|
|
159
|
+ * @param {Object} event - The event for opening or closing the dropdown.
|
|
160
|
+ * @private
|
|
161
|
+ * @returns {void}
|
|
162
|
+ */
|
|
163
|
+ _onFramerateDropdownOpenChange({ isOpen }) {
|
|
164
|
+ this.setState({ isFramerateSelectOpen: isOpen });
|
|
165
|
+ }
|
|
166
|
+
|
|
167
|
+ _onFramerateItemSelect: (Object) => void;
|
142
|
168
|
|
143
|
|
- if (showLanguageSettings) {
|
144
|
|
- content.push(this._renderLangaugeSelect());
|
145
|
|
- }
|
|
169
|
+ /**
|
|
170
|
+ * Callback invoked to select a frame rate from the select dropdown.
|
|
171
|
+ *
|
|
172
|
+ * @param {Object} e - The key event to handle.
|
|
173
|
+ * @private
|
|
174
|
+ * @returns {void}
|
|
175
|
+ */
|
|
176
|
+ _onFramerateItemSelect(e) {
|
|
177
|
+ const frameRate = e.currentTarget.getAttribute('data-framerate');
|
146
|
178
|
|
147
|
|
- return <div className = 'more-tab'>{ content }</div>;
|
|
179
|
+ super._onChange({ currentFramerate: frameRate });
|
148
|
180
|
}
|
149
|
181
|
|
150
|
182
|
_onLanguageDropdownOpenChange: (Object) => void;
|
|
@@ -246,13 +278,89 @@ class MoreTab extends AbstractDialogTab<Props, State> {
|
246
|
278
|
super._onChange({ keyboardShortcutEnable: checked });
|
247
|
279
|
}
|
248
|
280
|
|
|
281
|
+ /**
|
|
282
|
+ * Returns the React Element for the desktop share frame rate dropdown.
|
|
283
|
+ *
|
|
284
|
+ * @returns {ReactElement}
|
|
285
|
+ */
|
|
286
|
+ _renderFramerateSelect() {
|
|
287
|
+ const { currentFramerate, desktopShareFramerates, t } = this.props;
|
|
288
|
+ const frameRateItems = desktopShareFramerates.map(frameRate => (
|
|
289
|
+ <DropdownItem
|
|
290
|
+ data-framerate = { frameRate }
|
|
291
|
+ key = { frameRate }
|
|
292
|
+ onClick = { this._onFramerateItemSelect }>
|
|
293
|
+ { `${frameRate} ${t('settings.framesPerSecond')}` }
|
|
294
|
+ </DropdownItem>));
|
|
295
|
+
|
|
296
|
+ return (
|
|
297
|
+ <div
|
|
298
|
+ className = 'settings-sub-pane-element'
|
|
299
|
+ key = 'frameRate'>
|
|
300
|
+ <h2 className = 'mock-atlaskit-label'>
|
|
301
|
+ { t('settings.desktopShareFramerate') }
|
|
302
|
+ </h2>
|
|
303
|
+ <div className = 'dropdown-menu'>
|
|
304
|
+ <TouchmoveHack isModal = { true }>
|
|
305
|
+ <DropdownMenu
|
|
306
|
+ isOpen = { this.state.isFramerateSelectOpen }
|
|
307
|
+ onOpenChange = { this._onFramerateDropdownOpenChange }
|
|
308
|
+ shouldFitContainer = { true }
|
|
309
|
+ trigger = { currentFramerate
|
|
310
|
+ ? `${currentFramerate} ${t('settings.framesPerSecond')}`
|
|
311
|
+ : '' }
|
|
312
|
+ triggerButtonProps = {{
|
|
313
|
+ shouldFitContainer: true
|
|
314
|
+ }}
|
|
315
|
+ triggerType = 'button'>
|
|
316
|
+ <DropdownItemGroup>
|
|
317
|
+ { frameRateItems }
|
|
318
|
+ </DropdownItemGroup>
|
|
319
|
+ </DropdownMenu>
|
|
320
|
+ </TouchmoveHack>
|
|
321
|
+ </div>
|
|
322
|
+ <div
|
|
323
|
+ className = 'mock-atlaskit-label'>
|
|
324
|
+ { parseInt(currentFramerate, 10) > SS_DEFAULT_FRAME_RATE
|
|
325
|
+ ? t('settings.desktopShareHighFpsWarning')
|
|
326
|
+ : t('settings.desktopShareWarning') }
|
|
327
|
+ </div>
|
|
328
|
+ </div>
|
|
329
|
+ );
|
|
330
|
+ }
|
|
331
|
+
|
|
332
|
+ /**
|
|
333
|
+ * Returns the React Element for keyboardShortcut settings.
|
|
334
|
+ *
|
|
335
|
+ * @private
|
|
336
|
+ * @returns {ReactElement}
|
|
337
|
+ */
|
|
338
|
+ _renderKeyboardShortcutCheckbox() {
|
|
339
|
+ const { t } = this.props;
|
|
340
|
+
|
|
341
|
+ return (
|
|
342
|
+ <div
|
|
343
|
+ className = 'settings-sub-pane-element'
|
|
344
|
+ key = 'keyboard-shortcut'>
|
|
345
|
+ <h2 className = 'mock-atlaskit-label'>
|
|
346
|
+ { t('keyboardShortcuts.keyboardShortcuts') }
|
|
347
|
+ </h2>
|
|
348
|
+ <Checkbox
|
|
349
|
+ isChecked = { keyboardShortcut.getEnabled() }
|
|
350
|
+ label = { t('prejoin.keyboardShortcuts') }
|
|
351
|
+ name = 'enable-keyboard-shortcuts'
|
|
352
|
+ onChange = { this._onKeyboardShortcutEnableChanged } />
|
|
353
|
+ </div>
|
|
354
|
+ );
|
|
355
|
+ }
|
|
356
|
+
|
249
|
357
|
/**
|
250
|
358
|
* Returns the menu item for changing displayed language.
|
251
|
359
|
*
|
252
|
360
|
* @private
|
253
|
361
|
* @returns {ReactElement}
|
254
|
362
|
*/
|
255
|
|
- _renderLangaugeSelect() {
|
|
363
|
+ _renderLanguageSelect() {
|
256
|
364
|
const {
|
257
|
365
|
currentLanguage,
|
258
|
366
|
languages,
|
|
@@ -270,7 +378,7 @@ class MoreTab extends AbstractDialogTab<Props, State> {
|
270
|
378
|
|
271
|
379
|
return (
|
272
|
380
|
<div
|
273
|
|
- className = 'settings-sub-pane language-settings'
|
|
381
|
+ className = 'settings-sub-pane-element'
|
274
|
382
|
key = 'language'>
|
275
|
383
|
<h2 className = 'mock-atlaskit-label'>
|
276
|
384
|
{ t('settings.language') }
|
|
@@ -315,7 +423,7 @@ class MoreTab extends AbstractDialogTab<Props, State> {
|
315
|
423
|
|
316
|
424
|
return (
|
317
|
425
|
<div
|
318
|
|
- className = 'settings-sub-pane'
|
|
426
|
+ className = 'settings-sub-pane-element'
|
319
|
427
|
key = 'moderator'>
|
320
|
428
|
<h2 className = 'mock-atlaskit-label'>
|
321
|
429
|
{ t('settings.moderator') }
|
|
@@ -351,7 +459,7 @@ class MoreTab extends AbstractDialogTab<Props, State> {
|
351
|
459
|
|
352
|
460
|
return (
|
353
|
461
|
<div
|
354
|
|
- className = 'settings-sub-pane'
|
|
462
|
+ className = 'settings-sub-pane-element'
|
355
|
463
|
key = 'prejoin-screen'>
|
356
|
464
|
<h2 className = 'mock-atlaskit-label'>
|
357
|
465
|
{ t('prejoin.premeeting') }
|
|
@@ -366,26 +474,37 @@ class MoreTab extends AbstractDialogTab<Props, State> {
|
366
|
474
|
}
|
367
|
475
|
|
368
|
476
|
/**
|
369
|
|
- * Returns the React Element for keyboardShortcut settings.
|
|
477
|
+ * Returns the React element that needs to be displayed on the right half of the more tabs.
|
370
|
478
|
*
|
371
|
479
|
* @private
|
372
|
480
|
* @returns {ReactElement}
|
373
|
481
|
*/
|
374
|
|
- _renderKeyboardShortcutCheckbox() {
|
375
|
|
- const { t } = this.props;
|
|
482
|
+ _renderSettingsRight() {
|
|
483
|
+ const { showLanguageSettings } = this.props;
|
376
|
484
|
|
377
|
485
|
return (
|
378
|
486
|
<div
|
379
|
|
- className = 'settings-sub-pane'
|
380
|
|
- key = 'keyboard-shortcut'>
|
381
|
|
- <h2 className = 'mock-atlaskit-label'>
|
382
|
|
- { t('keyboardShortcuts.keyboardShortcuts') }
|
383
|
|
- </h2>
|
384
|
|
- <Checkbox
|
385
|
|
- isChecked = { keyboardShortcut.getEnabled() }
|
386
|
|
- label = { t('prejoin.keyboardShortcuts') }
|
387
|
|
- name = 'enable-keyboard-shortcuts'
|
388
|
|
- onChange = { this._onKeyboardShortcutEnableChanged } />
|
|
487
|
+ className = 'settings-sub-pane right'>
|
|
488
|
+ { showLanguageSettings && this._renderLanguageSelect() }
|
|
489
|
+ { this._renderFramerateSelect() }
|
|
490
|
+ </div>
|
|
491
|
+ );
|
|
492
|
+ }
|
|
493
|
+
|
|
494
|
+ /**
|
|
495
|
+ * Returns the React element that needs to be displayed on the left half of the more tabs.
|
|
496
|
+ *
|
|
497
|
+ * @returns {ReactElement}
|
|
498
|
+ */
|
|
499
|
+ _renderSettingsLeft() {
|
|
500
|
+ const { showPrejoinSettings, showModeratorSettings } = this.props;
|
|
501
|
+
|
|
502
|
+ return (
|
|
503
|
+ <div
|
|
504
|
+ className = 'settings-sub-pane left'>
|
|
505
|
+ { showPrejoinSettings && this._renderPrejoinScreenSettings() }
|
|
506
|
+ { this._renderKeyboardShortcutCheckbox() }
|
|
507
|
+ { showModeratorSettings && this._renderModeratorSettings() }
|
389
|
508
|
</div>
|
390
|
509
|
);
|
391
|
510
|
}
|