Co-authored-by: zubiden <19638254+zubiden@users.noreply.github.com> Co-authored-by: Dmitry Kabanov <dmitrykabanovdev@gmail.com>
406 lines
13 KiB
TypeScript
406 lines
13 KiB
TypeScript
import type { ChangeEvent } from 'react';
|
|
import type { FC } from '../../../lib/teact/teact';
|
|
import {
|
|
memo, useEffect, useMemo, useState,
|
|
} from '../../../lib/teact/teact';
|
|
import { getActions, withGlobal } from '../../../global';
|
|
|
|
import type {
|
|
ApiAvailableReaction, ApiChat, ApiChatFullInfo, ApiExportedInvite,
|
|
} from '../../../api/types';
|
|
import { ApiMediaFormat } from '../../../api/types';
|
|
import { ManagementProgress, ManagementScreens } from '../../../types';
|
|
|
|
import { getChatAvatarHash, getHasAdminRight, isChatChannel, isChatPublic } from '../../../global/helpers';
|
|
import { selectChat, selectChatFullInfo, selectTabState } from '../../../global/selectors';
|
|
import { formatInteger } from '../../../util/textFormat';
|
|
|
|
import useFlag from '../../../hooks/useFlag';
|
|
import useHistoryBack from '../../../hooks/useHistoryBack';
|
|
import useLastCallback from '../../../hooks/useLastCallback';
|
|
import useMedia from '../../../hooks/useMedia';
|
|
import useOldLang from '../../../hooks/useOldLang';
|
|
|
|
import Icon from '../../common/icons/Icon';
|
|
import AvatarEditable from '../../ui/AvatarEditable';
|
|
import ConfirmDialog from '../../ui/ConfirmDialog';
|
|
import FloatingActionButton from '../../ui/FloatingActionButton';
|
|
import InputText from '../../ui/InputText';
|
|
import ListItem from '../../ui/ListItem';
|
|
import Spinner from '../../ui/Spinner';
|
|
import Switcher from '../../ui/Switcher';
|
|
import TextArea from '../../ui/TextArea';
|
|
|
|
import './Management.scss';
|
|
|
|
type OwnProps = {
|
|
chatId: string;
|
|
onScreenSelect: (screen: ManagementScreens) => void;
|
|
onClose: NoneToVoidFunction;
|
|
isActive: boolean;
|
|
};
|
|
|
|
type StateProps = {
|
|
chat: ApiChat;
|
|
chatFullInfo?: ApiChatFullInfo;
|
|
progress?: ManagementProgress;
|
|
canChangeInfo?: boolean;
|
|
canInvite?: boolean;
|
|
exportedInvites?: ApiExportedInvite[];
|
|
availableReactions?: ApiAvailableReaction[];
|
|
hasAutoTranslation?: boolean;
|
|
canToggleAutoTranslation?: boolean;
|
|
};
|
|
|
|
const CHANNEL_TITLE_EMPTY = 'Channel title can\'t be empty';
|
|
const CHANNEL_MAX_DESCRIPTION = 255;
|
|
|
|
const ManageChannel: FC<OwnProps & StateProps> = ({
|
|
chatId,
|
|
chat,
|
|
chatFullInfo,
|
|
progress,
|
|
canChangeInfo,
|
|
canInvite,
|
|
exportedInvites,
|
|
isActive,
|
|
availableReactions,
|
|
onScreenSelect,
|
|
onClose,
|
|
hasAutoTranslation,
|
|
canToggleAutoTranslation,
|
|
}) => {
|
|
const {
|
|
updateChat,
|
|
closeManagement,
|
|
leaveChannel,
|
|
deleteChannel,
|
|
openChat,
|
|
loadExportedChatInvites,
|
|
loadChatJoinRequests,
|
|
toggleAutoTranslation,
|
|
} = getActions();
|
|
|
|
const currentTitle = chat?.title || '';
|
|
const currentAbout = chatFullInfo?.about || '';
|
|
const hasLinkedChat = Boolean(chatFullInfo?.linkedChatId);
|
|
|
|
const [isDeleteDialogOpen, openDeleteDialog, closeDeleteDialog] = useFlag();
|
|
const [isProfileFieldsTouched, setIsProfileFieldsTouched] = useState(false);
|
|
const [title, setTitle] = useState(currentTitle);
|
|
const [about, setAbout] = useState(currentAbout);
|
|
const [photo, setPhoto] = useState<File | undefined>();
|
|
const [error, setError] = useState<string | undefined>();
|
|
const imageHash = chat && getChatAvatarHash(chat);
|
|
const currentAvatarBlobUrl = useMedia(imageHash, false, ApiMediaFormat.BlobUrl);
|
|
const lang = useOldLang();
|
|
|
|
const hasAutoTranslationAvailable = chat && isChatChannel(chat);
|
|
|
|
const handleAutoTranslationChange = useLastCallback(() => {
|
|
toggleAutoTranslation({ chatId, isEnabled: !hasAutoTranslation });
|
|
});
|
|
|
|
useHistoryBack({
|
|
isActive,
|
|
onBack: onClose,
|
|
});
|
|
|
|
useEffect(() => {
|
|
if (!canInvite) return;
|
|
loadExportedChatInvites({ chatId });
|
|
loadExportedChatInvites({ chatId, isRevoked: true });
|
|
loadChatJoinRequests({ chatId });
|
|
}, [chatId, canInvite]);
|
|
|
|
useEffect(() => {
|
|
if (progress === ManagementProgress.Complete) {
|
|
setIsProfileFieldsTouched(false);
|
|
setError(undefined);
|
|
}
|
|
}, [progress]);
|
|
|
|
const adminsCount = useMemo(() => {
|
|
return Object.keys(chatFullInfo?.adminMembersById || {}).length;
|
|
}, [chatFullInfo?.adminMembersById]);
|
|
const removedUsersCount = chatFullInfo?.kickedMembers?.length || 0;
|
|
|
|
const handleClickEditType = useLastCallback(() => {
|
|
onScreenSelect(ManagementScreens.ChatPrivacyType);
|
|
});
|
|
|
|
const handleClickDiscussion = useLastCallback(() => {
|
|
onScreenSelect(ManagementScreens.Discussion);
|
|
});
|
|
|
|
const handleClickReactions = useLastCallback(() => {
|
|
onScreenSelect(ManagementScreens.Reactions);
|
|
});
|
|
|
|
const handleClickAdministrators = useLastCallback(() => {
|
|
onScreenSelect(ManagementScreens.ChatAdministrators);
|
|
});
|
|
|
|
const handleClickInvites = useLastCallback(() => {
|
|
onScreenSelect(ManagementScreens.Invites);
|
|
});
|
|
|
|
const handleClickRequests = useLastCallback(() => {
|
|
onScreenSelect(ManagementScreens.JoinRequests);
|
|
});
|
|
|
|
const handleSetPhoto = useLastCallback((file: File) => {
|
|
setPhoto(file);
|
|
setIsProfileFieldsTouched(true);
|
|
});
|
|
|
|
const handleTitleChange = useLastCallback((e: ChangeEvent<HTMLInputElement>) => {
|
|
setTitle(e.target.value);
|
|
setIsProfileFieldsTouched(true);
|
|
});
|
|
|
|
const handleAboutChange = useLastCallback((e: ChangeEvent<HTMLTextAreaElement>) => {
|
|
setAbout(e.target.value);
|
|
setIsProfileFieldsTouched(true);
|
|
});
|
|
|
|
const handleUpdateChannel = useLastCallback(() => {
|
|
const trimmedTitle = title.trim();
|
|
const trimmedAbout = about.trim();
|
|
|
|
if (!trimmedTitle.length) {
|
|
setError(CHANNEL_TITLE_EMPTY);
|
|
return;
|
|
}
|
|
|
|
updateChat({
|
|
chatId,
|
|
title: trimmedTitle,
|
|
about: trimmedAbout,
|
|
photo,
|
|
});
|
|
});
|
|
|
|
const handleClickSubscribers = useLastCallback(() => {
|
|
onScreenSelect(ManagementScreens.ChannelSubscribers);
|
|
});
|
|
|
|
const handleRemovedUsersClick = useLastCallback(() => {
|
|
onScreenSelect(ManagementScreens.ChannelRemovedUsers);
|
|
});
|
|
|
|
const handleDeleteChannel = useLastCallback(() => {
|
|
if (chat.isCreator) {
|
|
deleteChannel({ chatId: chat.id });
|
|
} else {
|
|
leaveChannel({ chatId: chat.id });
|
|
}
|
|
|
|
closeDeleteDialog();
|
|
closeManagement();
|
|
openChat({ id: undefined });
|
|
});
|
|
|
|
const chatReactionsDescription = useMemo(() => {
|
|
if (!chatFullInfo?.enabledReactions) {
|
|
return lang('ReactionsOff');
|
|
}
|
|
|
|
if (chatFullInfo.enabledReactions.type === 'all') {
|
|
return lang('ReactionsAll');
|
|
}
|
|
|
|
const enabledLength = chatFullInfo.enabledReactions.allowed.length;
|
|
const totalLength = availableReactions?.filter((reaction) => !reaction.isInactive).length || 0;
|
|
|
|
return totalLength ? `${enabledLength} / ${totalLength}` : enabledLength.toString();
|
|
}, [availableReactions, chatFullInfo?.enabledReactions, lang]);
|
|
const isChannelPublic = useMemo(() => isChatPublic(chat), [chat]);
|
|
|
|
if (chat.isRestricted || chat.isForbidden) {
|
|
return undefined;
|
|
}
|
|
|
|
const isLoading = progress === ManagementProgress.InProgress;
|
|
|
|
return (
|
|
<div className="Management">
|
|
<div className="panel-content custom-scroll">
|
|
<div className="section">
|
|
<AvatarEditable
|
|
currentAvatarBlobUrl={currentAvatarBlobUrl}
|
|
onChange={handleSetPhoto}
|
|
disabled={!canChangeInfo}
|
|
/>
|
|
<div className="settings-edit">
|
|
<InputText
|
|
id="channel-title"
|
|
label={lang('EnterChannelName')}
|
|
onChange={handleTitleChange}
|
|
value={title}
|
|
error={error === CHANNEL_TITLE_EMPTY ? error : undefined}
|
|
disabled={!canChangeInfo}
|
|
/>
|
|
<TextArea
|
|
id="channel-about"
|
|
label={lang('DescriptionPlaceholder')}
|
|
onChange={handleAboutChange}
|
|
value={about}
|
|
maxLength={CHANNEL_MAX_DESCRIPTION}
|
|
maxLengthIndicator={(CHANNEL_MAX_DESCRIPTION - about.length).toString()}
|
|
disabled={!canChangeInfo}
|
|
noReplaceNewlines
|
|
/>
|
|
</div>
|
|
{chat.isCreator && (
|
|
<ListItem icon="lock" multiline onClick={handleClickEditType}>
|
|
<span className="title">{lang('ChannelType')}</span>
|
|
<span className="subtitle">{isChannelPublic ? lang('TypePublic') : lang('TypePrivate')}</span>
|
|
</ListItem>
|
|
)}
|
|
<ListItem
|
|
icon="message"
|
|
multiline
|
|
onClick={handleClickDiscussion}
|
|
disabled={!canChangeInfo}
|
|
>
|
|
<span className="title">{lang('Discussion')}</span>
|
|
<span className="subtitle">{hasLinkedChat ? lang('DiscussionUnlink') : lang('Add')}</span>
|
|
</ListItem>
|
|
{canInvite && (
|
|
<ListItem
|
|
icon="link"
|
|
onClick={handleClickInvites}
|
|
multiline
|
|
disabled={!exportedInvites}
|
|
>
|
|
<span className="title">{lang('GroupInfo.InviteLinks')}</span>
|
|
<span className="subtitle">
|
|
{exportedInvites ? formatInteger(exportedInvites.length) : lang('Loading')}
|
|
</span>
|
|
</ListItem>
|
|
)}
|
|
{Boolean(chat.joinRequests?.length) && (
|
|
<ListItem
|
|
icon="add-user-filled"
|
|
onClick={handleClickRequests}
|
|
multiline
|
|
>
|
|
<span className="title">{lang('SubscribeRequests')}</span>
|
|
<span className="subtitle">
|
|
{formatInteger(chat.joinRequests.length)}
|
|
</span>
|
|
</ListItem>
|
|
)}
|
|
<ListItem
|
|
icon="heart-outline"
|
|
multiline
|
|
onClick={handleClickReactions}
|
|
disabled={!canChangeInfo}
|
|
>
|
|
<span className="title">{lang('Reactions')}</span>
|
|
<span className="subtitle" dir="auto">
|
|
{chatReactionsDescription}
|
|
</span>
|
|
</ListItem>
|
|
{hasAutoTranslationAvailable && (
|
|
<ListItem
|
|
icon="language"
|
|
narrow
|
|
ripple
|
|
disabled={!canToggleAutoTranslation}
|
|
onClick={handleAutoTranslationChange}
|
|
>
|
|
<span>{lang('AutomaticTranslation')}</span>
|
|
<Switcher
|
|
id="auto-translation"
|
|
label={lang('AutomaticTranslation')}
|
|
checked={hasAutoTranslation}
|
|
/>
|
|
</ListItem>
|
|
)}
|
|
</div>
|
|
<div className="section">
|
|
<ListItem
|
|
icon="admin"
|
|
multiline
|
|
onClick={handleClickAdministrators}
|
|
>
|
|
<span className="title">{lang('ChannelAdministrators')}</span>
|
|
<span className="subtitle">{adminsCount}</span>
|
|
</ListItem>
|
|
<ListItem
|
|
icon="group"
|
|
multiline
|
|
onClick={handleClickSubscribers}
|
|
>
|
|
<span className="title" dir="auto">{lang('ChannelSubscribers')}</span>
|
|
<span className="subtitle" dir="auto">{formatInteger(chat.membersCount!)}</span>
|
|
</ListItem>
|
|
<ListItem
|
|
icon="delete-user"
|
|
multiline
|
|
onClick={handleRemovedUsersClick}
|
|
>
|
|
<span className="title">{lang('ChannelBlockedUsers')}</span>
|
|
<span className="subtitle">{removedUsersCount}</span>
|
|
</ListItem>
|
|
</div>
|
|
<div className="section">
|
|
<ListItem icon="delete" ripple destructive onClick={openDeleteDialog}>
|
|
{chat.isCreator ? lang('ChannelDelete') : lang('LeaveChannel')}
|
|
</ListItem>
|
|
</div>
|
|
</div>
|
|
<FloatingActionButton
|
|
isShown={isProfileFieldsTouched}
|
|
onClick={handleUpdateChannel}
|
|
disabled={isLoading}
|
|
ariaLabel={lang('Save')}
|
|
>
|
|
{isLoading ? (
|
|
<Spinner color="white" />
|
|
) : (
|
|
<Icon name="check" />
|
|
)}
|
|
</FloatingActionButton>
|
|
<ConfirmDialog
|
|
isOpen={isDeleteDialogOpen}
|
|
onClose={closeDeleteDialog}
|
|
text={chat.isCreator ? lang('ChannelDeleteAlert') : lang('ChannelLeaveAlert')}
|
|
confirmLabel={chat.isCreator ? lang('ChannelDelete') : lang('LeaveChannel')}
|
|
confirmHandler={handleDeleteChannel}
|
|
confirmIsDestructive
|
|
/>
|
|
</div>
|
|
);
|
|
};
|
|
|
|
export default memo(withGlobal<OwnProps>(
|
|
(global, { chatId }): StateProps => {
|
|
const chat = selectChat(global, chatId)!;
|
|
const { management } = selectTabState(global);
|
|
const { progress } = management;
|
|
const { invites } = management.byChatId[chatId] || {};
|
|
const minLevelToToggleAutoTranslation = global.appConfig?.channelAutoTranslationLevelMin;
|
|
const hasAutoTranslation = chat?.hasAutoTranslation;
|
|
const chatBoostLevel = chat?.level;
|
|
const canToggleAutoTranslation = chatBoostLevel && minLevelToToggleAutoTranslation
|
|
? chatBoostLevel >= minLevelToToggleAutoTranslation : false;
|
|
return {
|
|
chat,
|
|
chatFullInfo: selectChatFullInfo(global, chatId),
|
|
progress,
|
|
canChangeInfo: getHasAdminRight(chat, 'changeInfo'),
|
|
canInvite: getHasAdminRight(chat, 'inviteUsers'),
|
|
exportedInvites: invites,
|
|
availableReactions: global.reactions.availableReactions,
|
|
hasAutoTranslation,
|
|
canToggleAutoTranslation,
|
|
};
|
|
},
|
|
(global, { chatId }) => {
|
|
return Boolean(selectChat(global, chatId));
|
|
},
|
|
)(ManageChannel));
|