Files
mattermost-mobile/app/components/swiper.js

314 lines
8.5 KiB
JavaScript

// Copyright (c) 2015-present Mattermost, Inc. All Rights Reserved.
// See LICENSE.txt for license information.
import React, {PureComponent} from 'react';
import PropTypes from 'prop-types';
import {
View,
ScrollView,
ViewPagerAndroid,
Platform,
StyleSheet,
InteractionManager,
} from 'react-native';
export default class Swiper extends PureComponent {
static propTypes = {
activeDotColor: PropTypes.string,
children: PropTypes.node.isRequired,
dotColor: PropTypes.string,
initialPage: PropTypes.number,
keyboardShouldPersistTaps: PropTypes.string,
onIndexChanged: PropTypes.func,
paginationStyle: PropTypes.oneOfType([
PropTypes.object,
PropTypes.number,
]),
scrollEnabled: PropTypes.bool,
showsPagination: PropTypes.bool,
style: PropTypes.oneOfType([
PropTypes.object,
PropTypes.number,
]),
width: PropTypes.number,
onScrollBegin: PropTypes.func,
};
static defaultProps = {
initialPage: 0,
keyboardShouldPersistTaps: 'handled',
onIndexChanged: () => null,
scrollEnabled: true,
showsPagination: true,
onScrollBegin: () => true,
};
constructor(props) {
super(props);
this.runOnLayout = true;
this.offset = props.width * props.initialPage;
this.state = this.initialState(props);
}
componentWillReceiveProps(nextProps) {
if (this.props.width !== nextProps.width) {
this.scrollByWidth(nextProps.width);
}
}
componentDidUpdate(prevProps, prevState) {
// If the index has changed, we notify the parent via the onIndexChanged callback
if (this.state.index !== prevState.index) {
this.props.onIndexChanged(this.state.index);
}
}
initialState = (props) => {
const index = props.initialPage;
return {
index,
total: React.Children.count(props.children),
};
};
onLayout = (e) => {
this.scrollByWidth(e.nativeEvent.layout.width);
};
onScrollBegin = () => {
this.props.onScrollBegin();
};
onScrollEnd = (e) => {
// making our events coming from android compatible to updateIndex logic
if (!e.nativeEvent.contentOffset) {
e.nativeEvent.contentOffset = {x: e.nativeEvent.position * this.props.width};
}
// get the index
this.updateIndex(e.nativeEvent.contentOffset.x);
};
onPageScrollStateChanged = (e) => {
switch (e) {
case 'dragging':
this.props.onScrollBegin();
break;
}
};
scrollByWidth = (width) => {
this.offset = width * this.state.index;
if (Platform.OS === 'ios') {
setTimeout(() => {
if (this.scrollView) {
this.scrollView.scrollTo({x: width * this.state.index, animated: false});
}
}, 0);
} else if (this.scrollView) {
this.scrollView.setPageWithoutAnimation(this.state.index);
}
};
scrollToStart = () => {
if (Platform.OS === 'ios') {
InteractionManager.runAfterInteractions(() => {
this.scrollView.scrollTo({x: 0, animated: false});
});
}
};
refScrollView = (view) => {
this.scrollView = view;
};
renderScrollView = (pages) => {
const {
keyboardShouldPersistTaps,
scrollEnabled,
} = this.props;
if (Platform.OS === 'ios') {
return (
<ScrollView
ref={this.refScrollView}
bounces={false}
horizontal={true}
removeClippedSubviews={true}
automaticallyAdjustContentInsets={true}
showsHorizontalScrollIndicator={false}
showsVerticalScrollIndicator={false}
contentContainerStyle={[styles.wrapperIOS, this.props.style]}
onScrollBeginDrag={this.onScrollBegin}
onMomentumScrollEnd={this.onScrollEnd}
pagingEnabled={scrollEnabled}
keyboardShouldPersistTaps={keyboardShouldPersistTaps}
scrollEnabled={scrollEnabled}
>
{pages}
</ScrollView>
);
}
return (
<ViewPagerAndroid
ref={this.refScrollView}
initialPage={this.props.initialPage}
onPageSelected={this.onScrollEnd}
onPageScrollStateChanged={this.onPageScrollStateChanged}
scrollEnabled={scrollEnabled}
key={pages.length}
style={[styles.wrapperAndroid, this.props.style]}
>
{pages}
</ViewPagerAndroid>
);
};
renderPagination = () => {
if (this.state.total <= 1 || !this.props.showsPagination) {
return null;
}
const dots = [];
const activeDot = (
<View
style={[
styles.dotStyle,
{backgroundColor: this.props.activeDotColor || '#007aff'},
]}
/>
);
const dot = (
<View
style={[
styles.dotStyle,
{backgroundColor: this.props.dotColor || 'rgba(0,0,0,.2)'},
]}
/>
);
for (let i = 0; i < this.state.total; i++) {
if (i === this.state.index) {
dots.push(React.cloneElement(activeDot, {key: i}));
} else {
dots.push(React.cloneElement(dot, {key: i}));
}
}
return (
<View
pointerEvents='none'
style={[styles.pagination, this.props.paginationStyle]}
>
{dots}
</View>
);
};
scrollToIndex = (index, animated) => {
if (this.state.total < 2) {
return;
}
if (Platform.OS === 'ios') {
this.scrollView.scrollTo({x: (index * this.props.width), animated});
} else {
this.scrollView[animated ? 'setPage' : 'setPageWithoutAnimation'](index);
}
// trigger onScrollEnd manually in android or if not animated
if (!animated || Platform.OS === 'android') {
setImmediate(() => {
this.onScrollEnd({
nativeEvent: {
position: index,
},
});
});
}
};
updateIndex = (offset) => {
let index = this.state.index;
const diff = offset - this.offset;
if (!diff) {
return;
}
index = parseInt(index + Math.round(diff / this.props.width), 10);
this.offset = offset;
this.setState({index});
};
render() {
const {
children,
width,
} = this.props;
const pages = React.Children.map(children, (page, i) => {
const pageStyle = page ? {width} : {width: 0};
return (
<View
style={[styles.slide, pageStyle]}
key={i}
>
{page}
</View>
);
});
return (
<View
style={[styles.container]}
onLayout={this.onLayout}
>
{this.renderScrollView(pages)}
{this.renderPagination()}
</View>
);
}
}
const styles = StyleSheet.create({
container: {
backgroundColor: 'transparent',
position: 'relative',
flex: 1,
},
wrapperIOS: {
backgroundColor: 'transparent',
},
wrapperAndroid: {
backgroundColor: 'transparent',
flex: 1,
},
slide: {
flex: 1,
width: '100%',
},
pagination: {
position: 'absolute',
bottom: 25,
left: 0,
right: 0,
flexDirection: 'row',
flex: 1,
justifyContent: 'center',
alignItems: 'center',
backgroundColor: 'transparent',
marginBottom: 13,
},
dotStyle: {
width: 8,
height: 8,
borderRadius: 4,
marginLeft: 4,
marginRight: 4,
marginTop: 3,
marginBottom: 3,
},
});