Files
mattermost-mobile/app/components/search/index.tsx
2022-06-24 06:52:05 -04:00

192 lines
6.8 KiB
TypeScript

// Copyright (c) 2015-present Mattermost, Inc. All Rights Reserved.
// See LICENSE.txt for license information.
/* eslint-disable react/prop-types */
// We disable the prop types check here as forwardRef & typescript has a bug
import React, {forwardRef, useCallback, useEffect, useImperativeHandle, useMemo, useRef, useState} from 'react';
import {useIntl} from 'react-intl';
import {ActivityIndicatorProps, Keyboard, Platform, StyleProp, TextInput, TextInputProps, TextStyle, TouchableOpacityProps, ViewStyle} from 'react-native';
import {SearchBar} from 'react-native-elements';
import CompassIcon from '@components/compass_icon';
import {SEARCH_INPUT_HEIGHT} from '@constants/view';
import {useTheme} from '@context/theme';
import {changeOpacity, makeStyleSheetFromTheme} from '@utils/theme';
import {typography} from '@utils/typography';
export type SearchProps = TextInputProps & {
cancelIcon?: React.ReactElement;
cancelButtonProps?: Partial<TouchableOpacityProps> & {
buttonStyle?: StyleProp<ViewStyle>;
buttonTextStyle?: StyleProp<TextStyle>;
color?: string;
buttonDisabledStyle?: StyleProp<ViewStyle>;
buttonDisabledTextStyle?: StyleProp<ViewStyle>;
};
cancelButtonTitle?: string;
clearIcon?: React.ReactElement;
clearIconColor?: string;
containerStyle?: StyleProp<ViewStyle>;
inputContainerStyle?: StyleProp<ViewStyle>;
inputStyle?: StyleProp<TextStyle>;
leftIconContainerStyle?: StyleProp<ViewStyle>;
loadingProps?: ActivityIndicatorProps;
onCancel?(): void;
onClear?(): void;
rightIconContainerStyle?: StyleProp<ViewStyle>;
searchIcon?: React.ReactElement;
searchIconColor?: string;
selectionColor?: string;
showCancel?: boolean;
showLoading?: boolean;
};
type SearchRef = {
blur: () => void;
cancel: () => void;
clear: () => void;
focus: () => void;
}
const getStyleSheet = makeStyleSheetFromTheme((theme: Theme) => ({
containerStyle: {
backgroundColor: undefined,
height: undefined,
paddingTop: 0,
paddingBottom: 0,
},
inputContainerStyle: {
backgroundColor: changeOpacity(theme.centerChannelColor, 0.12),
borderRadius: 8,
height: SEARCH_INPUT_HEIGHT,
marginLeft: 0,
},
inputStyle: {
color: theme.centerChannelColor,
marginLeft: Platform.select({ios: 6, android: 14}),
top: Platform.select({android: 1}),
...typography('Body', 200, 'Regular'),
lineHeight: undefined,
},
}));
const Search = forwardRef<SearchRef, SearchProps>((props: SearchProps, ref) => {
const intl = useIntl();
const theme = useTheme();
const styles = getStyleSheet(theme);
const searchRef = useRef<TextInput>(null);
const [value, setValue] = useState(props.defaultValue || props.value || '');
const searchClearButtonTestID = `${props.testID}.search.clear.button`;
const searchCancelButtonTestID = `${props.testID}.search.cancel.button`;
const searchInputTestID = `${props.testID}.search.input`;
const onCancel = useCallback(() => {
Keyboard.dismiss();
setValue('');
props.onCancel?.();
}, [props.onCancel]);
const onClear = useCallback(() => {
setValue('');
props.onClear?.();
}, []);
const onChangeText = useCallback((text: string) => {
setValue(text);
props.onChangeText?.(text);
}, [props.onChangeText]);
const cancelButtonProps = useMemo(() => ({
buttonTextStyle: {
color: changeOpacity(theme.centerChannelColor, 0.72),
...typography('Body', 100, 'Regular'),
},
}), [theme]);
useEffect(() => {
setValue(props.defaultValue || value || '');
}, [props.defaultValue]);
const clearIcon = (
<CompassIcon
color={changeOpacity(props.clearIconColor || theme.centerChannelColor, Platform.select({android: 0.56, default: 0.72}))}
name={Platform.select({android: 'close', default: 'close-circle'})}
onPress={searchRef.current?.clear}
size={Platform.select({android: 24, default: 18})}
testID={searchClearButtonTestID}
/>
);
const searchIcon = (
<CompassIcon
color={changeOpacity(props.searchIconColor || theme.centerChannelColor, Platform.select({android: 0.56, default: 0.72}))}
name='magnify'
onPress={searchRef.current?.focus}
size={24}
/>
);
const cancelIcon = (
<CompassIcon
color={changeOpacity(props.cancelButtonProps?.color || theme.centerChannelColor, Platform.select({android: 0.56, default: 0.72}))}
name='arrow-left'
onPress={onCancel}
size={24}
testID={searchCancelButtonTestID}
/>
);
useImperativeHandle(ref, () => ({
blur: () => {
searchRef.current?.blur();
},
cancel: () => {
// @ts-expect-error cancel is not part of TextInput does exist in SearchBar
searchRef.current?.cancel();
},
clear: () => {
searchRef.current?.clear();
},
focus: () => {
searchRef.current?.focus();
},
}), [searchRef]);
return (
<SearchBar
{...props}
cancelButtonProps={props.cancelButtonProps || cancelButtonProps}
cancelButtonTitle={props.cancelButtonTitle || intl.formatMessage({id: 'mobile.post.cancel', defaultMessage: 'Cancel'})}
cancelIcon={cancelIcon}
// @ts-expect-error clearIcon definition does not include a ReactElement
clearIcon={clearIcon}
containerStyle={[styles.containerStyle, props.containerStyle]}
inputContainerStyle={[styles.inputContainerStyle, props.inputContainerStyle]}
inputStyle={[styles.inputStyle, props.inputStyle]}
returnKeyType='search'
onCancel={onCancel}
onClear={onClear}
// @ts-expect-error onChangeText type definition is wrong in elements
onChangeText={onChangeText}
placeholder={props.placeholder || intl.formatMessage({id: 'search_bar.search', defaultMessage: 'Search'})}
placeholderTextColor={props.placeholderTextColor || changeOpacity(theme.centerChannelColor, Platform.select({android: 0.56, default: 0.72}))}
platform={Platform.select({android: 'android', default: 'ios'})}
ref={searchRef}
// @ts-expect-error searchIcon definition does not include a ReactElement
searchIcon={searchIcon}
selectionColor={Platform.select({android: changeOpacity(props.selectionColor || theme.centerChannelColor, 0.24), default: props.selectionColor || theme.centerChannelColor})}
testID={searchInputTestID}
value={value}
/>
);
});
Search.displayName = 'SeachBar';
export default Search;