signal-desktop/ts/components/StoriesAddStoryButton.tsx

140 lines
3.6 KiB
TypeScript
Raw Normal View History

2022-08-19 18:36:47 +00:00
// Copyright 2022 Signal Messenger, LLC
// SPDX-License-Identifier: AGPL-3.0-only
import type { ReactNode } from 'react';
import React, { useState, useCallback } from 'react';
2022-08-19 18:36:47 +00:00
import type { LocalizerType } from '../types/Util';
import type { ShowToastActionCreatorType } from '../state/ducks/toast';
import { ContextMenu } from './ContextMenu';
import { Theme } from '../util/theme';
import { ToastType } from '../types/Toast';
2022-08-19 18:36:47 +00:00
import {
isVideoGoodForStories,
ReasonVideoNotGood,
} from '../util/isVideoGoodForStories';
import { ConfirmationDialog } from './ConfirmationDialog';
2022-08-19 18:36:47 +00:00
export type PropsType = {
children?: ReactNode;
i18n: LocalizerType;
maxAttachmentSizeInKb: number;
2022-08-19 18:36:47 +00:00
moduleClassName?: string;
onAddStory: (file?: File) => unknown;
onContextMenuShowingChanged?: (value: boolean) => void;
2022-08-19 18:36:47 +00:00
showToast: ShowToastActionCreatorType;
};
2022-11-18 00:45:19 +00:00
export function StoriesAddStoryButton({
2022-08-19 18:36:47 +00:00
children,
i18n,
maxAttachmentSizeInKb,
2022-08-19 18:36:47 +00:00
moduleClassName,
onAddStory,
showToast,
onContextMenuShowingChanged,
2022-11-18 00:45:19 +00:00
}: PropsType): JSX.Element {
const [error, setError] = useState<string | undefined>();
const onAddMedia = useCallback(() => {
const input = document.createElement('input');
input.accept = 'image/*,video/mp4';
input.type = 'file';
input.onchange = async () => {
const file = input.files ? input.files[0] : undefined;
if (!file) {
return;
}
2022-08-19 18:36:47 +00:00
const result = await isVideoGoodForStories(file, {
maxAttachmentSizeInKb,
});
2022-08-19 18:36:47 +00:00
if (
result.reason === ReasonVideoNotGood.UnsupportedCodec ||
result.reason === ReasonVideoNotGood.UnsupportedContainer
) {
showToast(ToastType.StoryVideoUnsupported);
return;
}
2022-08-19 18:36:47 +00:00
if (result.reason === ReasonVideoNotGood.TooLong) {
setError(
i18n('icu:StoryCreator__error--video-too-long', {
maxDurationInSec: result.maxDurationInSec,
})
);
return;
}
2022-08-19 18:36:47 +00:00
if (result.reason === ReasonVideoNotGood.TooBig) {
setError(
i18n('icu:StoryCreator__error--video-too-big', {
limit: result.renderDetails.limit,
units: result.renderDetails.units,
})
);
return;
}
2022-08-19 18:36:47 +00:00
if (result.reason !== ReasonVideoNotGood.AllGoodNevermind) {
showToast(ToastType.StoryVideoError);
return;
}
2022-08-19 18:36:47 +00:00
onAddStory(file);
};
input.click();
}, [setError, showToast, i18n, maxAttachmentSizeInKb, onAddStory]);
return (
<>
<ContextMenu
2023-03-30 00:03:25 +00:00
ariaLabel={i18n('icu:Stories__add')}
i18n={i18n}
onMenuShowingChanged={onContextMenuShowingChanged}
menuOptions={[
{
2023-03-30 00:03:25 +00:00
label: i18n('icu:Stories__add-story--media'),
onClick: onAddMedia,
},
{
2023-03-30 00:03:25 +00:00
label: i18n('icu:Stories__add-story--text'),
onClick: () => onAddStory(),
2022-08-19 18:36:47 +00:00
},
]}
moduleClassName={moduleClassName}
popperOptions={{
placement: 'bottom',
strategy: 'absolute',
}}
theme={Theme.Dark}
>
{children}
</ContextMenu>
{error && (
<ConfirmationDialog
dialogName="StoriesAddStoryButton.error"
noDefaultCancelButton
actions={[
{
action: () => {
setError(undefined);
},
style: 'affirmative',
2023-03-30 00:03:25 +00:00
text: i18n('icu:Confirmation--confirm'),
},
]}
i18n={i18n}
onClose={() => {
setError(undefined);
}}
>
{error}
</ConfirmationDialog>
)}
</>
2022-08-19 18:36:47 +00:00
);
2022-11-18 00:45:19 +00:00
}