2021-06-25 16:08:16 +00:00
|
|
|
// Copyright 2018-2021 Signal Messenger, LLC
|
|
|
|
// SPDX-License-Identifier: AGPL-3.0-only
|
|
|
|
|
2021-10-26 19:15:33 +00:00
|
|
|
import type { KeyboardEvent } from 'react';
|
|
|
|
import React, { useCallback, useEffect, useState } from 'react';
|
2021-06-25 16:08:16 +00:00
|
|
|
import { noop } from 'lodash';
|
|
|
|
import { createPortal } from 'react-dom';
|
|
|
|
import classNames from 'classnames';
|
|
|
|
import { Manager, Popper, Reference } from 'react-popper';
|
2021-10-26 19:15:33 +00:00
|
|
|
import type { LocalizerType } from '../types/Util';
|
2021-06-25 16:08:16 +00:00
|
|
|
|
|
|
|
export type PropsType = {
|
|
|
|
i18n: LocalizerType;
|
|
|
|
isHighQuality: boolean;
|
|
|
|
onSelectQuality: (isHQ: boolean) => unknown;
|
|
|
|
};
|
|
|
|
|
|
|
|
export const MediaQualitySelector = ({
|
|
|
|
i18n,
|
|
|
|
isHighQuality,
|
|
|
|
onSelectQuality,
|
|
|
|
}: PropsType): JSX.Element => {
|
|
|
|
const [menuShowing, setMenuShowing] = useState(false);
|
|
|
|
const [popperRoot, setPopperRoot] = useState<HTMLElement | null>(null);
|
2021-07-07 17:05:03 +00:00
|
|
|
const [focusedOption, setFocusedOption] = useState<0 | 1 | undefined>(
|
|
|
|
undefined
|
|
|
|
);
|
2021-06-25 16:08:16 +00:00
|
|
|
|
|
|
|
// We use regular MouseEvent below, and this one uses React.MouseEvent
|
|
|
|
const handleClick = (ev: KeyboardEvent | React.MouseEvent) => {
|
|
|
|
setMenuShowing(true);
|
|
|
|
ev.stopPropagation();
|
|
|
|
ev.preventDefault();
|
|
|
|
};
|
|
|
|
|
2021-07-07 17:05:03 +00:00
|
|
|
const handleKeyDown = (ev: KeyboardEvent) => {
|
|
|
|
if (!popperRoot) {
|
|
|
|
if (ev.key === 'Enter') {
|
|
|
|
setFocusedOption(isHighQuality ? 1 : 0);
|
|
|
|
}
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (ev.key === 'ArrowDown' || ev.key === 'ArrowUp') {
|
|
|
|
setFocusedOption(oldFocusedOption => (oldFocusedOption === 1 ? 0 : 1));
|
|
|
|
ev.stopPropagation();
|
|
|
|
ev.preventDefault();
|
|
|
|
}
|
|
|
|
|
|
|
|
if (ev.key === 'Enter') {
|
|
|
|
onSelectQuality(Boolean(focusedOption));
|
|
|
|
setMenuShowing(false);
|
|
|
|
ev.stopPropagation();
|
|
|
|
ev.preventDefault();
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
2021-06-25 16:08:16 +00:00
|
|
|
const handleClose = useCallback(() => {
|
|
|
|
setMenuShowing(false);
|
2021-07-07 17:05:03 +00:00
|
|
|
setFocusedOption(undefined);
|
2021-06-25 16:08:16 +00:00
|
|
|
}, [setMenuShowing]);
|
|
|
|
|
|
|
|
useEffect(() => {
|
|
|
|
if (menuShowing) {
|
|
|
|
const root = document.createElement('div');
|
|
|
|
setPopperRoot(root);
|
|
|
|
document.body.appendChild(root);
|
|
|
|
const handleOutsideClick = (event: MouseEvent) => {
|
|
|
|
if (!root.contains(event.target as Node)) {
|
|
|
|
handleClose();
|
|
|
|
event.stopPropagation();
|
|
|
|
event.preventDefault();
|
|
|
|
}
|
|
|
|
};
|
|
|
|
document.addEventListener('click', handleOutsideClick);
|
|
|
|
|
|
|
|
return () => {
|
|
|
|
document.body.removeChild(root);
|
|
|
|
document.removeEventListener('click', handleOutsideClick);
|
|
|
|
setPopperRoot(null);
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
|
|
|
return noop;
|
|
|
|
}, [menuShowing, setPopperRoot, handleClose]);
|
|
|
|
|
|
|
|
return (
|
|
|
|
<Manager>
|
|
|
|
<Reference>
|
|
|
|
{({ ref }) => (
|
|
|
|
<button
|
|
|
|
aria-label={i18n('MediaQualitySelector--button')}
|
|
|
|
className={classNames({
|
|
|
|
MediaQualitySelector__button: true,
|
|
|
|
'MediaQualitySelector__button--hq': isHighQuality,
|
|
|
|
'MediaQualitySelector__button--active': menuShowing,
|
|
|
|
})}
|
|
|
|
onClick={handleClick}
|
2021-07-07 17:05:03 +00:00
|
|
|
onKeyDown={handleKeyDown}
|
2021-06-25 16:08:16 +00:00
|
|
|
ref={ref}
|
|
|
|
type="button"
|
|
|
|
/>
|
|
|
|
)}
|
|
|
|
</Reference>
|
|
|
|
{menuShowing && popperRoot
|
|
|
|
? createPortal(
|
2021-08-03 17:04:49 +00:00
|
|
|
<Popper placement="top-start" strategy="fixed">
|
2021-06-25 16:08:16 +00:00
|
|
|
{({ ref, style, placement }) => (
|
|
|
|
<div
|
|
|
|
className="MediaQualitySelector__popper"
|
|
|
|
data-placement={placement}
|
|
|
|
ref={ref}
|
|
|
|
style={style}
|
|
|
|
>
|
|
|
|
<div className="MediaQualitySelector__title">
|
|
|
|
{i18n('MediaQualitySelector--title')}
|
|
|
|
</div>
|
|
|
|
<button
|
|
|
|
aria-label={i18n(
|
|
|
|
'MediaQualitySelector--standard-quality-title'
|
|
|
|
)}
|
2021-07-07 17:05:03 +00:00
|
|
|
className={classNames({
|
|
|
|
MediaQualitySelector__option: true,
|
|
|
|
'MediaQualitySelector__option--focused':
|
|
|
|
focusedOption === 0,
|
|
|
|
})}
|
2021-06-25 16:08:16 +00:00
|
|
|
type="button"
|
|
|
|
onClick={() => {
|
|
|
|
onSelectQuality(false);
|
|
|
|
setMenuShowing(false);
|
|
|
|
}}
|
|
|
|
>
|
|
|
|
<div
|
|
|
|
className={classNames({
|
|
|
|
'MediaQualitySelector__option--checkmark': true,
|
|
|
|
'MediaQualitySelector__option--selected': !isHighQuality,
|
|
|
|
})}
|
|
|
|
/>
|
|
|
|
<div>
|
|
|
|
<div className="MediaQualitySelector__option--title">
|
|
|
|
{i18n('MediaQualitySelector--standard-quality-title')}
|
|
|
|
</div>
|
|
|
|
<div className="MediaQualitySelector__option--description">
|
|
|
|
{i18n(
|
|
|
|
'MediaQualitySelector--standard-quality-description'
|
|
|
|
)}
|
|
|
|
</div>
|
|
|
|
</div>
|
|
|
|
</button>
|
|
|
|
<button
|
|
|
|
aria-label={i18n(
|
|
|
|
'MediaQualitySelector--high-quality-title'
|
|
|
|
)}
|
2021-07-07 17:05:03 +00:00
|
|
|
className={classNames({
|
|
|
|
MediaQualitySelector__option: true,
|
|
|
|
'MediaQualitySelector__option--focused':
|
|
|
|
focusedOption === 1,
|
|
|
|
})}
|
2021-06-25 16:08:16 +00:00
|
|
|
type="button"
|
|
|
|
onClick={() => {
|
|
|
|
onSelectQuality(true);
|
|
|
|
setMenuShowing(false);
|
|
|
|
}}
|
|
|
|
>
|
|
|
|
<div
|
|
|
|
className={classNames({
|
|
|
|
'MediaQualitySelector__option--checkmark': true,
|
|
|
|
'MediaQualitySelector__option--selected': isHighQuality,
|
|
|
|
})}
|
|
|
|
/>
|
|
|
|
<div>
|
|
|
|
<div className="MediaQualitySelector__option--title">
|
|
|
|
{i18n('MediaQualitySelector--high-quality-title')}
|
|
|
|
</div>
|
|
|
|
<div className="MediaQualitySelector__option--description">
|
|
|
|
{i18n('MediaQualitySelector--high-quality-description')}
|
|
|
|
</div>
|
|
|
|
</div>
|
|
|
|
</button>
|
|
|
|
</div>
|
|
|
|
)}
|
|
|
|
</Popper>,
|
|
|
|
popperRoot
|
|
|
|
)
|
|
|
|
: null}
|
|
|
|
</Manager>
|
|
|
|
);
|
|
|
|
};
|