| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301 |
- import {
- useEffect,
- useState,
- type FC,
- useRef
- } from "react";
- import {
- TouchableWithoutFeedback,
- Animated,
- Easing,
- View
- } from "react-native";
- import type ISnackBarProps from "./type";
- import stylesheet, {
- getSnackBarType,
- useStyles
- } from "./stylesheet";
- import {
- NCoreUIKitTheme
- } from "../../core/hooks";
- import {
- useSafeAreaInsets
- } from "react-native-safe-area-context";
- import {
- X as XIcon
- } from "lucide-react-native";
- import {
- Portal
- } from "../../helpers/portalize";
- import Button from "../button";
- import Text from "../text";
- import {
- windowHeight
- } from "../../utils";
- const SnackBar: FC<ISnackBarProps> = ({
- isCloseOnPressActionButton = true,
- closeAnimationDelay = 350,
- openAnimationDelay = 200,
- contentContainerStyle,
- autoCloseDelay = 5000,
- isCloseOnPress = true,
- isShowAction = true,
- isFullWidth = false,
- isInlineSafeArea,
- icon: CustomIcon,
- type = "neutral",
- customTheme,
- onClosed,
- subTitle,
- children,
- onPress,
- action,
- style,
- title,
- id
- }) => {
- const {
- radiuses,
- colors,
- spaces
- } = NCoreUIKitTheme.useContext(customTheme);
- const {
- top
- } = useSafeAreaInsets();
- const [
- isMeasured,
- setIsMeasured
- ] = useState(false);
- const contentHeight = useRef<number>(windowHeight);
- const transformAnim = useRef(new Animated.Value(-contentHeight.current + -top + -spaces.spacingSm)).current;
- const opacityAnim = useRef(new Animated.Value(0)).current;
- const currentType = getSnackBarType({
- type
- });
- const {
- contentContainer: contentContainerDynamicStyle,
- containerObject: containerObjectDynamicStyle,
- iconContainer: iconContainerDynamicStyle,
- container: containerDynamicStyle,
- subTitle: subTitleDynamicStyle,
- action: actionDynamicStyle,
- title: titleDynamicStyle
- } = useStyles({
- isInlineSafeArea,
- safeAreaTop: top,
- isFullWidth,
- currentType,
- radiuses,
- spaces,
- colors,
- type
- });
- useEffect(() => {
- if(isMeasured) {
- transformAnim.setValue(-contentHeight.current + -top + -spaces.spacingSm);
- Animated.parallel([
- Animated.timing(opacityAnim, {
- duration: openAnimationDelay,
- useNativeDriver: true,
- easing: Easing.linear,
- toValue: 1
- }),
- Animated.timing(transformAnim, {
- duration: openAnimationDelay,
- useNativeDriver: true,
- easing: Easing.linear,
- toValue: 0
- })
- ]).start();
- setTimeout(() => {
- closeAnimation();
- }, autoCloseDelay);
- }
- }, [isMeasured]);
- const closeAnimation = (_onClosed?: (props: {
- id: string;
- }) => void) => {
- Animated.parallel([
- Animated.timing(transformAnim, {
- toValue: -contentHeight.current + -top + -spaces.spacingSm,
- duration: closeAnimationDelay,
- useNativeDriver: true,
- easing: Easing.linear
- }),
- Animated.timing(opacityAnim, {
- duration: closeAnimationDelay,
- useNativeDriver: true,
- easing: Easing.linear,
- toValue: 0
- })
- ]).start(({
- finished
- }) => {
- if(finished) {
- if(onClosed) onClosed({
- id
- });
- if(_onClosed) _onClosed({
- id
- });
- }
- });
- };
- const renderIcon = () => {
- if(!CustomIcon) {
- return null;
- }
- return <View
- style={[
- stylesheet.iconContainer,
- iconContainerDynamicStyle
- ]}
- >
- <CustomIcon
- color="default"
- size={18}
- />
- </View>;
- };
- const renderContent = () => {
- return <View
- style={[
- contentContainerStyle,
- stylesheet.contentContainer,
- contentContainerDynamicStyle
- ]}
- >
- <Text
- numberOfLines={3}
- style={{
- ...stylesheet.title,
- ...titleDynamicStyle
- }}
- >
- {title}
- </Text>
- {
- subTitle ?
- <Text
- variant="labelSmallSize"
- numberOfLines={2}
- color="low"
- style={{
- ...stylesheet.subTitle,
- ...subTitleDynamicStyle
- }}
- >
- {subTitle}
- </Text>
- :
- null
- }
- </View>;
- };
- const renderAction = () => {
- if(!isShowAction) {
- return null;
- }
- return <Button
- title={action && action.title ? action.title : undefined}
- isCustomPadding={true}
- spreadBehaviour="free"
- onPress={() => {
- if(isCloseOnPressActionButton) closeAnimation();
- if(action && action.onPress) action.onPress({
- closeAnimation: ({
- onClosed: _onClosed
- }) => closeAnimation(_onClosed)
- });
- }}
- icon={({
- color
- }) => {
- if(action?.title) {
- return null;
- }
- return <XIcon
- color={colors.content.icon[color ? color : "default"]}
- size={18}
- />;
- }}
- style={{
- ...action?.style,
- ...actionDynamicStyle
- }}
- variant="ghost"
- type="neutral"
- size="small"
- />;
- };
- const renderContainer = () => {
- return <View
- style={[
- stylesheet.containerObject,
- containerObjectDynamicStyle
- ]}
- >
- {renderIcon()}
- {renderContent()}
- {renderAction()}
- </View>;
- };
- return <Portal name="snack-bar-system">
- <Animated.View
- onLayout={(event) => {
- const _contentHeight = event.nativeEvent.layout.height;
- contentHeight.current = _contentHeight;
- setIsMeasured(true);
- }}
- style={[
- style,
- stylesheet.container,
- containerDynamicStyle,
- {
- opacity: opacityAnim,
- transform: [{
- translateY: transformAnim
- }]
- }
- ]}
- >
- <TouchableWithoutFeedback
- onPress={() => {
- if(onPress) onPress({
- id
- });
- if(isCloseOnPress) {
- closeAnimation();
- }
- }}
- >
- {children ? children : renderContainer()}
- </TouchableWithoutFeedback>
- </Animated.View>
- </Portal>;
- };
- export default SnackBar;
|