BlueWallet/components/ListItem.tsx
2024-07-24 17:42:09 -04:00

221 lines
6.2 KiB
TypeScript

import React, { useMemo } from 'react';
import { ActivityIndicator, I18nManager, Pressable, PressableProps, StyleSheet, Switch, TouchableOpacity, View } from 'react-native';
import { Avatar, ListItem as RNElementsListItem, Button } from '@rneui/themed'; // Replace with actual import paths
import { useTheme } from './themes';
// Update the type for the props
interface ListItemProps {
swipeable?: boolean;
rightIcon?: any;
leftAvatar?: React.JSX.Element;
containerStyle?: object;
Component?: typeof React.Component | typeof PressableWrapper;
bottomDivider?: boolean;
topDivider?: boolean;
testID?: string;
onPress?: () => void;
onLongPress?: () => void;
onDeletePressed?: () => void;
disabled?: boolean;
switch?: object; // Define more specific type if needed
leftIcon?: any; // Define more specific type if needed
title: string;
subtitle?: string | React.ReactNode;
subtitleNumberOfLines?: number;
rightTitle?: string;
rightTitleStyle?: object;
isLoading?: boolean;
chevron?: boolean;
checkmark?: boolean;
subtitleProps?: object;
swipeableLeftContent?: React.ReactNode;
swipeableRightContent?: React.ReactNode;
}
export class PressableWrapper extends React.Component<PressableProps> {
render() {
return <Pressable {...this.props} />;
}
}
export class TouchableOpacityWrapper extends React.Component {
render() {
return <TouchableOpacity {...this.props} />;
}
}
// Define Swipeable Button Components
const DefaultRightContent: React.FC<{ reset: () => void; onDeletePressed?: () => void }> = ({ reset, onDeletePressed }) => (
<Button
title="Delete"
onPress={() => {
reset();
onDeletePressed?.();
}}
icon={{ name: 'delete', color: 'white' }}
buttonStyle={styles.rightButton}
/>
);
const ListItem: React.FC<ListItemProps> = React.memo(
({
swipeable = false,
Component = TouchableOpacityWrapper,
rightIcon,
leftAvatar,
containerStyle,
bottomDivider = true,
topDivider = false,
testID,
onPress,
onLongPress,
onDeletePressed,
disabled,
switch: switchProps,
leftIcon,
title,
subtitle,
subtitleNumberOfLines,
rightTitle,
rightTitleStyle,
isLoading,
chevron,
checkmark,
swipeableLeftContent,
swipeableRightContent,
}: ListItemProps) => {
const { colors } = useTheme();
const stylesHook = StyleSheet.create({
title: {
color: disabled ? colors.buttonDisabledTextColor : colors.foregroundColor,
fontSize: 16,
fontWeight: '500',
writingDirection: I18nManager.isRTL ? 'rtl' : 'ltr',
},
subtitle: {
flexWrap: 'wrap',
writingDirection: I18nManager.isRTL ? 'rtl' : 'ltr',
color: colors.alternativeTextColor,
fontWeight: '400',
fontSize: 14,
},
containerStyle: {
backgroundColor: colors.background,
},
});
const memoizedSwitchProps = useMemo(() => {
return switchProps ? { ...switchProps } : undefined;
}, [switchProps]);
const renderContent = () => (
<>
{leftIcon && (
<>
<View style={styles.width16} />
<Avatar icon={leftIcon} />
</>
)}
{leftAvatar && (
<>
{leftAvatar}
<View style={styles.width16} />
</>
)}
<RNElementsListItem.Content>
<RNElementsListItem.Title style={stylesHook.title} numberOfLines={0} accessible={switchProps === undefined}>
{title}
</RNElementsListItem.Title>
{subtitle && (
<RNElementsListItem.Subtitle
numberOfLines={subtitleNumberOfLines ?? 1}
accessible={switchProps === undefined}
style={stylesHook.subtitle}
>
{subtitle}
</RNElementsListItem.Subtitle>
)}
</RNElementsListItem.Content>
{rightTitle && (
<View style={styles.margin8}>
<RNElementsListItem.Title style={rightTitleStyle} numberOfLines={0}>
{rightTitle}
</RNElementsListItem.Title>
</View>
)}
{isLoading ? (
<ActivityIndicator />
) : (
<>
{chevron && <RNElementsListItem.Chevron iconStyle={{ transform: [{ scaleX: I18nManager.isRTL ? -1 : 1 }] }} />}
{rightIcon && <Avatar icon={rightIcon} />}
{switchProps && <Switch {...memoizedSwitchProps} accessibilityLabel={title} accessible accessibilityRole="switch" />}
{checkmark && (
<RNElementsListItem.CheckBox
iconRight
containerStyle={stylesHook.containerStyle}
iconType="octaicon"
checkedIcon="check"
checked
/>
)}
</>
)}
</>
);
if (swipeable && !Component) {
console.warn('Component prop is required when swipeable is true.');
return null;
}
return swipeable ? (
<RNElementsListItem.Swipeable
containerStyle={containerStyle ?? stylesHook.containerStyle}
Component={Component}
bottomDivider={bottomDivider}
topDivider={topDivider}
testID={testID}
onPress={onPress}
onLongPress={onLongPress}
disabled={disabled}
leftContent={swipeableLeftContent}
rightContent={swipeableRightContent ?? <DefaultRightContent reset={() => {}} onDeletePressed={onDeletePressed} />}
accessible={switchProps === undefined}
>
{renderContent()}
</RNElementsListItem.Swipeable>
) : (
<RNElementsListItem
containerStyle={containerStyle ?? stylesHook.containerStyle}
Component={Component}
bottomDivider={bottomDivider}
topDivider={topDivider}
testID={testID}
onPress={onPress}
onLongPress={onLongPress}
disabled={disabled}
accessible={switchProps === undefined}
>
{renderContent()}
</RNElementsListItem>
);
},
);
export default ListItem;
const styles = StyleSheet.create({
rightButton: {
minHeight: '100%',
backgroundColor: 'red',
},
margin8: {
margin: 8,
},
width16: { width: 16 },
});