2019-12-07 13:28:52 +02:00
|
|
|
import React from 'react';
|
|
|
|
import { FormattedMessage as Message, injectIntl, IntlShape } from 'react-intl';
|
2019-12-07 21:43:08 +02:00
|
|
|
import clsx from 'clsx';
|
2019-06-30 16:32:50 +03:00
|
|
|
import { connect } from 'react-redux';
|
2019-12-07 21:02:00 +02:00
|
|
|
import { changeLang } from 'app/components/user/actions';
|
|
|
|
import LANGS from 'app/i18n';
|
|
|
|
import formStyles from 'app/components/ui/form/form.scss';
|
|
|
|
import popupStyles from 'app/components/ui/popup/popup.scss';
|
|
|
|
import icons from 'app/components/ui/icons.scss';
|
2019-06-30 16:32:50 +03:00
|
|
|
|
2017-10-15 22:00:21 +03:00
|
|
|
import styles from './languageSwitcher.scss';
|
|
|
|
import messages from './languageSwitcher.intl.json';
|
2017-12-30 23:38:54 +02:00
|
|
|
import LanguageList from './LanguageList';
|
2019-12-07 21:02:00 +02:00
|
|
|
import { RootState } from 'app/reducers';
|
2017-12-17 03:26:09 +03:00
|
|
|
|
2019-05-21 18:23:13 +03:00
|
|
|
const translateUrl = 'http://ely.by/translate';
|
2017-10-15 22:00:21 +03:00
|
|
|
|
2017-12-30 23:38:54 +02:00
|
|
|
export type LocaleData = {
|
2019-12-07 13:28:52 +02:00
|
|
|
code: string;
|
|
|
|
name: string;
|
|
|
|
englishName: string;
|
|
|
|
progress: number;
|
|
|
|
isReleased: boolean;
|
2017-12-30 23:38:54 +02:00
|
|
|
};
|
|
|
|
|
2019-11-27 11:03:32 +02:00
|
|
|
export type LocalesMap = { [code: string]: LocaleData };
|
2017-12-30 23:38:54 +02:00
|
|
|
|
2019-12-07 13:28:52 +02:00
|
|
|
type OwnProps = {
|
|
|
|
onClose: () => void;
|
|
|
|
langs: LocalesMap;
|
2019-11-27 11:03:32 +02:00
|
|
|
emptyCaptions: Array<{
|
2019-12-07 13:28:52 +02:00
|
|
|
src: string;
|
|
|
|
caption: string;
|
|
|
|
}>;
|
2019-06-30 16:32:50 +03:00
|
|
|
};
|
|
|
|
|
2019-12-07 13:28:52 +02:00
|
|
|
interface Props extends OwnProps {
|
|
|
|
intl: IntlShape;
|
|
|
|
selectedLocale: string;
|
|
|
|
changeLang: (lang: string) => void;
|
|
|
|
}
|
|
|
|
|
|
|
|
class LanguageSwitcher extends React.Component<
|
2019-11-27 11:03:32 +02:00
|
|
|
Props,
|
|
|
|
{
|
2019-12-07 13:28:52 +02:00
|
|
|
filter: string;
|
|
|
|
filteredLangs: LocalesMap;
|
|
|
|
}
|
2019-11-27 11:03:32 +02:00
|
|
|
> {
|
|
|
|
state = {
|
|
|
|
filter: '',
|
|
|
|
filteredLangs: this.props.langs,
|
|
|
|
};
|
|
|
|
|
|
|
|
static defaultProps = {
|
|
|
|
langs: LANGS,
|
|
|
|
onClose() {},
|
|
|
|
};
|
|
|
|
|
|
|
|
render() {
|
|
|
|
const { selectedLocale, onClose, intl } = this.props;
|
|
|
|
const { filteredLangs } = this.state;
|
|
|
|
|
|
|
|
return (
|
2019-12-28 13:13:11 +02:00
|
|
|
<div
|
|
|
|
className={styles.languageSwitcher}
|
|
|
|
data-testid="language-switcher"
|
|
|
|
data-e2e-active-locale={selectedLocale}
|
|
|
|
>
|
2019-11-27 11:03:32 +02:00
|
|
|
<div className={popupStyles.popup}>
|
|
|
|
<div className={popupStyles.header}>
|
|
|
|
<h2 className={popupStyles.headerTitle}>
|
|
|
|
<Message {...messages.siteLanguage} />
|
|
|
|
</h2>
|
|
|
|
<span
|
2019-12-07 21:43:08 +02:00
|
|
|
className={clsx(icons.close, popupStyles.close)}
|
2019-11-27 11:03:32 +02:00
|
|
|
onClick={onClose}
|
|
|
|
/>
|
|
|
|
</div>
|
|
|
|
|
|
|
|
<div className={styles.languageSwitcherBody}>
|
|
|
|
<div className={styles.searchBox}>
|
|
|
|
<input
|
2019-12-07 21:43:08 +02:00
|
|
|
className={clsx(
|
2019-11-27 11:03:32 +02:00
|
|
|
formStyles.lightTextField,
|
|
|
|
formStyles.greenTextField,
|
|
|
|
)}
|
|
|
|
placeholder={intl.formatMessage(messages.startTyping)}
|
|
|
|
onChange={this.onFilterUpdate}
|
|
|
|
onKeyPress={this.onFilterKeyPress()}
|
|
|
|
autoFocus
|
|
|
|
/>
|
|
|
|
<span className={styles.searchIcon} />
|
|
|
|
</div>
|
2017-10-15 22:00:21 +03:00
|
|
|
|
2019-11-27 11:03:32 +02:00
|
|
|
<LanguageList
|
|
|
|
selectedLocale={selectedLocale}
|
|
|
|
langs={filteredLangs}
|
|
|
|
onChangeLang={this.onChangeLang}
|
|
|
|
/>
|
|
|
|
|
|
|
|
<div className={styles.improveTranslates}>
|
|
|
|
<div className={styles.improveTranslatesIcon} />
|
|
|
|
<div className={styles.improveTranslatesContent}>
|
|
|
|
<div className={styles.improveTranslatesTitle}>
|
|
|
|
<Message {...messages.improveTranslates} />
|
2017-10-15 22:00:21 +03:00
|
|
|
</div>
|
2019-11-27 11:03:32 +02:00
|
|
|
<div className={styles.improveTranslatesText}>
|
|
|
|
<Message {...messages.improveTranslatesDescription} />{' '}
|
|
|
|
<a href={translateUrl} target="_blank">
|
|
|
|
<Message {...messages.improveTranslatesParticipate} />
|
|
|
|
</a>
|
|
|
|
</div>
|
|
|
|
</div>
|
2017-10-15 22:00:21 +03:00
|
|
|
</div>
|
2019-11-27 11:03:32 +02:00
|
|
|
</div>
|
|
|
|
</div>
|
|
|
|
</div>
|
|
|
|
);
|
|
|
|
}
|
2017-10-15 22:00:21 +03:00
|
|
|
|
2019-11-27 11:03:32 +02:00
|
|
|
onChangeLang = this.changeLang.bind(this);
|
2017-10-15 22:00:21 +03:00
|
|
|
|
2019-11-27 11:03:32 +02:00
|
|
|
changeLang(lang: string) {
|
|
|
|
this.props.changeLang(lang);
|
2017-12-30 23:38:54 +02:00
|
|
|
|
2019-11-27 11:03:32 +02:00
|
|
|
setTimeout(this.props.onClose, 300);
|
|
|
|
}
|
2017-10-15 22:00:21 +03:00
|
|
|
|
2019-12-07 13:28:52 +02:00
|
|
|
onFilterUpdate = (event: React.ChangeEvent<HTMLInputElement>) => {
|
2019-11-27 11:03:32 +02:00
|
|
|
const filter = event.currentTarget.value.trim().toLowerCase();
|
|
|
|
const { langs } = this.props;
|
2017-12-30 23:38:54 +02:00
|
|
|
|
2019-11-27 11:03:32 +02:00
|
|
|
const result = Object.keys(langs).reduce((previous, key) => {
|
|
|
|
if (
|
|
|
|
langs[key].englishName.toLowerCase().indexOf(filter) === -1 &&
|
|
|
|
langs[key].name.toLowerCase().indexOf(filter) === -1
|
|
|
|
) {
|
|
|
|
return previous;
|
|
|
|
}
|
2017-10-15 22:00:21 +03:00
|
|
|
|
2019-11-27 11:03:32 +02:00
|
|
|
previous[key] = langs[key];
|
2017-12-16 19:55:57 +03:00
|
|
|
|
2019-11-27 11:03:32 +02:00
|
|
|
return previous;
|
|
|
|
}, {});
|
2017-12-16 19:55:57 +03:00
|
|
|
|
2019-11-27 11:03:32 +02:00
|
|
|
this.setState({
|
|
|
|
filter,
|
|
|
|
filteredLangs: result,
|
|
|
|
});
|
|
|
|
};
|
2017-10-15 22:00:21 +03:00
|
|
|
|
2019-11-27 11:03:32 +02:00
|
|
|
onFilterKeyPress() {
|
2019-12-07 13:28:52 +02:00
|
|
|
return (event: React.KeyboardEvent<HTMLInputElement>) => {
|
2019-11-27 11:03:32 +02:00
|
|
|
if (event.key !== 'Enter' || this.state.filter === '') {
|
|
|
|
return;
|
|
|
|
}
|
2017-10-15 22:00:21 +03:00
|
|
|
|
2019-11-27 11:03:32 +02:00
|
|
|
const locales = Object.keys(this.state.filteredLangs);
|
2017-12-30 23:38:54 +02:00
|
|
|
|
2019-11-27 11:03:32 +02:00
|
|
|
if (locales.length === 0) {
|
|
|
|
return;
|
|
|
|
}
|
2017-10-15 22:00:21 +03:00
|
|
|
|
2019-11-27 11:03:32 +02:00
|
|
|
this.changeLang(locales[0]);
|
|
|
|
};
|
|
|
|
}
|
2017-10-15 22:00:21 +03:00
|
|
|
}
|
|
|
|
|
2019-11-11 10:40:05 +02:00
|
|
|
export default injectIntl(
|
2019-12-07 13:28:52 +02:00
|
|
|
connect(
|
|
|
|
(state: RootState) => ({
|
2019-11-27 11:03:32 +02:00
|
|
|
selectedLocale: state.i18n.locale,
|
|
|
|
}),
|
|
|
|
{
|
|
|
|
changeLang,
|
|
|
|
},
|
|
|
|
)(LanguageSwitcher),
|
2019-11-11 10:40:05 +02:00
|
|
|
);
|