mirror of
https://github.com/cheeaun/phanpy.git
synced 2025-01-05 09:07:19 +03:00
155 lines
4.7 KiB
React
155 lines
4.7 KiB
React
|
import './translation-block.css';
|
||
|
|
||
|
import { useEffect, useRef, useState } from 'preact/hooks';
|
||
|
|
||
|
import sourceLanguages from '../data/lingva-source-languages';
|
||
|
import getTranslateTargetLanguage from '../utils/get-translate-target-language';
|
||
|
import localeCode2Text from '../utils/localeCode2Text';
|
||
|
|
||
|
import Icon from './icon';
|
||
|
import Loader from './loader';
|
||
|
|
||
|
function TranslationBlock({
|
||
|
forceTranslate,
|
||
|
sourceLanguage,
|
||
|
onTranslate,
|
||
|
text = '',
|
||
|
}) {
|
||
|
const targetLang = getTranslateTargetLanguage(true);
|
||
|
const [uiState, setUIState] = useState('default');
|
||
|
const [pronunciationContent, setPronunciationContent] = useState(null);
|
||
|
const [translatedContent, setTranslatedContent] = useState(null);
|
||
|
const [detectedLang, setDetectedLang] = useState(null);
|
||
|
const detailsRef = useRef();
|
||
|
|
||
|
const sourceLangText = sourceLanguage
|
||
|
? localeCode2Text(sourceLanguage)
|
||
|
: null;
|
||
|
const targetLangText = localeCode2Text(targetLang);
|
||
|
const apiSourceLang = useRef('auto');
|
||
|
|
||
|
if (!onTranslate)
|
||
|
onTranslate = (source, target) => {
|
||
|
console.log('TRANSLATE', source, target, text);
|
||
|
// Using another API instance instead of lingva.ml because of this bug (slashes don't work):
|
||
|
// https://github.com/thedaviddelta/lingva-translate/issues/68
|
||
|
return fetch(
|
||
|
`https://lingva.garudalinux.org/api/v1/${source}/${target}/${encodeURIComponent(
|
||
|
text,
|
||
|
)}`,
|
||
|
)
|
||
|
.then((res) => res.json())
|
||
|
.then((res) => {
|
||
|
return {
|
||
|
provider: 'lingva',
|
||
|
content: res.translation,
|
||
|
detectedSourceLanguage: res.info.detectedSource,
|
||
|
info: res.info,
|
||
|
};
|
||
|
});
|
||
|
// return masto.v1.statuses.translate(id, {
|
||
|
// lang: DEFAULT_LANG,
|
||
|
// });
|
||
|
};
|
||
|
|
||
|
const translate = async () => {
|
||
|
setUIState('loading');
|
||
|
const { content, detectedSourceLanguage, provider, ...props } =
|
||
|
await onTranslate(apiSourceLang.current, targetLang);
|
||
|
if (content) {
|
||
|
if (detectedSourceLanguage) {
|
||
|
const detectedLangText = localeCode2Text(detectedSourceLanguage);
|
||
|
setDetectedLang(detectedLangText);
|
||
|
}
|
||
|
if (provider === 'lingva') {
|
||
|
const pronunciation = props?.info?.pronunciation?.query;
|
||
|
if (pronunciation) {
|
||
|
setPronunciationContent(pronunciation);
|
||
|
}
|
||
|
}
|
||
|
setTranslatedContent(content);
|
||
|
setUIState('default');
|
||
|
detailsRef.current.open = true;
|
||
|
detailsRef.current.scrollIntoView({
|
||
|
behavior: 'smooth',
|
||
|
block: 'nearest',
|
||
|
});
|
||
|
} else {
|
||
|
console.error(result);
|
||
|
setUIState('error');
|
||
|
}
|
||
|
};
|
||
|
|
||
|
useEffect(() => {
|
||
|
if (forceTranslate) {
|
||
|
translate();
|
||
|
}
|
||
|
}, [forceTranslate]);
|
||
|
|
||
|
return (
|
||
|
<div class="status-translation-block">
|
||
|
<details ref={detailsRef}>
|
||
|
<summary>
|
||
|
<button
|
||
|
type="button"
|
||
|
onClick={async (e) => {
|
||
|
e.preventDefault();
|
||
|
e.stopPropagation();
|
||
|
detailsRef.current.open = !detailsRef.current.open;
|
||
|
if (uiState === 'loading') return;
|
||
|
if (!translatedContent) translate();
|
||
|
}}
|
||
|
>
|
||
|
<Icon icon="translate" />{' '}
|
||
|
<span>
|
||
|
{uiState === 'loading'
|
||
|
? 'Translating…'
|
||
|
: sourceLanguage && !detectedLang
|
||
|
? `Translate from ${sourceLangText}`
|
||
|
: `Translate`}
|
||
|
</span>
|
||
|
</button>
|
||
|
</summary>
|
||
|
<div class="translated-block">
|
||
|
<div class="translation-info insignificant">
|
||
|
<select
|
||
|
class="translated-source-select"
|
||
|
disabled={uiState === 'loading'}
|
||
|
onChange={(e) => {
|
||
|
apiSourceLang.current = e.target.value;
|
||
|
translate();
|
||
|
}}
|
||
|
>
|
||
|
{sourceLanguages.map((l) => (
|
||
|
<option value={l.code}>
|
||
|
{l.code === 'auto' ? `Auto (${detectedLang ?? '…'})` : l.name}
|
||
|
</option>
|
||
|
))}
|
||
|
</select>{' '}
|
||
|
<span>→ {targetLangText}</span>
|
||
|
<Loader abrupt hidden={uiState !== 'loading'} />
|
||
|
</div>
|
||
|
{uiState === 'error' ? (
|
||
|
<p class="ui-state">Failed to translate</p>
|
||
|
) : (
|
||
|
!!translatedContent && (
|
||
|
<>
|
||
|
{!!pronunciationContent && (
|
||
|
<output class="translated-pronunciation-content">
|
||
|
{pronunciationContent}
|
||
|
</output>
|
||
|
)}
|
||
|
<output class="translated-content" lang={targetLang}>
|
||
|
{translatedContent}
|
||
|
</output>
|
||
|
</>
|
||
|
)
|
||
|
)}
|
||
|
</div>
|
||
|
</details>
|
||
|
</div>
|
||
|
);
|
||
|
}
|
||
|
|
||
|
export default TranslationBlock;
|