RWKV-Runner/frontend/src/components/ReadButton.tsx

53 lines
1.6 KiB
TypeScript
Raw Normal View History

2023-05-19 12:55:12 +00:00
import {FC, useState} from 'react';
import {MuteIcon, UnmuteIcon} from '@primer/octicons-react';
import {useTranslation} from 'react-i18next';
import {ToolTipButton} from './ToolTipButton';
import commonStore from '../stores/commonStore';
import {observer} from 'mobx-react-lite';
const synth = window.speechSynthesis;
export const ReadButton: FC<{ content: string }> = observer(({content}) => {
const {t} = useTranslation();
const [speaking, setSpeaking] = useState(false);
let lang: string = commonStore.settings.language;
if (lang === 'dev')
lang = 'en';
const startSpeak = () => {
synth.cancel();
const utterance = new SpeechSynthesisUtterance(content);
const voices = synth.getVoices();
let voice;
if (lang === 'en')
voice = voices.find((v) => v.name.toLowerCase().includes('microsoft aria'));
else if (lang === 'zh')
voice = voices.find((v) => v.name.toLowerCase().includes('xiaoyi'));
if (!voice) voice = voices.find((v) => v.lang.substring(0, 2) === lang);
if (!voice) voice = voices.find((v) => v.lang === navigator.language);
Object.assign(utterance, {
rate: 1,
volume: 1,
onend: () => setSpeaking(false),
onerror: () => setSpeaking(false),
voice: voice
});
synth.speak(utterance);
setSpeaking(true);
};
const stopSpeak = () => {
synth.cancel();
setSpeaking(false);
};
return (
<ToolTipButton desc={t('Read Aloud')} size="small" appearance="subtle" icon={speaking ? <MuteIcon/> : <UnmuteIcon/>}
onClick={speaking ? stopSpeak : startSpeak}/>
);
});