FlatList ScrollView Error on any State Change - Invariant Violation: Changing onViewableItemsChanged on the fly is not supported

前端 未结 7 1251
北荒
北荒 2020-12-31 10:11

onViewableItemsChanged does not seem to work when there is a state change in the app. Is this correct?

Seems like it wouldn\'t be very useful if this

相关标签:
7条回答
  • 2020-12-31 10:24

    Remove your viewabilityConfig prop to a const value outside the render functions as well as your onViewableItemsChanged function

    0 讨论(0)
  • 2020-12-31 10:25

    Move the viewabilityConfig object to the constructor.

    constructor() {
        this.viewabilityConfig = {
            viewAreaCoveragePercentThreshold: 50
        };
    }
    
    render() {
         return(
            <FlatList
                data={this.state.cardData}
                horizontal={true}
                pagingEnabled={true}
                showsHorizontalScrollIndicator={false}
                onViewableItemsChanged={(info) =>console.log(info)}
                viewabilityConfig={this.viewabilityConfig}
                renderItem={({item}) =>
                    <View style={{width: width, borderColor: 'white', borderWidth: 20,}}>
                        <Text>Dogs and Cats</Text>
                    </View>
                }
            />
        )
    }
    
    0 讨论(0)
  • 2020-12-31 10:26

    The error Changing onViewableItemsChanged on the fly is not supported occurs because when you update the state, you are creating a new onViewableItemsChanged function reference, so you are changing it on the fly.

    While the accepted answer may solve the issue with useRef, it is not the correct hook in this case. You should be using useCallback to return a memoized callback and useState to get the current state without needing to create a new reference to the function.

    Here is an example that save all viewed items index on state:

    const MyComp = () => {
      const [cardData] = useState(['a', 'b', 'c', 'd', 'e', 'f', 'g', 'h', 'i']);
      const [viewedItems, setViewedItems] = useState([]);
    
      const handleVieweableItemsChanged = useCallback(({ changed }) => {
        setViewedItems(oldViewedItems => {
          // We can have access to the current state without adding it
          //  to the useCallback dependencies
    
          let newViewedItems = null;
    
          changed.forEach(({ index, isViewable }) => {
            if (index != null && isViewable && !oldViewedItems.includes(index)) {
              
               if (newViewedItems == null) {
                 newViewedItems = [...oldViewedItems];
               }
               newViewedItems.push(index);
            }
          });
    
          // If the items didn't change, we return the old items so
          //  an unnecessary re-render is avoided.
          return newViewedItems == null ? oldViewedItems : newViewedItems;
        });
    
        // Since it has no dependencies, this function is created only once
      }, []);
    
      function renderItem({ index, item }) {
        const viewed = '' + viewedItems.includes(index);
        return (
          <View>
            <Text>Data: {item}, Viewed: {viewed}</Text>
          </View>
        );
      }
    
      return (
        <FlatList
          data={cardData}
          onViewableItemsChanged={handleVieweableItemsChanged}
          viewabilityConfig={this.viewabilityConfig}
          renderItem={renderItem}
        />
      );
    }
    

    You can see it working on Snack.

    0 讨论(0)
  • 2020-12-31 10:31

    this works for me, is there any way to pass an additional argument to onViewRef? Like in the below code how can i pass type argument to onViewRef. Code:

            function getScrollItems(items, isPendingList, type) {
        return (
            <FlatList
                data={items}
                style={{width: wp("100%"), paddingLeft: wp("4%"), paddingRight: wp("10%")}}
                horizontal={true}
                keyExtractor={(item, index) => index.toString()}
                showsHorizontalScrollIndicator={false}
                renderItem={({item, index}) => renderScrollItem(item, index, isPendingList, type)}
                viewabilityConfig={viewConfigRef.current}
                onViewableItemsChanged={onViewRef.current}
            />
        )
    }
    
    0 讨论(0)
  • 2020-12-31 10:32

    You must pass in a function to onViewableItemsChanged that is bound in the constructor of the component and you must set viewabilityConfig as a constant outside of the Flatlist.

    Example:

    class YourComponent extends Component {
    
        constructor() {
            super()
            this.onViewableItemsChanged.bind(this)
        }
    
        onViewableItemsChanged({viewableItems, changed}) {
            console.log('viewableItems', viewableItems)
            console.log('changed', changed)
        }
    
        viewabilityConfig = {viewAreaCoveragePercentThreshold: 50}
    
        render() {
            return(
              <FlatList
                data={this.state.cardData}
                horizontal={true}
                pagingEnabled={true}
                showsHorizontalScrollIndicator={false}
                onViewableItemsChanged={this.onViewableItemsChanged}
                viewabilityConfig={this.viewabilityConfig}
                renderItem={({item}) =>
                    <View style={{width: width, borderColor: 'white', borderWidth: 20,}}>
                        <Text>Dogs and Cats</Text>
                     </View>}
              />
            )
        }
    }
    
    0 讨论(0)
  • 2020-12-31 10:32

    Sombody suggest to use extraData property of Flatlist to let Flatlist notice, that something changed.

    But this didn't work for me, here is what work for me:

    Use key={this.state.orientation} while orientation e.g is "portrait" or "landscape"... it can be everything you want, but it had to change, if the orientation changed. If Flatlist notice that the key-property is changed, it rerenders.

    works for react-native 0.56

    0 讨论(0)
提交回复
热议问题