React Native:如何临时禁用PanResponder?

时间:2017-05-04 07:40:37

标签: javascript android ios reactjs react-native

以下是用于创建panResponder的实例的代码段:

  constructor( props ) {
    super( props );

    this.position = new Animated.ValueXY();

    this.panResponder = PanResponder.create( {
      onStartShouldSetPanResponder: ( ) => true,
      onPanResponderMove: ( event, gesture ) => {
        this.position.setValue( { x: 0, y: gesture.dy } );
      },
      onPanResponderRelease: ( event, gesture ) => {
        if ( gesture.dy > SWIPE_THRESHOLD ) {
          this.forceSwipe( 'up' );
        } else if ( gesture.dy < -SWIPE_THRESHOLD ) {
          this.forceSwipe( 'down' );
        } else {
          this.resetPosition();
        }
      },
      onMoveShouldSetPanResponderCapture: ( evt, gestureState ) =>
      gestureState.dx !== 0 && gestureState.dy !== 0,
    } );
  }

  getCardStyle() {
    return {
      ...this.position.getLayout(),
    };
  }

以下是我的组件的代码段:

<Animated.View
   key={ item.id }
   style={ [ this.getCardStyle(), styles.cardStyle ] }
   { ...this.panResponder.panHandlers }
>
  <Card
    shouldPanRespond={ this.shouldPanRespond }
     item={ item }
  />
</Animated.View>

卡组件的代码段:

  <TouchableOpacity
    activeOpacity={ 1 }
    style={ cardStyle }
    onPress={ this.onPress }
  >
    <ScrollView contentContainerStyle={ cardStyle }>
      <Image
        resizeMode="cover"
        style={ imageStyle }
        source={ { uri: img_url } }
      />

      <View style={ titleWrapStyle }>
        <Text style={ textStyle }>{ title }</Text>
      </View>
    </ScrollView>
  </TouchableOpacity>

我正在制作绝对定位的卡片。然后可以使用panResponder向上或向下滑动这些卡片。但是,在点击屏幕时,我应该在同一屏幕上显示详细页面。

现在,我可以检测到点击,但每当用户点击/点击时,我想要禁用panResponder,以便无法刷卡,用户可以滚动浏览内容。一旦用户滚动到最后,我想重新启用滑动,即我想启用平移。

我知道onStartShouldSetResponder: () => false禁用panResponder但是我们如何在创建实例后禁用它。我无法在其他地方找到太多。

4 个答案:

答案 0 :(得分:1)

对于想要暂时禁用PanResponder的人,以下是我的解决方案。

像@RaHuL - House Javascript注意到,另一个答案并没有真正解决问题,因为将onStartShouldSetResponder设置为对应于某个状态值的值可能会从true变为false将不会影响PanResponder(至少如果它是用true创建的。

我通过更改onPanResponderMove功能解决了类似的问题。我在州内有一个值:panResponderEnabled,我的函数看起来像这样:

 onPanResponderMove: ( event, gesture ) => {
    if(this.state.panResponderEnabled){
       this.position.setValue( { x: 0, y: gesture.dy } ); 
    }
}

当我不想从PanRespoder读取更改时,我只是将panResponderEnabled设置为false。

答案 1 :(得分:1)

我想我找到了一种使用onStartShouldSetResponder暂时禁用PanResponder的方法。目标是在真与假之间切换,即使它似乎并不响应状态值的更改。如果您使用的是功能组件,则钩子useRef似乎可以解决问题。

const [panResponderEnabled, _setPanResponderEnabled] = useState(true)

const panResponderEnabledRef = useRef(panResponderEnabled)
const setPanResponderEnabled = data => {
  panResponderEnabledRef.current = data
  _setPanResponderEnabled(data)
}

然后,当您创建PanResponder时:

onStartShouldSetPanResponder: () => panResponderEnabledRef.current,

答案 2 :(得分:0)

如果您想要一个非常简单的暴力解决方案,请在动画视图上使用pointerEvents道具。

pointerEvents={stopPan ? 'none' : 'auto'}

这将在stopPan为true时停止视图上的所有触摸事件及其下的所有事件。

答案 3 :(得分:-1)

您可以使用类函数并使用state返回true / false,而不是onStartShouldSetResponder的箭头函数。像这样的东西

 constructor( props ) {
    super( props );
    this.state = {
      panResponderEnabled: true
    }
    this.position = new Animated.ValueXY();

    this.panResponder = PanResponder.create( {
      onStartShouldSetPanResponder: ( ) => true,
      onPanResponderMove: ( event, gesture ) => {
        this.position.setValue( { x: 0, y: gesture.dy } );
      },
      onPanResponderRelease: ( event, gesture ) => {
        if ( gesture.dy > SWIPE_THRESHOLD ) {
          this.forceSwipe( 'up' );
        } else if ( gesture.dy < -SWIPE_THRESHOLD ) {
          this.forceSwipe( 'down' );
        } else {
          this.resetPosition();
        }
      },
      onMoveShouldSetPanResponderCapture: ( evt, gestureState ) =>
      gestureState.dx !== 0 && gestureState.dy !== 0,
    } );
  }

  handlePanResponderStart = () => {
    return this.state.panResponderEnabled;
  }
  onCardPress = () => {
    this.setState({
      panResponderEnabled: false
    });
  }
  getCardStyle() {
    return {
      ...this.position.getLayout(),
    };
  }