Skip to content

Commit

Permalink
feature: allow serialized animated event as onScroll
Browse files Browse the repository at this point in the history
  • Loading branch information
Jberivera committed Dec 13, 2018
1 parent 85b7e68 commit d27d7ed
Showing 1 changed file with 39 additions and 15 deletions.
54 changes: 39 additions & 15 deletions src/carousel/Carousel.js
Original file line number Diff line number Diff line change
Expand Up @@ -143,20 +143,7 @@ export default class Carousel extends Component {

this._getKeyExtractor = this._getKeyExtractor.bind(this);

// Native driver for scroll events
const scrollEventConfig = {
listener: this._onScroll,
useNativeDriver: true
};
this._scrollPos = new Animated.Value(0);
this._onScrollHandler = props.vertical ?
Animated.event(
[{ nativeEvent: { contentOffset: { y: this._scrollPos } } }],
scrollEventConfig
) : Animated.event(
[{ nativeEvent: { contentOffset: { x: this._scrollPos } } }],
scrollEventConfig
);
this._setScrollHandler(props);

// This bool aims at fixing an iOS bug due to scrollTo that triggers onMomentumScrollEnd.
// onMomentumScrollEnd fires this._snapScroll, thus creating an infinite loop.
Expand Down Expand Up @@ -273,6 +260,10 @@ export default class Carousel extends Component {
this._previousFirstItem = nextFirstItem;
this._snapToItem(nextFirstItem, true, true, false, false);
}

if (nextProps.onScroll !== this.props.onScroll) {
this._setScrollHandler(nextProps);
}
}

componentWillUnmount () {
Expand All @@ -299,6 +290,39 @@ export default class Carousel extends Component {
return this._currentContentOffset;
}

_setScrollHandler(props) {
// Native driver for scroll events
const scrollEventConfig = {
listener: this._onScroll,
useNativeDriver: true,
};
this._scrollPos = new Animated.Value(0);
const argMapping = props.vertical
? [{ nativeEvent: { contentOffset: { y: this._scrollPos } } }]
: [{ nativeEvent: { contentOffset: { x: this._scrollPos } } }];

if (props.onScroll && Array.isArray(props.onScroll._argMapping)) {
// Because of a react-native issue https://github.com/facebook/react-native/issues/13294
argMapping.pop();
const [ argMap ] = props.onScroll._argMapping;
if (argMap && argMap.nativeEvent && argMap.nativeEvent.contentOffset) {
// Shares the same animated value passed in props
this._scrollPos =
argMap.nativeEvent.contentOffset.x ||
argMap.nativeEvent.contentOffset.y ||
this._scrollPos;
}
argMapping.push(...props.onScroll._argMapping);
}
this._onScrollHandler = Animated.event(
argMapping,
scrollEventConfig
);

if (this._mounted)
this.forceUpdate();
}

_needsScrollView () {
const { useScrollView } = this.props;
return useScrollView || !AnimatedFlatList || this._shouldUseStackLayout() || this._shouldUseTinderLayout();
Expand Down Expand Up @@ -794,7 +818,7 @@ export default class Carousel extends Component {
this._repositionScroll(nextActiveItem);
}

if (onScroll && event) {
if (typeof onScroll === "function" && event) {
onScroll(event);
}
}
Expand Down

0 comments on commit d27d7ed

Please sign in to comment.