2022-06-27 19:32:34 +00:00
|
|
|
import React, { useEffect, useState } from 'react';
|
|
|
|
import { View, Text, TouchableOpacity, TextInput, FlatList } from 'react-native';
|
|
|
|
import { orderBy } from 'lodash';
|
|
|
|
|
2022-07-14 13:50:41 +00:00
|
|
|
import { FormTextInput } from '../TextInput/FormTextInput';
|
2022-06-27 19:32:34 +00:00
|
|
|
import { useTheme } from '../../theme';
|
|
|
|
import I18n from '../../i18n';
|
|
|
|
import { CustomIcon } from '../CustomIcon';
|
|
|
|
import { IEmoji } from '../../definitions';
|
|
|
|
import shortnameToUnicode from '../../lib/methods/helpers/shortnameToUnicode';
|
|
|
|
import CustomEmoji from '../EmojiPicker/CustomEmoji';
|
|
|
|
import database from '../../lib/database';
|
2022-06-27 21:30:21 +00:00
|
|
|
import styles from './styles';
|
2022-06-27 19:32:34 +00:00
|
|
|
|
2022-06-28 14:41:52 +00:00
|
|
|
const BUTTON_HIT_SLOP = { top: 15, right: 15, bottom: 15, left: 15 };
|
|
|
|
|
2022-06-27 19:32:34 +00:00
|
|
|
interface IEmojiSearchbarProps {
|
|
|
|
openEmoji: () => void;
|
|
|
|
onChangeText: (value: string) => void;
|
|
|
|
emojis: IEmoji[];
|
|
|
|
onEmojiSelected: (emoji: IEmoji) => void;
|
|
|
|
baseUrl: string;
|
|
|
|
}
|
|
|
|
|
|
|
|
const renderEmoji = (emoji: IEmoji, size: number, baseUrl: string) => {
|
|
|
|
if (emoji.name) {
|
2022-06-27 21:30:21 +00:00
|
|
|
return <CustomEmoji style={{ height: size, width: size, margin: 4 }} emoji={emoji} baseUrl={baseUrl} />;
|
2022-06-27 19:32:34 +00:00
|
|
|
}
|
2022-06-27 21:30:21 +00:00
|
|
|
return <Text style={[styles.searchedEmoji, { fontSize: size }]}>{shortnameToUnicode(`:${emoji}:`)}</Text>;
|
2022-06-27 19:32:34 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
const EmojiSearchbar = React.forwardRef<TextInput, IEmojiSearchbarProps>(
|
|
|
|
({ openEmoji, onChangeText, emojis, onEmojiSelected, baseUrl }, ref) => {
|
2022-07-14 13:50:41 +00:00
|
|
|
const { colors } = useTheme();
|
2022-06-27 19:32:34 +00:00
|
|
|
const [searchText, setSearchText] = useState<string>('');
|
|
|
|
const [frequentlyUsed, setFrequentlyUsed] = useState([]);
|
|
|
|
|
|
|
|
const getFrequentlyUsedEmojis = async () => {
|
|
|
|
const db = database.active;
|
|
|
|
const frequentlyUsedRecords = await db.get('frequently_used_emojis').query().fetch();
|
|
|
|
const frequentlyUsedOrdered = orderBy(frequentlyUsedRecords, ['count'], ['desc']);
|
|
|
|
const frequentlyUsedEmojis = frequentlyUsedOrdered.map(item => {
|
|
|
|
if (item.isCustom) {
|
|
|
|
return { name: item.content, extension: item.extension };
|
|
|
|
}
|
|
|
|
return item.content;
|
|
|
|
});
|
|
|
|
// @ts-ignore
|
|
|
|
setFrequentlyUsed(frequentlyUsedEmojis);
|
|
|
|
};
|
|
|
|
|
|
|
|
useEffect(() => {
|
|
|
|
getFrequentlyUsedEmojis();
|
|
|
|
}, []);
|
|
|
|
|
|
|
|
const handleTextChange = (text: string) => {
|
|
|
|
setSearchText(text);
|
|
|
|
onChangeText(text);
|
|
|
|
if (!text) getFrequentlyUsedEmojis();
|
|
|
|
};
|
|
|
|
|
|
|
|
const renderItem = (emoji: IEmoji) => {
|
|
|
|
const emojiSize = 30;
|
|
|
|
return (
|
2022-06-27 21:30:21 +00:00
|
|
|
<View style={[styles.emojiContainer]}>
|
2022-06-28 14:41:52 +00:00
|
|
|
<TouchableOpacity activeOpacity={0.7} onPress={() => onEmojiSelected(emoji)}>
|
2022-06-27 19:32:34 +00:00
|
|
|
{renderEmoji(emoji, emojiSize, baseUrl)}
|
|
|
|
</TouchableOpacity>
|
|
|
|
</View>
|
|
|
|
);
|
|
|
|
};
|
|
|
|
return (
|
2022-06-28 14:41:52 +00:00
|
|
|
<View style={{ borderTopWidth: 1, borderTopColor: colors.borderColor, backgroundColor: colors.backgroundColor }}>
|
2022-06-27 21:30:21 +00:00
|
|
|
<FlatList
|
|
|
|
horizontal
|
|
|
|
data={searchText ? emojis : frequentlyUsed}
|
|
|
|
renderItem={({ item }) => renderItem(item)}
|
|
|
|
showsHorizontalScrollIndicator={false}
|
2022-06-28 14:41:52 +00:00
|
|
|
ListEmptyComponent={() => (
|
|
|
|
<View style={styles.listEmptyComponent}>
|
|
|
|
<Text style={{ color: colors.auxiliaryText }}>{I18n.t('No_results_found')}</Text>
|
|
|
|
</View>
|
|
|
|
)}
|
|
|
|
// @ts-ignore
|
|
|
|
keyExtractor={item => item.name || item}
|
2022-06-27 21:30:21 +00:00
|
|
|
contentContainerStyle={styles.emojiListContainer}
|
|
|
|
keyboardShouldPersistTaps='always'
|
|
|
|
/>
|
|
|
|
<View style={styles.emojiSearchbarContainer}>
|
2022-06-28 14:41:52 +00:00
|
|
|
<TouchableOpacity style={styles.openEmojiKeyboard} activeOpacity={0.7} onPress={openEmoji} hitSlop={BUTTON_HIT_SLOP}>
|
2022-06-27 19:32:34 +00:00
|
|
|
<CustomIcon name='chevron-left' size={30} color={colors.collapsibleChevron} />
|
|
|
|
</TouchableOpacity>
|
|
|
|
<View style={{ flex: 1 }}>
|
|
|
|
<FormTextInput
|
|
|
|
inputRef={ref}
|
|
|
|
autoCapitalize='none'
|
|
|
|
autoCorrect={false}
|
|
|
|
blurOnSubmit
|
|
|
|
placeholder={I18n.t('Search_emoji')}
|
|
|
|
returnKeyType='search'
|
|
|
|
underlineColorAndroid='transparent'
|
|
|
|
onChangeText={handleTextChange}
|
2022-06-27 21:30:21 +00:00
|
|
|
style={[styles.emojiSearchbar, { backgroundColor: colors.passcodeButtonActive }]}
|
|
|
|
containerStyle={styles.textInputContainer}
|
2022-06-27 19:32:34 +00:00
|
|
|
value={searchText}
|
|
|
|
onClearInput={() => handleTextChange('')}
|
|
|
|
iconRight={'search'}
|
|
|
|
/>
|
|
|
|
</View>
|
|
|
|
</View>
|
|
|
|
</View>
|
|
|
|
);
|
|
|
|
}
|
|
|
|
);
|
|
|
|
|
|
|
|
export default EmojiSearchbar;
|