Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

[iOS on expo 52] - Swipe-Back Animates Last Card, Then Jumps Another Card Back Without Animation, Breaking Swiper! (Might be the latest version of React Native) #400

Open
ShmuelPickRanky opened this issue Dec 22, 2024 · 1 comment

Comments

@ShmuelPickRanky
Copy link

On iOS with Expo SDK 52, using the swipe-back gesture results in the following critical issue:

  1. The last card animates back correctly.
  2. Immediately after, the swiper skips an additional card backward without animation.
  3. The swiper becomes unresponsive to further interactions.

NOTE: This happens regardless of the infinite prop value. If the swiper is on the first card, swiping back unexpectedly takes it to the last card instead of stopping at the beginning. This severely impacts usability on iOS.

Steps to Reproduce:

  1. Swipe a card back using the swipe-back gesture.
  2. Observe that the card animates back correctly.
  3. Notice that it skips another card backward without animation and the swiper becomes unresponsive.

Expected Behavior:
The swiper should stop at the previous card with no unexpected jumps and remain responsive to further actions.

Environment:
Expo SDK: 52
Platform: iOS
Library Version: react-native-deck-swiper v2.0.17

This is a critical bug for swipe-back functionality. Any help or fix would be greatly appreciated!
If needed, I can provide additional context or examples to help debug this issue.

Code example:

import {
    useSafeAreaInsets,
} from 'react-native-safe-area-context';
import { Text } from 'react-native-paper';
import { useState, useEffect, useRef } from 'react';
import { View, StyleSheet, Dimensions, Image, Platform } from 'react-native';
import { LinearGradient } from 'expo-linear-gradient';
import Swiper from 'react-native-deck-swiper';

const { width, height } = Dimensions.get('window');

const CardsSwiper = () => {
    const swiperRef = useRef(null);
    const [cards] = useState([
        { id: '1', text: 'Card 1' },
        { id: '2', text: 'Card 2' },
        { id: '3', text: 'Card 3' },
        { id: '4', text: 'Card 4' },
        { id: '5', text: 'Card 5' },
        { id: '6', text: 'Card 6' },
    ]);
    const [measurements, setMeasurements] = useState(null);
    const { top, bottom } = useSafeAreaInsets();
    const headerHeight = Platform.select({ android: top + 56, ios: top + 44 });
    const tabHeight = bottom + 49;

    useEffect(() => {
        setMeasurements({
            width: width,
            height: height - headerHeight - tabHeight,
        });
    }, [top]);

    if (!measurements) {
        return null;
    }

    return (
        <View style={{ width: measurements.width, height: measurements.height }}>
            <Swiper
                ref={swiperRef}
                cards={cards}
                keyExtractor={(card) => card.id}
                renderCard={(card) => (
                    <View style={styles.cardContainer}>
                        <View
                            style={{
                                width: '100%',
                                height: '86%',
                            }}
                        >
                            <Image
                                source={{
                                    uri: 'https://media.gq.com/photos/623df9b425ea3c548d988e1d/master/w_1600%2Cc_limit/GQ0422_Gunna_07.jpg',
                                }}
                                style={{ width: '100%', height: '100%' }}
                                resizeMode="cover"
                            />
                            <Text style={{ color: 'red', position: 'absolute', top: 0 }}>
                                {card.text}
                            </Text>
                        </View>
                        <LinearGradient
                            pointerEvents={'none'}
                            colors={[
                                'transparent',
                                'rgba(0, 0, 0, 0.1)',
                                'rgba(0, 0, 0, 0.2)',
                                'rgba(0, 0, 0, 0.3)',
                                'rgba(0, 0, 0, 0.4)',
                                'rgba(0, 0, 0, 0.5)',
                                'rgba(0, 0, 0, 0.6)',
                                'rgba(0, 0, 0, 0.7)',
                                'rgba(0, 0, 0, 0.8)',
                                'rgba(0, 0, 0, 0.9)',
                                'rgba(0, 0, 0, 0.99)',
                                'black',
                                'black',
                            ]}
                            style={[
                                styles.linearGradient,
                                { height: measurements.height * 0.8 },
                            ]}
                        />
                    </View>
                )}
                onTapCard={() => {
                    swiperRef.current?.swipeBack();
                }}
                swipeBackCard={true}
                stackSize={3}
                stackScale={3}
                stackSeparation={0}
                cardIndex={0}
                showSecondCard={true}
                verticalSwipe={true}
                horizontalSwipe={true}
                cardVerticalMargin={0}
                cardHorizontalMargin={0}
                containerStyle={{
                    width: measurements.width,
                    height: measurements.height,
                }}
                cardStyle={{
                    width: measurements.width,
                    height: measurements.height,
                }}
                backgroundColor={'transparent'}
            />
        </View>
    );
};

const styles = StyleSheet.create({
    cardContainer: {
        flex: 1,
        borderRadius: 10,
        overflow: 'hidden',
        backgroundColor: 'white'
    },
    linearGradient: {
        position: 'absolute',
        bottom: 0,
        width: '100%',
    },
});
@ShmuelPickRanky ShmuelPickRanky changed the title [iOS on expo 52] - Swipe-Back Animates Last Card, Then Jumps Another Card Back Without Animation, Breaking Swiper! [iOS on expo 52] - Swipe-Back Animates Last Card, Then Jumps Another Card Back Without Animation, Breaking Swiper! (Might be the latest version of React Native) Dec 30, 2024
@ShmuelPickRanky
Copy link
Author

Any update on this?

Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
None yet
Projects
None yet
Development

No branches or pull requests

1 participant