pl-fe: migrate status translations to tanstack query
Signed-off-by: mkljczk <git@mkljczk.pl>
This commit is contained in:
parent
a99c61d454
commit
f0542c2a62
8 changed files with 134 additions and 145 deletions
|
@ -9,7 +9,7 @@ import { setComposeToStatus } from './compose';
|
||||||
import { importEntities } from './importer';
|
import { importEntities } from './importer';
|
||||||
import { deleteFromTimelines } from './timelines';
|
import { deleteFromTimelines } from './timelines';
|
||||||
|
|
||||||
import type { CreateStatusParams, Status as BaseStatus, ScheduledStatus, Translation } from 'pl-api';
|
import type { CreateStatusParams, Status as BaseStatus, ScheduledStatus } from 'pl-api';
|
||||||
import type { Status } from 'pl-fe/normalizers/status';
|
import type { Status } from 'pl-fe/normalizers/status';
|
||||||
import type { AppDispatch, RootState } from 'pl-fe/store';
|
import type { AppDispatch, RootState } from 'pl-fe/store';
|
||||||
import type { IntlShape } from 'react-intl';
|
import type { IntlShape } from 'react-intl';
|
||||||
|
@ -48,11 +48,6 @@ const STATUS_HIDE_MEDIA = 'STATUS_HIDE_MEDIA' as const;
|
||||||
const STATUS_EXPAND_SPOILER = 'STATUS_EXPAND_SPOILER' as const;
|
const STATUS_EXPAND_SPOILER = 'STATUS_EXPAND_SPOILER' as const;
|
||||||
const STATUS_COLLAPSE_SPOILER = 'STATUS_COLLAPSE_SPOILER' as const;
|
const STATUS_COLLAPSE_SPOILER = 'STATUS_COLLAPSE_SPOILER' as const;
|
||||||
|
|
||||||
const STATUS_TRANSLATE_REQUEST = 'STATUS_TRANSLATE_REQUEST' as const;
|
|
||||||
const STATUS_TRANSLATE_SUCCESS = 'STATUS_TRANSLATE_SUCCESS' as const;
|
|
||||||
const STATUS_TRANSLATE_FAIL = 'STATUS_TRANSLATE_FAIL' as const;
|
|
||||||
const STATUS_TRANSLATE_UNDO = 'STATUS_TRANSLATE_UNDO' as const;
|
|
||||||
|
|
||||||
const STATUS_UNFILTER = 'STATUS_UNFILTER' as const;
|
const STATUS_UNFILTER = 'STATUS_UNFILTER' as const;
|
||||||
|
|
||||||
const STATUS_LANGUAGE_CHANGE = 'STATUS_LANGUAGE_CHANGE' as const;
|
const STATUS_LANGUAGE_CHANGE = 'STATUS_LANGUAGE_CHANGE' as const;
|
||||||
|
@ -269,75 +264,54 @@ const expandStatusSpoiler = (statusIds: string[] | string) => {
|
||||||
};
|
};
|
||||||
};
|
};
|
||||||
|
|
||||||
let TRANSLATIONS_QUEUE: Set<string> = new Set();
|
// let TRANSLATIONS_QUEUE: Set<string> = new Set();
|
||||||
let TRANSLATIONS_TIMEOUT: NodeJS.Timeout | null = null;
|
// let TRANSLATIONS_TIMEOUT: NodeJS.Timeout | null = null;
|
||||||
|
|
||||||
const translateStatus = (statusId: string, targetLanguage: string, lazy?: boolean) =>
|
// const translateStatus = (statusId: string, targetLanguage: string, lazy?: boolean) =>
|
||||||
(dispatch: AppDispatch, getState: () => RootState) => {
|
// (dispatch: AppDispatch, getState: () => RootState) => {
|
||||||
const client = getClient(getState);
|
// const client = getClient(getState);
|
||||||
const features = client.features;
|
// const features = client.features;
|
||||||
|
|
||||||
dispatch<StatusesAction>({ type: STATUS_TRANSLATE_REQUEST, statusId });
|
// const handleTranslateMany = () => {
|
||||||
|
// const copy = [...TRANSLATIONS_QUEUE];
|
||||||
|
// TRANSLATIONS_QUEUE = new Set();
|
||||||
|
// if (TRANSLATIONS_TIMEOUT) clearTimeout(TRANSLATIONS_TIMEOUT);
|
||||||
|
|
||||||
const handleTranslateMany = () => {
|
// return client.statuses.translateStatuses(copy, targetLanguage).then((response) => {
|
||||||
const copy = [...TRANSLATIONS_QUEUE];
|
// response.forEach((translation) => {
|
||||||
TRANSLATIONS_QUEUE = new Set();
|
// dispatch<StatusesAction>({
|
||||||
if (TRANSLATIONS_TIMEOUT) clearTimeout(TRANSLATIONS_TIMEOUT);
|
// type: STATUS_TRANSLATE_SUCCESS,
|
||||||
|
// statusId: translation.id,
|
||||||
|
// translation: translation,
|
||||||
|
// });
|
||||||
|
|
||||||
return client.statuses.translateStatuses(copy, targetLanguage).then((response) => {
|
// copy
|
||||||
response.forEach((translation) => {
|
// .filter((statusId) => !response.some(({ id }) => id === statusId))
|
||||||
dispatch<StatusesAction>({
|
// .forEach((statusId) => dispatch<StatusesAction>({
|
||||||
type: STATUS_TRANSLATE_SUCCESS,
|
// type: STATUS_TRANSLATE_FAIL,
|
||||||
statusId: translation.id,
|
// statusId,
|
||||||
translation: translation,
|
// }));
|
||||||
});
|
// });
|
||||||
|
// }).catch(error => {
|
||||||
|
// dispatch<StatusesAction>({
|
||||||
|
// type: STATUS_TRANSLATE_FAIL,
|
||||||
|
// statusId,
|
||||||
|
// error,
|
||||||
|
// });
|
||||||
|
// });
|
||||||
|
// };
|
||||||
|
|
||||||
copy
|
// if (features.lazyTranslations && lazy) {
|
||||||
.filter((statusId) => !response.some(({ id }) => id === statusId))
|
// TRANSLATIONS_QUEUE.add(statusId);
|
||||||
.forEach((statusId) => dispatch<StatusesAction>({
|
|
||||||
type: STATUS_TRANSLATE_FAIL,
|
|
||||||
statusId,
|
|
||||||
}));
|
|
||||||
});
|
|
||||||
}).catch(error => {
|
|
||||||
dispatch<StatusesAction>({
|
|
||||||
type: STATUS_TRANSLATE_FAIL,
|
|
||||||
statusId,
|
|
||||||
error,
|
|
||||||
});
|
|
||||||
});
|
|
||||||
};
|
|
||||||
|
|
||||||
if (features.lazyTranslations && lazy) {
|
// if (TRANSLATIONS_TIMEOUT) clearTimeout(TRANSLATIONS_TIMEOUT);
|
||||||
TRANSLATIONS_QUEUE.add(statusId);
|
// TRANSLATIONS_TIMEOUT = setTimeout(() => handleTranslateMany(), 3000);
|
||||||
|
// } else if (features.lazyTranslations && TRANSLATIONS_QUEUE.size) {
|
||||||
|
// TRANSLATIONS_QUEUE.add(statusId);
|
||||||
|
|
||||||
if (TRANSLATIONS_TIMEOUT) clearTimeout(TRANSLATIONS_TIMEOUT);
|
// handleTranslateMany();
|
||||||
TRANSLATIONS_TIMEOUT = setTimeout(() => handleTranslateMany(), 3000);
|
// }
|
||||||
} else if (features.lazyTranslations && TRANSLATIONS_QUEUE.size) {
|
// };
|
||||||
TRANSLATIONS_QUEUE.add(statusId);
|
|
||||||
|
|
||||||
handleTranslateMany();
|
|
||||||
} else {
|
|
||||||
return client.statuses.translateStatus(statusId, targetLanguage).then(response => {
|
|
||||||
dispatch<StatusesAction>({
|
|
||||||
type: STATUS_TRANSLATE_SUCCESS,
|
|
||||||
statusId,
|
|
||||||
translation: response,
|
|
||||||
});
|
|
||||||
}).catch(error => {
|
|
||||||
dispatch<StatusesAction>({
|
|
||||||
type: STATUS_TRANSLATE_FAIL,
|
|
||||||
statusId,
|
|
||||||
error,
|
|
||||||
});
|
|
||||||
});
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
const undoStatusTranslation = (statusId: string) => ({
|
|
||||||
type: STATUS_TRANSLATE_UNDO,
|
|
||||||
statusId,
|
|
||||||
});
|
|
||||||
|
|
||||||
const unfilterStatus = (statusId: string) => ({
|
const unfilterStatus = (statusId: string) => ({
|
||||||
type: STATUS_UNFILTER,
|
type: STATUS_UNFILTER,
|
||||||
|
@ -376,10 +350,6 @@ type StatusesAction =
|
||||||
| ReturnType<typeof revealStatusMedia>
|
| ReturnType<typeof revealStatusMedia>
|
||||||
| ReturnType<typeof collapseStatusSpoiler>
|
| ReturnType<typeof collapseStatusSpoiler>
|
||||||
| ReturnType<typeof expandStatusSpoiler>
|
| ReturnType<typeof expandStatusSpoiler>
|
||||||
| { type: typeof STATUS_TRANSLATE_REQUEST; statusId: string }
|
|
||||||
| { type: typeof STATUS_TRANSLATE_SUCCESS; statusId: string | null; translation: Translation }
|
|
||||||
| { type: typeof STATUS_TRANSLATE_FAIL; statusId: string; error?: unknown }
|
|
||||||
| ReturnType<typeof undoStatusTranslation>
|
|
||||||
| ReturnType<typeof unfilterStatus>
|
| ReturnType<typeof unfilterStatus>
|
||||||
| ReturnType<typeof changeStatusLanguage>;
|
| ReturnType<typeof changeStatusLanguage>;
|
||||||
|
|
||||||
|
@ -409,10 +379,6 @@ export {
|
||||||
STATUS_HIDE_MEDIA,
|
STATUS_HIDE_MEDIA,
|
||||||
STATUS_EXPAND_SPOILER,
|
STATUS_EXPAND_SPOILER,
|
||||||
STATUS_COLLAPSE_SPOILER,
|
STATUS_COLLAPSE_SPOILER,
|
||||||
STATUS_TRANSLATE_REQUEST,
|
|
||||||
STATUS_TRANSLATE_SUCCESS,
|
|
||||||
STATUS_TRANSLATE_FAIL,
|
|
||||||
STATUS_TRANSLATE_UNDO,
|
|
||||||
STATUS_UNFILTER,
|
STATUS_UNFILTER,
|
||||||
STATUS_LANGUAGE_CHANGE,
|
STATUS_LANGUAGE_CHANGE,
|
||||||
createStatus,
|
createStatus,
|
||||||
|
@ -430,8 +396,6 @@ export {
|
||||||
toggleStatusMediaHidden,
|
toggleStatusMediaHidden,
|
||||||
expandStatusSpoiler,
|
expandStatusSpoiler,
|
||||||
collapseStatusSpoiler,
|
collapseStatusSpoiler,
|
||||||
translateStatus,
|
|
||||||
undoStatusTranslation,
|
|
||||||
unfilterStatus,
|
unfilterStatus,
|
||||||
changeStatusLanguage,
|
changeStatusLanguage,
|
||||||
type StatusesAction,
|
type StatusesAction,
|
||||||
|
|
|
@ -0,0 +1,18 @@
|
||||||
|
import { useQuery } from '@tanstack/react-query';
|
||||||
|
|
||||||
|
import { useClient } from 'pl-fe/hooks/use-client';
|
||||||
|
|
||||||
|
import type { Translation } from 'pl-api';
|
||||||
|
|
||||||
|
const useStatusTranslation = (statusId: string, targetLanguage?: string) => {
|
||||||
|
const client = useClient();
|
||||||
|
|
||||||
|
return useQuery<Translation | false>({
|
||||||
|
queryKey: ['statuses', 'translations', statusId, targetLanguage],
|
||||||
|
queryFn: () => client.statuses.translateStatus(statusId, targetLanguage)
|
||||||
|
.then(translation => translation).catch(() => false),
|
||||||
|
enabled: !!targetLanguage,
|
||||||
|
});
|
||||||
|
};
|
||||||
|
|
||||||
|
export { useStatusTranslation };
|
|
@ -11,7 +11,7 @@ import { toggleBookmark, toggleDislike, toggleFavourite, togglePin, toggleReblog
|
||||||
import { deleteStatusModal, toggleStatusSensitivityModal } from 'pl-fe/actions/moderation';
|
import { deleteStatusModal, toggleStatusSensitivityModal } from 'pl-fe/actions/moderation';
|
||||||
import { initReport, ReportableEntities } from 'pl-fe/actions/reports';
|
import { initReport, ReportableEntities } from 'pl-fe/actions/reports';
|
||||||
import { changeSetting } from 'pl-fe/actions/settings';
|
import { changeSetting } from 'pl-fe/actions/settings';
|
||||||
import { deleteStatus, editStatus, toggleMuteStatus, translateStatus, undoStatusTranslation } from 'pl-fe/actions/statuses';
|
import { deleteStatus, editStatus, toggleMuteStatus } from 'pl-fe/actions/statuses';
|
||||||
import { deleteFromTimelines } from 'pl-fe/actions/timelines';
|
import { deleteFromTimelines } from 'pl-fe/actions/timelines';
|
||||||
import { useBlockGroupMember } from 'pl-fe/api/hooks/groups/use-block-group-member';
|
import { useBlockGroupMember } from 'pl-fe/api/hooks/groups/use-block-group-member';
|
||||||
import { useDeleteGroupStatus } from 'pl-fe/api/hooks/groups/use-delete-group-status';
|
import { useDeleteGroupStatus } from 'pl-fe/api/hooks/groups/use-delete-group-status';
|
||||||
|
@ -33,6 +33,7 @@ import { useSettings } from 'pl-fe/hooks/use-settings';
|
||||||
import { useChats } from 'pl-fe/queries/chats';
|
import { useChats } from 'pl-fe/queries/chats';
|
||||||
import { RootState } from 'pl-fe/store';
|
import { RootState } from 'pl-fe/store';
|
||||||
import { useModalsStore } from 'pl-fe/stores/modals';
|
import { useModalsStore } from 'pl-fe/stores/modals';
|
||||||
|
import { useStatusMetaStore } from 'pl-fe/stores/status-meta';
|
||||||
import toast from 'pl-fe/toast';
|
import toast from 'pl-fe/toast';
|
||||||
import copy from 'pl-fe/utils/copy';
|
import copy from 'pl-fe/utils/copy';
|
||||||
|
|
||||||
|
@ -574,7 +575,6 @@ interface IMenuButton extends IActionButton {
|
||||||
const MenuButton: React.FC<IMenuButton> = ({
|
const MenuButton: React.FC<IMenuButton> = ({
|
||||||
status,
|
status,
|
||||||
statusActionButtonTheme,
|
statusActionButtonTheme,
|
||||||
withLabels,
|
|
||||||
me,
|
me,
|
||||||
expandable,
|
expandable,
|
||||||
fromBookmarks,
|
fromBookmarks,
|
||||||
|
@ -585,6 +585,8 @@ const MenuButton: React.FC<IMenuButton> = ({
|
||||||
const match = useRouteMatch<{ groupId: string }>('/groups/:groupId');
|
const match = useRouteMatch<{ groupId: string }>('/groups/:groupId');
|
||||||
const { boostModal } = useSettings();
|
const { boostModal } = useSettings();
|
||||||
|
|
||||||
|
const { statuses: statusesMeta, fetchTranslation, hideTranslation } = useStatusMetaStore();
|
||||||
|
const targetLanguage = statusesMeta[status.id]?.targetLanguage;
|
||||||
const { openModal } = useModalsStore();
|
const { openModal } = useModalsStore();
|
||||||
const { group } = useGroup((status.group as Group)?.id as string);
|
const { group } = useGroup((status.group as Group)?.id as string);
|
||||||
const deleteGroupStatus = useDeleteGroupStatus(group as Group, status.id);
|
const deleteGroupStatus = useDeleteGroupStatus(group as Group, status.id);
|
||||||
|
@ -774,10 +776,10 @@ const MenuButton: React.FC<IMenuButton> = ({
|
||||||
};
|
};
|
||||||
|
|
||||||
const handleTranslate = () => {
|
const handleTranslate = () => {
|
||||||
if (status.translation) {
|
if (targetLanguage) {
|
||||||
dispatch(undoStatusTranslation(status.id));
|
hideTranslation(status.id);
|
||||||
} else {
|
} else {
|
||||||
dispatch(translateStatus(status.id, intl.locale));
|
fetchTranslation(status.id, intl.locale);
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
|
@ -941,7 +943,7 @@ const MenuButton: React.FC<IMenuButton> = ({
|
||||||
}
|
}
|
||||||
|
|
||||||
if (autoTranslating) {
|
if (autoTranslating) {
|
||||||
if (status.translation) {
|
if (targetLanguage) {
|
||||||
menu.push({
|
menu.push({
|
||||||
text: intl.formatMessage(messages.hideTranslation),
|
text: intl.formatMessage(messages.hideTranslation),
|
||||||
action: handleTranslate,
|
action: handleTranslate,
|
||||||
|
|
|
@ -3,6 +3,7 @@ import React, { useState, useRef, useLayoutEffect, useMemo, useEffect } from 're
|
||||||
import { FormattedMessage } from 'react-intl';
|
import { FormattedMessage } from 'react-intl';
|
||||||
|
|
||||||
import { collapseStatusSpoiler, expandStatusSpoiler } from 'pl-fe/actions/statuses';
|
import { collapseStatusSpoiler, expandStatusSpoiler } from 'pl-fe/actions/statuses';
|
||||||
|
import { useStatusTranslation } from 'pl-fe/api/hooks/statuses/use-status-translation';
|
||||||
import Icon from 'pl-fe/components/icon';
|
import Icon from 'pl-fe/components/icon';
|
||||||
import Button from 'pl-fe/components/ui/button';
|
import Button from 'pl-fe/components/ui/button';
|
||||||
import Stack from 'pl-fe/components/ui/stack';
|
import Stack from 'pl-fe/components/ui/stack';
|
||||||
|
@ -11,6 +12,7 @@ import Emojify from 'pl-fe/features/emoji/emojify';
|
||||||
import QuotedStatus from 'pl-fe/features/status/containers/quoted-status-container';
|
import QuotedStatus from 'pl-fe/features/status/containers/quoted-status-container';
|
||||||
import { useAppDispatch } from 'pl-fe/hooks/use-app-dispatch';
|
import { useAppDispatch } from 'pl-fe/hooks/use-app-dispatch';
|
||||||
import { useSettings } from 'pl-fe/hooks/use-settings';
|
import { useSettings } from 'pl-fe/hooks/use-settings';
|
||||||
|
import { useStatusMetaStore } from 'pl-fe/stores/status-meta';
|
||||||
import { onlyEmoji as isOnlyEmoji } from 'pl-fe/utils/rich-content';
|
import { onlyEmoji as isOnlyEmoji } from 'pl-fe/utils/rich-content';
|
||||||
|
|
||||||
import { getTextDirection } from '../utils/rtl';
|
import { getTextDirection } from '../utils/rtl';
|
||||||
|
@ -91,6 +93,9 @@ const StatusContent: React.FC<IStatusContent> = React.memo(({
|
||||||
const node = useRef<HTMLDivElement>(null);
|
const node = useRef<HTMLDivElement>(null);
|
||||||
const spoilerNode = useRef<HTMLSpanElement>(null);
|
const spoilerNode = useRef<HTMLSpanElement>(null);
|
||||||
|
|
||||||
|
const { statuses: statusesMeta } = useStatusMetaStore();
|
||||||
|
const { data: translation } = useStatusTranslation(status.id, statusesMeta[status.id]?.targetLanguage);
|
||||||
|
|
||||||
const maybeSetCollapsed = (): void => {
|
const maybeSetCollapsed = (): void => {
|
||||||
if (!node.current) return;
|
if (!node.current) return;
|
||||||
|
|
||||||
|
@ -125,12 +130,12 @@ const StatusContent: React.FC<IStatusContent> = React.memo(({
|
||||||
});
|
});
|
||||||
|
|
||||||
const content = useMemo(
|
const content = useMemo(
|
||||||
(): string => translatable && status.translation
|
(): string => translation
|
||||||
? status.translation.content!
|
? translation.content
|
||||||
: (status.content_map && status.currentLanguage)
|
: (status.content_map && status.currentLanguage)
|
||||||
? (status.content_map[status.currentLanguage] || status.content)
|
? (status.content_map[status.currentLanguage] || status.content)
|
||||||
: status.content,
|
: status.content,
|
||||||
[status.content, status.translation, status.currentLanguage],
|
[status.content, translation, status.currentLanguage],
|
||||||
);
|
);
|
||||||
|
|
||||||
const { content: parsedContent, hashtags } = useMemo(() => parseContent({
|
const { content: parsedContent, hashtags } = useMemo(() => parseContent({
|
||||||
|
|
|
@ -1,26 +1,25 @@
|
||||||
import React, { useEffect } from 'react';
|
import React, { useEffect } from 'react';
|
||||||
import { FormattedMessage, useIntl } from 'react-intl';
|
import { FormattedMessage, useIntl } from 'react-intl';
|
||||||
|
|
||||||
import { translateStatus, undoStatusTranslation } from 'pl-fe/actions/statuses';
|
|
||||||
import { useTranslationLanguages } from 'pl-fe/api/hooks/instance/use-translation-languages';
|
import { useTranslationLanguages } from 'pl-fe/api/hooks/instance/use-translation-languages';
|
||||||
|
import { useStatusTranslation } from 'pl-fe/api/hooks/statuses/use-status-translation';
|
||||||
import HStack from 'pl-fe/components/ui/hstack';
|
import HStack from 'pl-fe/components/ui/hstack';
|
||||||
import Icon from 'pl-fe/components/ui/icon';
|
import Icon from 'pl-fe/components/ui/icon';
|
||||||
import Stack from 'pl-fe/components/ui/stack';
|
import Stack from 'pl-fe/components/ui/stack';
|
||||||
import Text from 'pl-fe/components/ui/text';
|
import Text from 'pl-fe/components/ui/text';
|
||||||
import { useAppDispatch } from 'pl-fe/hooks/use-app-dispatch';
|
|
||||||
import { useAppSelector } from 'pl-fe/hooks/use-app-selector';
|
import { useAppSelector } from 'pl-fe/hooks/use-app-selector';
|
||||||
import { useFeatures } from 'pl-fe/hooks/use-features';
|
import { useFeatures } from 'pl-fe/hooks/use-features';
|
||||||
import { useInstance } from 'pl-fe/hooks/use-instance';
|
import { useInstance } from 'pl-fe/hooks/use-instance';
|
||||||
import { useSettings } from 'pl-fe/hooks/use-settings';
|
import { useSettings } from 'pl-fe/hooks/use-settings';
|
||||||
|
import { useStatusMetaStore } from 'pl-fe/stores/status-meta';
|
||||||
|
|
||||||
import type { Status } from 'pl-fe/normalizers/status';
|
import type { Status } from 'pl-fe/normalizers/status';
|
||||||
|
|
||||||
interface ITranslateButton {
|
interface ITranslateButton {
|
||||||
status: Pick<Status, 'id' | 'account' | 'content' | 'content_map' | 'language' | 'translating' | 'translation' | 'visibility'>;
|
status: Pick<Status, 'id' | 'account' | 'content' | 'content_map' | 'language' | 'visibility'>;
|
||||||
}
|
}
|
||||||
|
|
||||||
const TranslateButton: React.FC<ITranslateButton> = ({ status }) => {
|
const TranslateButton: React.FC<ITranslateButton> = ({ status }) => {
|
||||||
const dispatch = useAppDispatch();
|
|
||||||
const intl = useIntl();
|
const intl = useIntl();
|
||||||
const features = useFeatures();
|
const features = useFeatures();
|
||||||
const instance = useInstance();
|
const instance = useInstance();
|
||||||
|
@ -30,6 +29,10 @@ const TranslateButton: React.FC<ITranslateButton> = ({ status }) => {
|
||||||
|
|
||||||
const me = useAppSelector((state) => state.me);
|
const me = useAppSelector((state) => state.me);
|
||||||
const { translationLanguages } = useTranslationLanguages();
|
const { translationLanguages } = useTranslationLanguages();
|
||||||
|
const { statuses: statusesMeta, fetchTranslation, hideTranslation } = useStatusMetaStore();
|
||||||
|
|
||||||
|
const targetLanguage = statusesMeta[status.id]?.targetLanguage;
|
||||||
|
const translationQuery = useStatusTranslation(status.id, targetLanguage);
|
||||||
|
|
||||||
const {
|
const {
|
||||||
allow_remote: allowRemote,
|
allow_remote: allowRemote,
|
||||||
|
@ -43,45 +46,45 @@ const TranslateButton: React.FC<ITranslateButton> = ({ status }) => {
|
||||||
const handleTranslate: React.MouseEventHandler<HTMLButtonElement> = (e) => {
|
const handleTranslate: React.MouseEventHandler<HTMLButtonElement> = (e) => {
|
||||||
e.stopPropagation();
|
e.stopPropagation();
|
||||||
|
|
||||||
if (status.translation) {
|
if (targetLanguage) {
|
||||||
dispatch(undoStatusTranslation(status.id));
|
hideTranslation(status.id);
|
||||||
} else {
|
} else {
|
||||||
dispatch(translateStatus(status.id, intl.locale));
|
fetchTranslation(status.id, intl.locale);
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
useEffect(() => {
|
useEffect(() => {
|
||||||
if (status.translation === null && settings.autoTranslate && features.translations && renderTranslate && supportsLanguages && status.translation !== false && status.language !== null && !knownLanguages.includes(status.language)) {
|
if (translationQuery.data === undefined && settings.autoTranslate && features.translations && renderTranslate && supportsLanguages && translationQuery.data !== false && status.language !== null && !knownLanguages.includes(status.language)) {
|
||||||
dispatch(translateStatus(status.id, intl.locale, true));
|
fetchTranslation(status.id, intl.locale);
|
||||||
}
|
}
|
||||||
}, []);
|
}, []);
|
||||||
|
|
||||||
if (!features.translations || !renderTranslate || !supportsLanguages || status.translation === false) return null;
|
if (!features.translations || !renderTranslate || !supportsLanguages || translationQuery.data === false) return null;
|
||||||
|
|
||||||
const button = (
|
const button = (
|
||||||
<button className='w-fit' onClick={handleTranslate}>
|
<button className='w-fit' onClick={handleTranslate}>
|
||||||
<HStack alignItems='center' space={1} className='text-primary-600 hover:underline dark:text-gray-600'>
|
<HStack alignItems='center' space={1} className='text-primary-600 hover:underline dark:text-gray-600'>
|
||||||
<Icon src={require('@tabler/icons/outline/language.svg')} className='size-4' />
|
<Icon src={require('@tabler/icons/outline/language.svg')} className='size-4' />
|
||||||
<span>
|
<span>
|
||||||
{status.translation ? (
|
{translationQuery.data ? (
|
||||||
<FormattedMessage id='status.show_original' defaultMessage='Show original' />
|
<FormattedMessage id='status.show_original' defaultMessage='Show original' />
|
||||||
) : status.translating ? (
|
) : translationQuery.isLoading ? (
|
||||||
<FormattedMessage id='status.translating' defaultMessage='Translating…' />
|
<FormattedMessage id='status.translating' defaultMessage='Translating…' />
|
||||||
) : (
|
) : (
|
||||||
<FormattedMessage id='status.translate' defaultMessage='Translate' />
|
<FormattedMessage id='status.translate' defaultMessage='Translate' />
|
||||||
)}
|
)}
|
||||||
</span>
|
</span>
|
||||||
{status.translating && (
|
{translationQuery.isLoading && (
|
||||||
<Icon src={require('@tabler/icons/outline/loader-2.svg')} className='size-4 animate-spin' />
|
<Icon src={require('@tabler/icons/outline/loader-2.svg')} className='size-4 animate-spin' />
|
||||||
)}
|
)}
|
||||||
</HStack>
|
</HStack>
|
||||||
</button>
|
</button>
|
||||||
);
|
);
|
||||||
|
|
||||||
if (status.translation) {
|
if (translationQuery.data) {
|
||||||
const languageNames = new Intl.DisplayNames([intl.locale], { type: 'language' });
|
const languageNames = new Intl.DisplayNames([intl.locale], { type: 'language' });
|
||||||
const languageName = languageNames.of(status.language!);
|
const languageName = languageNames.of(status.language!);
|
||||||
const provider = status.translation.provider;
|
const provider = translationQuery.data.provider;
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<Stack space={3} alignItems='start'>
|
<Stack space={3} alignItems='start'>
|
||||||
|
|
|
@ -3,7 +3,7 @@
|
||||||
* Converts API statuses into our internal format.
|
* Converts API statuses into our internal format.
|
||||||
* @see {@link https://docs.joinmastodon.org/entities/status/}
|
* @see {@link https://docs.joinmastodon.org/entities/status/}
|
||||||
*/
|
*/
|
||||||
import { type Account as BaseAccount, type Status as BaseStatus, type MediaAttachment, mentionSchema, type Translation } from 'pl-api';
|
import { type Account as BaseAccount, type Status as BaseStatus, type MediaAttachment, mentionSchema } from 'pl-api';
|
||||||
import * as v from 'valibot';
|
import * as v from 'valibot';
|
||||||
|
|
||||||
import { unescapeHTML } from 'pl-fe/utils/html';
|
import { unescapeHTML } from 'pl-fe/utils/html';
|
||||||
|
@ -20,7 +20,6 @@ type CalculatedValues = {
|
||||||
search_index: string;
|
search_index: string;
|
||||||
expanded?: boolean | null;
|
expanded?: boolean | null;
|
||||||
hidden?: boolean | null;
|
hidden?: boolean | null;
|
||||||
translation?: Translation | null | false;
|
|
||||||
currentLanguage?: string;
|
currentLanguage?: string;
|
||||||
};
|
};
|
||||||
|
|
||||||
|
@ -57,11 +56,11 @@ const buildSearchContent = (status: Pick<BaseStatus, 'poll' | 'mentions' | 'spoi
|
||||||
const calculateStatus = (status: BaseStatus, oldStatus?: OldStatus): CalculatedValues => {
|
const calculateStatus = (status: BaseStatus, oldStatus?: OldStatus): CalculatedValues => {
|
||||||
if (oldStatus && oldStatus.content === status.content && oldStatus.spoiler_text === status.spoiler_text) {
|
if (oldStatus && oldStatus.content === status.content && oldStatus.spoiler_text === status.spoiler_text) {
|
||||||
const {
|
const {
|
||||||
search_index, hidden, expanded, translation, currentLanguage,
|
search_index, hidden, expanded, currentLanguage,
|
||||||
} = oldStatus;
|
} = oldStatus;
|
||||||
|
|
||||||
return {
|
return {
|
||||||
search_index, hidden, expanded, translation, currentLanguage,
|
search_index, hidden, expanded, currentLanguage,
|
||||||
};
|
};
|
||||||
} else {
|
} else {
|
||||||
const searchContent = buildSearchContent(status);
|
const searchContent = buildSearchContent(status);
|
||||||
|
@ -129,7 +128,6 @@ const normalizeStatus = (status: BaseStatus & {
|
||||||
reblog_id: status.reblog?.id || null,
|
reblog_id: status.reblog?.id || null,
|
||||||
poll_id: status.poll?.id || null,
|
poll_id: status.poll?.id || null,
|
||||||
group_id: status.group?.id || null,
|
group_id: status.group?.id || null,
|
||||||
translating: false,
|
|
||||||
expectsCard: false,
|
expectsCard: false,
|
||||||
showFiltered: null as null | boolean,
|
showFiltered: null as null | boolean,
|
||||||
...status,
|
...status,
|
||||||
|
@ -144,7 +142,6 @@ const normalizeStatus = (status: BaseStatus & {
|
||||||
group,
|
group,
|
||||||
media_attachments,
|
media_attachments,
|
||||||
...calculated,
|
...calculated,
|
||||||
translation: (status.translation || calculated.translation || null) as Translation | null | false,
|
|
||||||
};
|
};
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|
|
@ -39,10 +39,6 @@ import {
|
||||||
STATUS_HIDE_MEDIA,
|
STATUS_HIDE_MEDIA,
|
||||||
STATUS_MUTE_SUCCESS,
|
STATUS_MUTE_SUCCESS,
|
||||||
STATUS_REVEAL_MEDIA,
|
STATUS_REVEAL_MEDIA,
|
||||||
STATUS_TRANSLATE_FAIL,
|
|
||||||
STATUS_TRANSLATE_REQUEST,
|
|
||||||
STATUS_TRANSLATE_SUCCESS,
|
|
||||||
STATUS_TRANSLATE_UNDO,
|
|
||||||
STATUS_UNFILTER,
|
STATUS_UNFILTER,
|
||||||
STATUS_UNMUTE_SUCCESS,
|
STATUS_UNMUTE_SUCCESS,
|
||||||
STATUS_LANGUAGE_CHANGE,
|
STATUS_LANGUAGE_CHANGE,
|
||||||
|
@ -52,7 +48,7 @@ import {
|
||||||
} from '../actions/statuses';
|
} from '../actions/statuses';
|
||||||
import { TIMELINE_DELETE, type TimelineAction } from '../actions/timelines';
|
import { TIMELINE_DELETE, type TimelineAction } from '../actions/timelines';
|
||||||
|
|
||||||
import type { Status as BaseStatus, CreateStatusParams, Translation } from 'pl-api';
|
import type { Status as BaseStatus, CreateStatusParams } from 'pl-api';
|
||||||
|
|
||||||
type State = Record<string, MinifiedStatus>;
|
type State = Record<string, MinifiedStatus>;
|
||||||
|
|
||||||
|
@ -162,18 +158,6 @@ const simulateDislike = (
|
||||||
state[statusId] = updatedStatus;
|
state[statusId] = updatedStatus;
|
||||||
};
|
};
|
||||||
|
|
||||||
/** Import translation from translation service into the store. */
|
|
||||||
const importTranslation = (state: State, statusId: string, translation: Translation) => {
|
|
||||||
if (!state[statusId]) return;
|
|
||||||
state[statusId].translation = translation;
|
|
||||||
state[statusId].translating = false;
|
|
||||||
};
|
|
||||||
|
|
||||||
/** Delete translation from the store. */
|
|
||||||
const deleteTranslation = (state: State, statusId: string) => {
|
|
||||||
state[statusId].translation = null;
|
|
||||||
};
|
|
||||||
|
|
||||||
const initialState: State = {};
|
const initialState: State = {};
|
||||||
|
|
||||||
const statuses = (state = initialState, action: EmojiReactsAction | EventsAction | ImporterAction | InteractionsAction | StatusesAction | TimelineAction): State => {
|
const statuses = (state = initialState, action: EmojiReactsAction | EventsAction | ImporterAction | InteractionsAction | StatusesAction | TimelineAction): State => {
|
||||||
|
@ -300,7 +284,6 @@ const statuses = (state = initialState, action: EmojiReactsAction | EventsAction
|
||||||
const status = draft[id];
|
const status = draft[id];
|
||||||
if (status) {
|
if (status) {
|
||||||
status.expanded = false;
|
status.expanded = false;
|
||||||
status.translation = false;
|
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
@ -308,25 +291,6 @@ const statuses = (state = initialState, action: EmojiReactsAction | EventsAction
|
||||||
return create(state, (draft) => decrementReplyCount(draft, action.params));
|
return create(state, (draft) => decrementReplyCount(draft, action.params));
|
||||||
case STATUS_DELETE_FAIL:
|
case STATUS_DELETE_FAIL:
|
||||||
return create(state, (draft) => incrementReplyCount(draft, action.params));
|
return create(state, (draft) => incrementReplyCount(draft, action.params));
|
||||||
case STATUS_TRANSLATE_REQUEST:
|
|
||||||
return create(state, (draft) => {
|
|
||||||
const status = draft[action.statusId];
|
|
||||||
if (status) {
|
|
||||||
status.translating = true;
|
|
||||||
}
|
|
||||||
});
|
|
||||||
case STATUS_TRANSLATE_SUCCESS:
|
|
||||||
return action.statusId !== null ? create(state, (draft) => importTranslation(draft, action.statusId!, action.translation)) : state;
|
|
||||||
case STATUS_TRANSLATE_FAIL:
|
|
||||||
return create(state, (draft) => {
|
|
||||||
const status = draft[action.statusId];
|
|
||||||
if (status) {
|
|
||||||
status.translating = false;
|
|
||||||
status.translation = false;
|
|
||||||
}
|
|
||||||
});
|
|
||||||
case STATUS_TRANSLATE_UNDO:
|
|
||||||
return create(state, (draft) => deleteTranslation(draft, action.statusId));
|
|
||||||
case STATUS_UNFILTER:
|
case STATUS_UNFILTER:
|
||||||
return create(state, (draft) => {
|
return create(state, (draft) => {
|
||||||
const status = draft[action.statusId];
|
const status = draft[action.statusId];
|
||||||
|
|
36
packages/pl-fe/src/stores/status-meta.ts
Normal file
36
packages/pl-fe/src/stores/status-meta.ts
Normal file
|
@ -0,0 +1,36 @@
|
||||||
|
import { create } from 'zustand';
|
||||||
|
import { mutative } from 'zustand-mutative';
|
||||||
|
|
||||||
|
type State = {
|
||||||
|
statuses: Record<string, { visible?: boolean; targetLanguage?: string }>;
|
||||||
|
revealStatus: (statusId: string) => void;
|
||||||
|
hideStatus: (statusId: string) => void;
|
||||||
|
fetchTranslation: (statusId: string, targetLanguage: string) => void;
|
||||||
|
hideTranslation: (statusId: string) => void;
|
||||||
|
};
|
||||||
|
|
||||||
|
const useStatusMetaStore = create<State>()(mutative((set) => ({
|
||||||
|
statuses: {},
|
||||||
|
revealStatus: (statusId) => set((state: State) => {
|
||||||
|
if (!state.statuses[statusId]) state.statuses[statusId] = {};
|
||||||
|
|
||||||
|
state.statuses[statusId].visible = true;
|
||||||
|
}),
|
||||||
|
hideStatus: (statusId) => set((state: State) => {
|
||||||
|
if (!state.statuses[statusId]) state.statuses[statusId] = {};
|
||||||
|
|
||||||
|
state.statuses[statusId].visible = false;
|
||||||
|
}),
|
||||||
|
fetchTranslation: (statusId, targetLanguage) => set((state: State) => {
|
||||||
|
if (!state.statuses[statusId]) state.statuses[statusId] = {};
|
||||||
|
|
||||||
|
state.statuses[statusId].targetLanguage = targetLanguage;
|
||||||
|
}),
|
||||||
|
hideTranslation: (statusId) => set((state: State) => {
|
||||||
|
if (!state.statuses[statusId]) state.statuses[statusId] = {};
|
||||||
|
|
||||||
|
state.statuses[statusId].targetLanguage = undefined;
|
||||||
|
}),
|
||||||
|
})));
|
||||||
|
|
||||||
|
export { useStatusMetaStore };
|
Loading…
Reference in a new issue