Alexander Zinchuk 9b82953426 Profile: Support auto translation in channels (#5891)
Co-authored-by: zubiden <19638254+zubiden@users.noreply.github.com>
Co-authored-by: Dmitry Kabanov <dmitrykabanovdev@gmail.com>
2025-06-18 17:40:58 +02:00

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));