mirror of
https://github.com/facebook/react-native.git
synced 2025-11-01 09:14:26 +00:00
8351a5d186
Summary: Pull Request resolved: https://github.com/facebook/react-native/pull/53342 Commands ``` scripts/flow/tool add-comments --comment 'Error discovered during Constant Condition roll out. See https://fburl.com/workplace/4oq3zi07.' . ``` ``` arc f ``` drop-conflicts Reviewed By: SamChou19815 Differential Revision: D80487235 fbshipit-source-id: 9e7c1a2641ddc0da0400fa1aff598b112a0434d5
1517 lines
44 KiB
JavaScript
1517 lines
44 KiB
JavaScript
/**
|
|
* Copyright (c) Meta Platforms, Inc. and affiliates.
|
|
*
|
|
* This source code is licensed under the MIT license found in the
|
|
* LICENSE file in the root directory of this source tree.
|
|
*
|
|
* @flow strict-local
|
|
* @format
|
|
*/
|
|
|
|
import type {RNTesterModuleExample} from '../../types/RNTesterTypes';
|
|
import type {ViewStyleProp} from 'react-native/Libraries/StyleSheet/StyleSheet';
|
|
|
|
import RNTesterText from '../../components/RNTesterText';
|
|
import ScrollViewPressableStickyHeaderExample from './ScrollViewPressableStickyHeaderExample';
|
|
import nullthrows from 'nullthrows';
|
|
import * as React from 'react';
|
|
import {cloneElement, useCallback, useRef, useState} from 'react';
|
|
import {
|
|
Platform,
|
|
RefreshControl,
|
|
ScrollView,
|
|
StyleSheet,
|
|
Text,
|
|
TextInput,
|
|
TouchableOpacity,
|
|
View,
|
|
} from 'react-native';
|
|
|
|
class EnableDisableList extends React.Component<{}, {scrollEnabled: boolean}> {
|
|
state: {scrollEnabled: boolean} = {
|
|
scrollEnabled: true,
|
|
};
|
|
render(): React.Node {
|
|
return (
|
|
<View>
|
|
<ScrollView
|
|
automaticallyAdjustContentInsets={false}
|
|
nestedScrollEnabled
|
|
style={styles.scrollView}
|
|
scrollEnabled={this.state.scrollEnabled}>
|
|
{ITEMS.map(createItemRow)}
|
|
</ScrollView>
|
|
<RNTesterText>
|
|
Scrolling enabled = {this.state.scrollEnabled.toString()}
|
|
</RNTesterText>
|
|
<Button
|
|
label="Disable Scrolling"
|
|
onPress={() => {
|
|
this.setState({scrollEnabled: false});
|
|
}}
|
|
/>
|
|
<Button
|
|
label="Enable Scrolling"
|
|
onPress={() => {
|
|
this.setState({scrollEnabled: true});
|
|
}}
|
|
/>
|
|
</View>
|
|
);
|
|
}
|
|
}
|
|
|
|
let AppendingListItemCount = 6;
|
|
class AppendingList extends React.Component<
|
|
{},
|
|
{items: Array<ExactReactElement_DEPRECATED<Class<Item>>>},
|
|
> {
|
|
state: {items: Array<ExactReactElement_DEPRECATED<Class<Item>>>} = {
|
|
items: [...Array(AppendingListItemCount)].map((_, ii) => (
|
|
<Item msg={`Item ${ii}`} />
|
|
)),
|
|
};
|
|
render(): React.Node {
|
|
return (
|
|
<View>
|
|
<ScrollView
|
|
automaticallyAdjustContentInsets={false}
|
|
maintainVisibleContentPosition={{
|
|
minIndexForVisible: 0,
|
|
autoscrollToTopThreshold: 10,
|
|
}}
|
|
nestedScrollEnabled
|
|
style={styles.scrollView}>
|
|
{this.state.items.map(item =>
|
|
// $FlowFixMe[prop-missing] React.Element internal inspection
|
|
cloneElement(item, {key: item.props.msg}),
|
|
)}
|
|
</ScrollView>
|
|
<ScrollView
|
|
horizontal={true}
|
|
automaticallyAdjustContentInsets={false}
|
|
maintainVisibleContentPosition={{
|
|
minIndexForVisible: 1,
|
|
autoscrollToTopThreshold: 10,
|
|
}}
|
|
style={[styles.scrollView, styles.horizontalScrollView]}>
|
|
{this.state.items.map(item =>
|
|
// $FlowFixMe[prop-missing] React.Element internal inspection
|
|
cloneElement(item, {key: item.props.msg, style: null}),
|
|
)}
|
|
</ScrollView>
|
|
<View style={styles.row}>
|
|
<Button
|
|
label="Add to top"
|
|
onPress={() => {
|
|
this.setState(state => {
|
|
const idx = AppendingListItemCount++;
|
|
return {
|
|
items: [
|
|
<Item style={{paddingTop: idx * 5}} msg={`Item ${idx}`} />,
|
|
].concat(state.items),
|
|
};
|
|
});
|
|
}}
|
|
/>
|
|
<Button
|
|
label="Remove top"
|
|
onPress={() => {
|
|
this.setState(state => ({
|
|
items: state.items.slice(1),
|
|
}));
|
|
}}
|
|
/>
|
|
<Button
|
|
label="Change height top"
|
|
onPress={() => {
|
|
this.setState(state => ({
|
|
items: [
|
|
cloneElement(state.items[0], {
|
|
style: {paddingBottom: Math.random() * 40},
|
|
}),
|
|
].concat(state.items.slice(1)),
|
|
}));
|
|
}}
|
|
/>
|
|
</View>
|
|
<View style={styles.row}>
|
|
<Button
|
|
label="Add to end"
|
|
onPress={() => {
|
|
this.setState(state => ({
|
|
items: state.items.concat(
|
|
<Item msg={`Item ${AppendingListItemCount++}`} />,
|
|
),
|
|
}));
|
|
}}
|
|
/>
|
|
<Button
|
|
label="Remove end"
|
|
onPress={() => {
|
|
this.setState(state => ({
|
|
items: state.items.slice(0, -1),
|
|
}));
|
|
}}
|
|
/>
|
|
<Button
|
|
label="Change height end"
|
|
onPress={() => {
|
|
this.setState(state => ({
|
|
items: state.items.slice(0, -1).concat(
|
|
cloneElement(state.items[state.items.length - 1], {
|
|
style: {paddingBottom: Math.random() * 40},
|
|
}),
|
|
),
|
|
}));
|
|
}}
|
|
/>
|
|
</View>
|
|
</View>
|
|
);
|
|
}
|
|
}
|
|
|
|
function CenterContentList(): React.Node {
|
|
return (
|
|
<ScrollView
|
|
nestedScrollEnabled
|
|
style={styles.scrollView}
|
|
centerContent={true}>
|
|
<Text>This should be in center.</Text>
|
|
</ScrollView>
|
|
);
|
|
}
|
|
|
|
function ContentOffsetList(): React.Node {
|
|
return (
|
|
<ScrollView
|
|
style={[styles.scrollView, {height: 100}]}
|
|
horizontal={true}
|
|
contentOffset={{x: 100, y: 0}}>
|
|
{ITEMS.map(createItemRow)}
|
|
</ScrollView>
|
|
);
|
|
}
|
|
|
|
function ScrollViewScrollToExample(): React.Node {
|
|
let _scrollView: ?React.ElementRef<typeof ScrollView>;
|
|
const [scrolledToTop, setScrolledToTop] = useState(false);
|
|
const textStyle = {color: 'blue', marginBottom: 10, textAlign: 'center'};
|
|
return (
|
|
<View>
|
|
{scrolledToTop ? (
|
|
/* $FlowFixMe[incompatible-type] Natural Inference rollout. See
|
|
* https://fburl.com/workplace/6291gfvu */
|
|
<RNTesterText style={textStyle}>scrolledToTop invoked</RNTesterText>
|
|
) : null}
|
|
<ScrollView
|
|
accessibilityRole="grid"
|
|
ref={scrollView => {
|
|
_scrollView = scrollView;
|
|
}}
|
|
automaticallyAdjustContentInsets={false}
|
|
nestedScrollEnabled
|
|
onScroll={() => {
|
|
console.log('onScroll!');
|
|
setScrolledToTop(false);
|
|
}}
|
|
onScrollToTop={() => {
|
|
setScrolledToTop(true);
|
|
}}
|
|
scrollEventThrottle={200}
|
|
style={[styles.scrollView, {height: 200}]}
|
|
testID="scroll_vertical">
|
|
{ITEMS.map(createItemRow)}
|
|
</ScrollView>
|
|
<Button
|
|
label="Scroll to top"
|
|
onPress={() => {
|
|
nullthrows<$FlowFixMe>(_scrollView).scrollTo({y: 0});
|
|
}}
|
|
testID="scroll_to_top_button"
|
|
/>
|
|
<Button
|
|
label="Scroll to bottom"
|
|
onPress={() => {
|
|
nullthrows<$FlowFixMe>(_scrollView).scrollToEnd({animated: true});
|
|
}}
|
|
testID="scroll_to_bottom_button"
|
|
/>
|
|
<Button
|
|
label="Flash scroll indicators"
|
|
onPress={() => {
|
|
nullthrows<$FlowFixMe>(_scrollView).flashScrollIndicators();
|
|
}}
|
|
testID="flash_scroll_indicators_button"
|
|
/>
|
|
</View>
|
|
);
|
|
}
|
|
|
|
exports.displayName = 'ScrollViewExample';
|
|
exports.title = 'ScrollView';
|
|
exports.documentationURL = 'https://reactnative.dev/docs/scrollview';
|
|
exports.category = 'Basic';
|
|
exports.description =
|
|
'Component that enables scrolling through child components';
|
|
const examples: Array<RNTesterModuleExample> = [
|
|
{
|
|
name: 'scrollTo',
|
|
title: '<ScrollView>\n',
|
|
description:
|
|
'To make content scrollable, wrap it within a <ScrollView> component',
|
|
render: ScrollViewScrollToExample,
|
|
},
|
|
{
|
|
name: 'horizontalScrollTo',
|
|
title: '<ScrollView> (horizontal = true)\n',
|
|
description:
|
|
"You can display <ScrollView>'s child components horizontally rather than vertically",
|
|
render(): React.Node {
|
|
return (
|
|
<View>
|
|
<HorizontalScrollView direction="ltr" />
|
|
</View>
|
|
);
|
|
},
|
|
},
|
|
{
|
|
name: 'horizontalScrollToRTL',
|
|
title: '<ScrollView> (horizontal = true) in RTL\n',
|
|
description:
|
|
"You can display <ScrollView>'s child components horizontally rather than vertically",
|
|
render(): React.Node {
|
|
return (
|
|
<View>
|
|
<HorizontalScrollView direction="rtl" />
|
|
</View>
|
|
);
|
|
},
|
|
},
|
|
{
|
|
name: 'stubbyHorizontalScrollView',
|
|
title: '<ScrollView> (horizontal = true) in RTL not filling content\n',
|
|
description:
|
|
'A horizontal RTL ScrollView whose content is smaller thatn its containner',
|
|
render(): React.Node {
|
|
return (
|
|
<View testID="stubby-horizontal-rtl-scrollview">
|
|
<HorizontalScrollView direction="rtl" itemCount={1} />
|
|
</View>
|
|
);
|
|
},
|
|
},
|
|
{
|
|
title: '<ScrollView> enable & disable\n',
|
|
description: 'ScrollView scrolling behaviour can be disabled and enabled',
|
|
render(): React.Node {
|
|
return <EnableDisableList />;
|
|
},
|
|
},
|
|
{
|
|
title: '<ScrollView> Content\n',
|
|
description: 'Adjust properties of content inside ScrollView.',
|
|
render(): React.Node {
|
|
return <ContentExample />;
|
|
},
|
|
},
|
|
{
|
|
title: '<ScrollView> Deceleration Rate\n',
|
|
description:
|
|
'Determines how quickly the scroll view decelerates after the user lifts their finger.',
|
|
render(): React.Node {
|
|
return <DecelerationRateExample />;
|
|
},
|
|
},
|
|
{
|
|
title: '<ScrollView> Enable & Disable Scrolling Behavior\n',
|
|
description:
|
|
'DirectionalLockEnabled (iOS), disableIntervalMomentum, disableScrollViewPanResponder can be enabled or disabled.',
|
|
render(): React.Node {
|
|
return <DisableEnable />;
|
|
},
|
|
},
|
|
{
|
|
name: 'invertStickyHeaders',
|
|
title: '<ScrollView> Invert Sticky Headers\n',
|
|
description:
|
|
'If sticky headers should stick at the bottom instead of the top of the ScrollView. This is usually used with inverted ScrollViews.',
|
|
render(): React.Node {
|
|
return <InvertStickyHeaders />;
|
|
},
|
|
},
|
|
{
|
|
name: 'multipleStickyHeaders',
|
|
title: '<ScrollView> Multiple Sticky Headers\n',
|
|
description:
|
|
'Scroll down to see 3 sticky headers stick when they get to the top.',
|
|
render(): React.Node {
|
|
return <MultipleStickyHeaders />;
|
|
},
|
|
},
|
|
{
|
|
name: 'pressableStickyHeader',
|
|
title: '<ScrollView> Pressable Sticky Header\n',
|
|
description:
|
|
'Press the blue box to toggle it between blue and yellow. The box should remain Pressable after scrolling.',
|
|
render(): React.Node {
|
|
return (
|
|
<View style={{height: 400}}>
|
|
<ScrollViewPressableStickyHeaderExample />
|
|
</View>
|
|
);
|
|
},
|
|
},
|
|
{
|
|
name: 'keyboardShouldPersistTaps',
|
|
title: '<ScrollView> Keyboard Options\n',
|
|
description:
|
|
'Toggle the keyboard using the search bar and determine keyboard behavior in response to drag and tap.',
|
|
render(): React.Node {
|
|
return <KeyboardExample />;
|
|
},
|
|
},
|
|
{
|
|
title: '<ScrollView> OnContentSizeChange\n',
|
|
description:
|
|
'The text below will change when scrollable content view of the ScrollView changes.',
|
|
render(): React.Node {
|
|
return <OnContentSizeChange />;
|
|
},
|
|
},
|
|
{
|
|
title: '<ScrollView> OnMomentumScroll\n',
|
|
description:
|
|
'An alert will be called when the momentum scroll starts or ends.',
|
|
render(): React.Node {
|
|
return <OnMomentumScroll />;
|
|
},
|
|
},
|
|
{
|
|
title: '<ScrollView> OnScroll Options\n',
|
|
description:
|
|
'Change the behavior of onScroll using these options: onScrollBeginDrag, onScrollEndDrag, onScrollToTop (iOS), and overScrollMode (Android).',
|
|
render(): React.Node {
|
|
return <OnScrollOptions />;
|
|
},
|
|
},
|
|
{
|
|
title: '<ScrollView> RefreshControl\n',
|
|
description: 'Pull down to see RefreshControl indicator.',
|
|
render(): React.Node {
|
|
return <RefreshControlExample />;
|
|
},
|
|
},
|
|
{
|
|
title: '<ScrollView> Remove Clipped Subviews\n',
|
|
description:
|
|
'When true, offscreen child views (whose overflow value is hidden) are removed from their native backing superview when offscreen.',
|
|
render(): React.Node {
|
|
return <RemoveClippedSubviews />;
|
|
},
|
|
},
|
|
{
|
|
title: '<ScrollView> Scroll Indicator\n',
|
|
description: 'Adjust properties of the scroll indicator.',
|
|
render(): React.Node {
|
|
return <ScrollIndicatorExample />;
|
|
},
|
|
},
|
|
{
|
|
title: '<ScrollView> SnapTo Options\n',
|
|
description: 'Adjust properties of snapping to the scroll view.',
|
|
render(): React.Node {
|
|
return <SnapToOptions />;
|
|
},
|
|
},
|
|
{
|
|
title: '<ScrollView> (contentOffset = {x: 100, y: 0})\n',
|
|
description: 'Initial contentOffset can be set on ScrollView.',
|
|
render(): React.Node {
|
|
return <ContentOffsetList />;
|
|
},
|
|
},
|
|
{
|
|
title: '<ScrollView> smooth bi-directional content loading\n',
|
|
description:
|
|
'The `maintainVisibleContentPosition` prop allows insertions to either end of the content ' +
|
|
'without causing the visible content to jump. Re-ordering is not supported.',
|
|
render() {
|
|
return <AppendingList />;
|
|
},
|
|
},
|
|
{
|
|
name: 'clipToPaddingBox',
|
|
title: '<ScrollView> clip to padding box\n',
|
|
description:
|
|
'Children should be clipped to the padding box of the ScrollView',
|
|
render() {
|
|
return <ClippingExampleVertical />;
|
|
},
|
|
},
|
|
{
|
|
name: 'clipToPaddingBoxHorizontal',
|
|
title: '<ScrollView> clip to padding box (horizontal = true)\n',
|
|
description:
|
|
'Children should be clipped to the padding box of the horizontal ScrollView',
|
|
render() {
|
|
return <ClippingExampleHorizontal />;
|
|
},
|
|
},
|
|
{
|
|
name: 'touchableChildrenOverflowingContainerHorizontal',
|
|
title:
|
|
'<ScrollView> touchable children overflow content container (horizontal = true)\n',
|
|
description:
|
|
"Children that overflow ScrollView's content container should still receive touch events",
|
|
render() {
|
|
return <ChildrenWithTouchEventsOverflowingContainerHorizontal />;
|
|
},
|
|
},
|
|
];
|
|
|
|
if (Platform.OS === 'ios') {
|
|
examples.push({
|
|
title: '<ScrollView> (centerContent = true)\n',
|
|
description:
|
|
'ScrollView puts its content in the center if the content is smaller than scroll view',
|
|
render(): React.Node {
|
|
return <CenterContentList />;
|
|
},
|
|
});
|
|
examples.push({
|
|
title: '<ScrollView> Always Bounces\n',
|
|
description: 'Always bounce vertically or horizontally.',
|
|
render(): React.Node {
|
|
return (
|
|
<>
|
|
<RNTesterText style={styles.text}>Vertical</RNTesterText>
|
|
<BouncesExampleVertical />
|
|
<RNTesterText style={styles.text}>Horizontal</RNTesterText>
|
|
<BouncesExampleHorizontal />
|
|
</>
|
|
);
|
|
},
|
|
});
|
|
examples.push({
|
|
title: '<ScrollView> Bounces & Bounces Zoom\n',
|
|
description: 'There are different options for bouncing behavior.',
|
|
render(): React.Node {
|
|
return <BouncesExample />;
|
|
},
|
|
});
|
|
examples.push({
|
|
title: '<ScrollView> Indicator Style\n',
|
|
description: 'There are different options for indicator style colors.',
|
|
render(): React.Node {
|
|
return <IndicatorStyle />;
|
|
},
|
|
});
|
|
examples.push({
|
|
title: '<ScrollView> Maximum & Minimum Zoom Scale\n',
|
|
description: 'Set the maximum and minimum allowed zoom scale.',
|
|
render(): React.Node {
|
|
return <MaxMinZoomScale />;
|
|
},
|
|
});
|
|
examples.push({
|
|
title: '<ScrollView> Maximum & Minimum Zoom Scale\n',
|
|
description: 'Set the maximum and minimum allowed zoom scale.',
|
|
render(): React.Node {
|
|
return <MaxMinZoomScale />;
|
|
},
|
|
});
|
|
examples.push({
|
|
title: '<ScrollView> ScrollTo Options\n',
|
|
description:
|
|
'Toggle scrollToOverflowEnabled and scrollsToTop. When scrollToOverflowEnabled is true, the scroll view can be programmatically scrolled beyond its content size. When scrollsToTop is true, the scroll view scrolls to top when the status bar is tapped.',
|
|
render(): React.Node {
|
|
return <ScrollToOptions />;
|
|
},
|
|
});
|
|
} else if (Platform.OS === 'android') {
|
|
examples.push({
|
|
title: '<ScrollView> EndFillColor & FadingEdgeLength\n',
|
|
description: 'Toggle to set endFillColor and fadingEdgeLength.',
|
|
render(): React.Node {
|
|
return <EndFillColorFadingEdgeLen />;
|
|
},
|
|
});
|
|
examples.push({
|
|
title: '<ScrollView> persistentScrollBar\n',
|
|
description: 'Toggle to set persistentScrollbar option.',
|
|
render(): React.Node {
|
|
return <AndroidScrollBarOptions />;
|
|
},
|
|
});
|
|
}
|
|
exports.examples = examples;
|
|
|
|
const AndroidScrollBarOptions = () => {
|
|
const [persistentScrollBar, setPersistentScrollBar] = useState(false);
|
|
return (
|
|
<View>
|
|
<ScrollView
|
|
style={[styles.scrollView, {height: 200}]}
|
|
nestedScrollEnabled
|
|
persistentScrollbar={persistentScrollBar}>
|
|
{ITEMS.map(createItemRow)}
|
|
</ScrollView>
|
|
<Button
|
|
label={'persistentScrollBar: ' + persistentScrollBar.toString()}
|
|
onPress={() => setPersistentScrollBar(!persistentScrollBar)}
|
|
/>
|
|
</View>
|
|
);
|
|
};
|
|
|
|
const HorizontalScrollView = (props: {
|
|
direction: 'ltr' | 'rtl',
|
|
itemCount?: number,
|
|
}) => {
|
|
const {direction} = props;
|
|
const scrollRef = useRef<?React.ElementRef<typeof ScrollView>>();
|
|
const title = direction === 'ltr' ? 'LTR Layout' : 'RTL Layout';
|
|
const items =
|
|
props.itemCount == null ? ITEMS : ITEMS.slice(0, props.itemCount);
|
|
|
|
return (
|
|
<View style={{direction}}>
|
|
<RNTesterText style={styles.text}>{title}</RNTesterText>
|
|
{/* $FlowFixMe[incompatible-use] */}
|
|
<ScrollView
|
|
ref={scrollRef}
|
|
automaticallyAdjustContentInsets={false}
|
|
horizontal={true}
|
|
style={[styles.scrollView, styles.horizontalScrollView]}
|
|
testID={'scroll_horizontal'}>
|
|
{items.map(createItemRow)}
|
|
</ScrollView>
|
|
<Button
|
|
label="Scroll to start"
|
|
onPress={() => {
|
|
nullthrows<$FlowFixMe>(scrollRef.current).scrollTo({x: 0});
|
|
}}
|
|
testID={'scroll_to_start_button'}
|
|
/>
|
|
<Button
|
|
label="Scroll to end"
|
|
onPress={() => {
|
|
nullthrows<$FlowFixMe>(scrollRef.current).scrollToEnd({
|
|
animated: true,
|
|
});
|
|
}}
|
|
testID={'scroll_to_end_button'}
|
|
/>
|
|
</View>
|
|
);
|
|
};
|
|
|
|
const EndFillColorFadingEdgeLen = () => {
|
|
const [endFillColor, setEndFillColor] = useState('');
|
|
const [fadingEdgeLen, setFadingEdgeLen] = useState(0);
|
|
return (
|
|
<View>
|
|
<ScrollView
|
|
style={[styles.scrollView, {height: 200}]}
|
|
endFillColor={endFillColor}
|
|
fadingEdgeLength={fadingEdgeLen}
|
|
nestedScrollEnabled>
|
|
{ITEMS.map(createItemRow)}
|
|
</ScrollView>
|
|
<Button
|
|
label={endFillColor === '' ? 'setEndFillColor' : 'resetEndFillColor'}
|
|
onPress={() =>
|
|
endFillColor === '' ? setEndFillColor('#A9DFD0') : setEndFillColor('')
|
|
}
|
|
/>
|
|
<Button
|
|
label={fadingEdgeLen === 0 ? 'setFadingEdgeLen' : 'resetFadingEdgeLen'}
|
|
onPress={() =>
|
|
fadingEdgeLen === 0 ? setFadingEdgeLen(300) : setFadingEdgeLen(0)
|
|
}
|
|
/>
|
|
</View>
|
|
);
|
|
};
|
|
|
|
const SnapToOptions = () => {
|
|
const [snapToAlignment, setSnapToAlignment] = useState('start');
|
|
const snapToAlignmentModes = ['start', 'center', 'end'];
|
|
const [snapToEnd, setSnapToEnd] = useState(true);
|
|
const [snapToInterval, setSnapToInterval] = useState(0);
|
|
const [snapToOffsets, setSnapToOffsets] = useState<Array<number>>([]);
|
|
const [snapToStart, setSnapToStart] = useState(true);
|
|
|
|
return (
|
|
<View>
|
|
{/* $FlowFixMe[incompatible-use] Natural Inference rollout. See
|
|
* https://fburl.com/workplace/6291gfvu */}
|
|
<ScrollView
|
|
style={[styles.scrollView, {height: 200}]}
|
|
snapToAlignment={snapToAlignment}
|
|
snapToEnd={snapToEnd}
|
|
snapToInterval={snapToInterval}
|
|
snapToOffsets={snapToOffsets}
|
|
snapToStart={snapToStart}
|
|
nestedScrollEnabled>
|
|
{ITEMS.map(createItemRow)}
|
|
</ScrollView>
|
|
{Platform.OS === 'ios' ? (
|
|
<>
|
|
<RNTesterText style={styles.rowTitle}>
|
|
Select Snap to Alignment Mode
|
|
</RNTesterText>
|
|
<View style={styles.row}>
|
|
{snapToAlignmentModes.map(label => (
|
|
<Button
|
|
active={snapToAlignment === label}
|
|
key={label}
|
|
label={label}
|
|
onPress={() => setSnapToAlignment(label)}
|
|
/>
|
|
))}
|
|
</View>
|
|
</>
|
|
) : null}
|
|
<Button
|
|
label={'snapToEnd: ' + snapToEnd.toString()}
|
|
onPress={() => setSnapToEnd(!snapToEnd)}
|
|
/>
|
|
<Button
|
|
label={'snapToStart: ' + snapToStart.toString()}
|
|
onPress={() => setSnapToStart(!snapToStart)}
|
|
/>
|
|
<Button
|
|
label={
|
|
snapToInterval === 0 ? 'setSnapToInterval' : 'reset snapToInterval'
|
|
}
|
|
onPress={() =>
|
|
snapToInterval === 0 ? setSnapToInterval(2) : setSnapToInterval(0)
|
|
}
|
|
/>
|
|
<Button
|
|
label={
|
|
/* $FlowFixMe[invalid-compare] Error discovered during Constant
|
|
* Condition roll out. See https://fburl.com/workplace/4oq3zi07. */
|
|
snapToOffsets === [] ? 'setSnapToOffsets' : 'reset snapToOffsets'
|
|
}
|
|
onPress={() =>
|
|
/* $FlowFixMe[invalid-compare] Error discovered during Constant
|
|
* Condition roll out. See https://fburl.com/workplace/4oq3zi07. */
|
|
snapToOffsets === []
|
|
? setSnapToOffsets([2, 4, 6, 8, 10])
|
|
: setSnapToOffsets([])
|
|
}
|
|
/>
|
|
</View>
|
|
);
|
|
};
|
|
|
|
const ScrollToOptions = () => {
|
|
const [scrollToOverflowEnabled, setScrollToOverflowEnabled] = useState(false);
|
|
const [scrollsToTop, setScrollsToTop] = useState(true);
|
|
return (
|
|
<View>
|
|
<ScrollView
|
|
style={[styles.scrollView, {height: 200}]}
|
|
scrollToOverflowEnabled={scrollToOverflowEnabled}
|
|
scrollsToTop={scrollsToTop}
|
|
nestedScrollEnabled>
|
|
{ITEMS.map(createItemRow)}
|
|
</ScrollView>
|
|
<Button
|
|
label={'scrollToOverflowEnabled: ' + scrollToOverflowEnabled.toString()}
|
|
onPress={() => setScrollToOverflowEnabled(!scrollToOverflowEnabled)}
|
|
/>
|
|
<Button
|
|
label={'scrollsToTop: ' + scrollsToTop.toString()}
|
|
onPress={() => setScrollsToTop(!scrollsToTop)}
|
|
/>
|
|
</View>
|
|
);
|
|
};
|
|
|
|
const ScrollIndicatorExample = () => {
|
|
const [scrollIndicatorInsets, setScrollIndicatorInsets] = useState<null | {
|
|
bottom: number,
|
|
left: number,
|
|
right: number,
|
|
top: number,
|
|
}>(null);
|
|
const [showsHorizontalScrollIndic, setShowsHorizontalScrollIndic] =
|
|
useState(true);
|
|
const [showsVerticalScrollIndic, setShowsVerticalScrollIndic] =
|
|
useState(true);
|
|
return (
|
|
<View>
|
|
<ScrollView
|
|
style={[styles.scrollView, {height: 200}]}
|
|
contentInset={{top: 10, bottom: 10, left: 10, right: 10}}
|
|
scrollIndicatorInsets={scrollIndicatorInsets}
|
|
showsHorizontalScrollIndicator={showsHorizontalScrollIndic}
|
|
showsVerticalScrollIndicator={showsVerticalScrollIndic}
|
|
nestedScrollEnabled>
|
|
{ITEMS.map(createItemRow)}
|
|
</ScrollView>
|
|
<Button
|
|
label={
|
|
scrollIndicatorInsets == null
|
|
? 'setScrollIndicatorInsets'
|
|
: 'Reset scrollIndicatorInsets'
|
|
}
|
|
onPress={() =>
|
|
scrollIndicatorInsets == null
|
|
? setScrollIndicatorInsets({
|
|
top: 10,
|
|
left: 10,
|
|
bottom: 10,
|
|
right: 10,
|
|
})
|
|
: setScrollIndicatorInsets(null)
|
|
}
|
|
/>
|
|
<Button
|
|
label={
|
|
'showsHorizontalScrollIndicator: ' +
|
|
showsHorizontalScrollIndic.toString()
|
|
}
|
|
onPress={() =>
|
|
setShowsHorizontalScrollIndic(!showsHorizontalScrollIndic)
|
|
}
|
|
/>
|
|
<Button
|
|
label={
|
|
'showsVerticalScrollIndicator: ' + showsVerticalScrollIndic.toString()
|
|
}
|
|
onPress={() => setShowsVerticalScrollIndic(!showsVerticalScrollIndic)}
|
|
/>
|
|
</View>
|
|
);
|
|
};
|
|
|
|
const RemoveClippedSubviews = () => {
|
|
const [removeClippedSubviews, setRemoveClippedSubviews] = useState(false);
|
|
return (
|
|
<View>
|
|
<ScrollView
|
|
style={[styles.scrollView, {height: 200}]}
|
|
removeClippedSubviews={removeClippedSubviews}
|
|
nestedScrollEnabled>
|
|
{ITEMS.map(createItemRow)}
|
|
</ScrollView>
|
|
<Button
|
|
label={'removeClippedSubviews: ' + removeClippedSubviews.toString()}
|
|
onPress={() => setRemoveClippedSubviews(!removeClippedSubviews)}
|
|
/>
|
|
</View>
|
|
);
|
|
};
|
|
|
|
const RefreshControlExample = () => {
|
|
const [refreshing, setRefreshing] = useState(false);
|
|
const onRefresh = useCallback(() => {
|
|
setRefreshing(true);
|
|
// $FlowFixMe[unused-promise]
|
|
wait(2000).then(() => setRefreshing(false));
|
|
}, []);
|
|
|
|
const wait = (timeout: number) => {
|
|
return new Promise(resolve => {
|
|
setTimeout(resolve, timeout);
|
|
});
|
|
};
|
|
|
|
return (
|
|
<View>
|
|
<ScrollView
|
|
style={[styles.scrollView, {height: 200}]}
|
|
refreshControl={
|
|
<RefreshControl refreshing={refreshing} onRefresh={onRefresh} />
|
|
}
|
|
nestedScrollEnabled>
|
|
{ITEMS.map(createItemRow)}
|
|
</ScrollView>
|
|
</View>
|
|
);
|
|
};
|
|
|
|
const OnScrollOptions = () => {
|
|
const [onScrollDrag, setOnScrollDrag] = useState('none');
|
|
const [overScrollMode, setOverScrollMode] = useState('auto');
|
|
const overScrollModeOptions = ['auto', 'always', 'never'];
|
|
return (
|
|
<View>
|
|
<RNTesterText>onScroll: {onScrollDrag}</RNTesterText>
|
|
{/* $FlowFixMe[incompatible-use] Natural Inference rollout. See
|
|
* https://fburl.com/workplace/6291gfvu */}
|
|
<ScrollView
|
|
style={[styles.scrollView, {height: 200}]}
|
|
onScrollBeginDrag={() => setOnScrollDrag('onScrollBeginDrag')}
|
|
onScrollEndDrag={() => setOnScrollDrag('onScrollEndDrag')}
|
|
onScrollToTop={() => setOnScrollDrag('onScrollToTop')}
|
|
overScrollMode={overScrollMode}
|
|
nestedScrollEnabled>
|
|
{ITEMS.map(createItemRow)}
|
|
</ScrollView>
|
|
{Platform.OS === 'android' ? (
|
|
<>
|
|
<RNTesterText style={styles.rowTitle}>Over Scroll Mode</RNTesterText>
|
|
<View style={styles.row}>
|
|
{overScrollModeOptions.map(value => (
|
|
<Button
|
|
active={value === overScrollMode}
|
|
label={value}
|
|
key={value}
|
|
onPress={() => setOverScrollMode(value)}
|
|
/>
|
|
))}
|
|
</View>
|
|
</>
|
|
) : null}
|
|
</View>
|
|
);
|
|
};
|
|
|
|
const OnMomentumScroll = () => {
|
|
const [scroll, setScroll] = useState('none');
|
|
return (
|
|
<View>
|
|
<RNTesterText>Scroll State: {scroll}</RNTesterText>
|
|
<ScrollView
|
|
style={[styles.scrollView, {height: 200}]}
|
|
onMomentumScrollBegin={() => setScroll('onMomentumScrollBegin')}
|
|
onMomentumScrollEnd={() => setScroll('onMomentumScrollEnd')}
|
|
nestedScrollEnabled>
|
|
{ITEMS.map(createItemRow)}
|
|
</ScrollView>
|
|
</View>
|
|
);
|
|
};
|
|
|
|
const OnContentSizeChange = () => {
|
|
const [items, setItems] = useState(ITEMS);
|
|
const [contentSizeChanged, setContentSizeChanged] = useState('original');
|
|
return (
|
|
<View>
|
|
<RNTesterText>Content Size Changed: {contentSizeChanged}</RNTesterText>
|
|
<ScrollView
|
|
style={[styles.scrollView, {height: 200}]}
|
|
onContentSizeChange={() =>
|
|
contentSizeChanged === 'original'
|
|
? setContentSizeChanged('changed')
|
|
: setContentSizeChanged('original')
|
|
}
|
|
nestedScrollEnabled>
|
|
{items.map(createItemRow)}
|
|
</ScrollView>
|
|
<Button
|
|
label="Change Content Size"
|
|
onPress={() =>
|
|
items === ITEMS
|
|
? setItems(['1', '2', '3', '4', '5'])
|
|
: setItems(ITEMS)
|
|
}
|
|
/>
|
|
</View>
|
|
);
|
|
};
|
|
|
|
const MaxMinZoomScale = () => {
|
|
const [maxZoomScale, setMaxZoomScale] = useState('1.0');
|
|
const [minZoomScale, setMinZoomScale] = useState('1.0');
|
|
const [zoomScale, setZoomScale] = useState('1.0');
|
|
return (
|
|
<View>
|
|
<ScrollView
|
|
style={[styles.scrollView, {height: 200}]}
|
|
pinchGestureEnabled
|
|
maximumZoomScale={maxZoomScale !== '' ? parseFloat(maxZoomScale) : 0.0}
|
|
minimumZoomScale={minZoomScale !== '' ? parseFloat(minZoomScale) : 0.0}
|
|
nestedScrollEnabled>
|
|
{ITEMS.map(createItemRow)}
|
|
</ScrollView>
|
|
<RNTesterText style={styles.rowTitle}>
|
|
Set Maximum Zoom Scale
|
|
</RNTesterText>
|
|
<TextInput
|
|
style={styles.textInput}
|
|
value={maxZoomScale}
|
|
onChangeText={val => setMaxZoomScale(val)}
|
|
keyboardType="decimal-pad"
|
|
/>
|
|
<RNTesterText style={styles.rowTitle}>
|
|
Set Minimum Zoom Scale
|
|
</RNTesterText>
|
|
<TextInput
|
|
style={styles.textInput}
|
|
value={minZoomScale.toString()}
|
|
onChangeText={val => setMinZoomScale(val)}
|
|
keyboardType="decimal-pad"
|
|
/>
|
|
{Platform.OS === 'ios' ? (
|
|
<>
|
|
<RNTesterText style={styles.rowTitle}>Set Zoom Scale</RNTesterText>
|
|
<TextInput
|
|
style={styles.textInput}
|
|
value={zoomScale.toString()}
|
|
onChangeText={val => setZoomScale(val)}
|
|
keyboardType="decimal-pad"
|
|
/>
|
|
</>
|
|
) : null}
|
|
</View>
|
|
);
|
|
};
|
|
|
|
const KeyboardExample = () => {
|
|
const [keyboardDismissMode, setKeyboardDismissMode] = useState('none');
|
|
const [keyboardShouldPersistTaps, setKeyboardShouldPersistTaps] =
|
|
useState('never');
|
|
const [textInputValue, setTextInputValue] = useState('Tap to open Keyboard');
|
|
const dismissOptions =
|
|
Platform.OS === 'ios'
|
|
? ['none', 'on-drag', 'interactive']
|
|
: ['none', 'on-drag'];
|
|
const persistOptions = ['never', 'always', 'handled'];
|
|
return (
|
|
<View>
|
|
<TextInput
|
|
style={styles.textInput}
|
|
value={textInputValue}
|
|
onChangeText={val => setTextInputValue(val)}
|
|
/>
|
|
{/* $FlowFixMe[incompatible-use] Natural Inference rollout. See
|
|
* https://fburl.com/workplace/6291gfvu */}
|
|
<ScrollView
|
|
style={[styles.scrollView, {height: 200}]}
|
|
keyboardDismissMode={keyboardDismissMode}
|
|
keyboardShouldPersistTaps={keyboardShouldPersistTaps}
|
|
nestedScrollEnabled>
|
|
<Button
|
|
onPress={() => console.log('button pressed!')}
|
|
label={'Button'}
|
|
/>
|
|
{ITEMS.map(createItemRow)}
|
|
</ScrollView>
|
|
<RNTesterText style={styles.rowTitle}>Keyboard Dismiss Mode</RNTesterText>
|
|
<View style={styles.row}>
|
|
{dismissOptions.map(value => (
|
|
<Button
|
|
active={value === keyboardDismissMode}
|
|
label={value}
|
|
key={value}
|
|
onPress={() => setKeyboardDismissMode(value)}
|
|
/>
|
|
))}
|
|
</View>
|
|
<RNTesterText style={styles.rowTitle}>
|
|
Keyboard Should Persist taps
|
|
</RNTesterText>
|
|
<View style={styles.row}>
|
|
{persistOptions.map(value => (
|
|
<Button
|
|
active={value === keyboardShouldPersistTaps}
|
|
label={value}
|
|
key={value}
|
|
onPress={() => setKeyboardShouldPersistTaps(value)}
|
|
/>
|
|
))}
|
|
</View>
|
|
</View>
|
|
);
|
|
};
|
|
|
|
const InvertStickyHeaders = () => {
|
|
const [invertStickyHeaders, setInvertStickyHeaders] = useState(false);
|
|
const _scrollView = useRef<?React.ElementRef<typeof ScrollView>>(null);
|
|
return (
|
|
<View>
|
|
{/* $FlowFixMe[incompatible-use] */}
|
|
<ScrollView
|
|
ref={_scrollView}
|
|
style={[styles.scrollView, {height: 200}]}
|
|
stickyHeaderIndices={[0]}
|
|
invertStickyHeaders={invertStickyHeaders}
|
|
nestedScrollEnabled
|
|
testID="scroll_sticky_header">
|
|
<Text>STICKY HEADER</Text>
|
|
{ITEMS.map(createItemRow)}
|
|
</ScrollView>
|
|
<View>
|
|
<Button
|
|
onPress={() => setInvertStickyHeaders(!invertStickyHeaders)}
|
|
label={'invertStickyHeaders: ' + invertStickyHeaders.toString()}
|
|
/>
|
|
<Button
|
|
label="Scroll to top"
|
|
onPress={() => {
|
|
nullthrows<$FlowFixMe>(_scrollView.current).scrollTo({y: 0});
|
|
}}
|
|
testID="scroll_to_top_button"
|
|
/>
|
|
<Button
|
|
label="Scroll to bottom"
|
|
onPress={() => {
|
|
nullthrows<$FlowFixMe>(_scrollView.current).scrollToEnd({
|
|
animated: true,
|
|
});
|
|
}}
|
|
testID="scroll_to_bottom_button"
|
|
/>
|
|
</View>
|
|
</View>
|
|
);
|
|
};
|
|
|
|
const MultipleStickyHeaders = () => {
|
|
const _scrollView = useRef<?React.ElementRef<typeof ScrollView>>(null);
|
|
const stickyHeaderStyle = {backgroundColor: 'yellow'};
|
|
return (
|
|
<View>
|
|
{/* $FlowFixMe[incompatible-use] */}
|
|
<ScrollView
|
|
ref={_scrollView}
|
|
style={[styles.scrollView, {height: 200}]}
|
|
stickyHeaderIndices={[0, 13, 26]}
|
|
nestedScrollEnabled
|
|
testID="scroll_multiple_sticky_headers">
|
|
<Item msg={'Sticky Header 1'} style={stickyHeaderStyle} />
|
|
{ITEMS.map(createItemRow)}
|
|
<Item msg={'Sticky Header 2'} style={stickyHeaderStyle} />
|
|
{ITEMS.map(createItemRow)}
|
|
<Item msg={'Sticky Header 3'} style={stickyHeaderStyle} />
|
|
{ITEMS.map(createItemRow)}
|
|
</ScrollView>
|
|
<View>
|
|
<Button
|
|
label="Scroll to top"
|
|
onPress={() => {
|
|
nullthrows<$FlowFixMe>(_scrollView.current).scrollTo({y: 0});
|
|
}}
|
|
testID="scroll_to_top_button"
|
|
/>
|
|
<Button
|
|
label="Scroll to bottom"
|
|
onPress={() => {
|
|
nullthrows<$FlowFixMe>(_scrollView.current).scrollToEnd({
|
|
animated: true,
|
|
});
|
|
}}
|
|
testID="scroll_to_bottom_button"
|
|
/>
|
|
</View>
|
|
</View>
|
|
);
|
|
};
|
|
|
|
const IndicatorStyle = () => {
|
|
const [indicatorStyle, setIndicatorStyle] = useState('default');
|
|
return (
|
|
<View>
|
|
{/* $FlowFixMe[incompatible-use] Natural Inference rollout. See
|
|
* https://fburl.com/workplace/6291gfvu */}
|
|
<ScrollView
|
|
style={[styles.scrollView, {height: 200}]}
|
|
indicatorStyle={indicatorStyle}
|
|
nestedScrollEnabled>
|
|
{ITEMS.map(createItemRow)}
|
|
</ScrollView>
|
|
<View>
|
|
<Button
|
|
onPress={() =>
|
|
indicatorStyle === 'default'
|
|
? setIndicatorStyle('white')
|
|
: setIndicatorStyle('default')
|
|
}
|
|
label={'Indicator Style: ' + indicatorStyle}
|
|
/>
|
|
</View>
|
|
</View>
|
|
);
|
|
};
|
|
|
|
const DisableEnable = () => {
|
|
const [directionalLockEnabled, setDirectionalLockEnabled] = useState(false);
|
|
const [disableIntervalMomentum, setDisableIntervalMomentum] = useState(false);
|
|
const [disableScrollViewPanResponder, setDisableScrollViewPanResponder] =
|
|
useState(false);
|
|
return (
|
|
<View>
|
|
<ScrollView
|
|
style={[styles.scrollView, {height: 200}]}
|
|
contentInset={{top: 10, bottom: 10, left: 10, right: 10}}
|
|
snapToInterval={0}
|
|
directionalLockEnabled={directionalLockEnabled}
|
|
disableIntervalMomentum={disableIntervalMomentum}
|
|
disableScrollViewPanResponder={disableScrollViewPanResponder}
|
|
nestedScrollEnabled>
|
|
{ITEMS.map(createItemRow)}
|
|
</ScrollView>
|
|
<View>
|
|
{Platform.OS === 'ios' ? (
|
|
<Button
|
|
onPress={() => setDirectionalLockEnabled(!directionalLockEnabled)}
|
|
label={
|
|
'directionalLockEnabled: ' + directionalLockEnabled.toString()
|
|
}
|
|
/>
|
|
) : null}
|
|
<Button
|
|
onPress={() => setDisableIntervalMomentum(!disableIntervalMomentum)}
|
|
label={
|
|
'setDisableIntervalMomentum: ' + disableIntervalMomentum.toString()
|
|
}
|
|
/>
|
|
<Button
|
|
onPress={() =>
|
|
setDisableScrollViewPanResponder(!disableScrollViewPanResponder)
|
|
}
|
|
label={
|
|
'setDisableScrollViewPanResponder: ' +
|
|
disableScrollViewPanResponder.toString()
|
|
}
|
|
/>
|
|
</View>
|
|
</View>
|
|
);
|
|
};
|
|
|
|
const DecelerationRateExample = () => {
|
|
const [decelRate, setDecelRate] = useState('normal');
|
|
return (
|
|
<View>
|
|
{/* $FlowFixMe[incompatible-use] Natural Inference rollout. See
|
|
* https://fburl.com/workplace/6291gfvu */}
|
|
<ScrollView
|
|
style={[styles.scrollView, {height: 200}]}
|
|
decelerationRate={decelRate}
|
|
nestedScrollEnabled>
|
|
{ITEMS.map(createItemRow)}
|
|
</ScrollView>
|
|
<View>
|
|
<Button
|
|
onPress={() =>
|
|
decelRate === 'normal'
|
|
? setDecelRate('fast')
|
|
: setDecelRate('normal')
|
|
}
|
|
label={'Deceleration Rate: ' + decelRate}
|
|
/>
|
|
</View>
|
|
</View>
|
|
);
|
|
};
|
|
|
|
const ContentExample = () => {
|
|
const [canCancelContentTouches, setCanCancelContentTouches] = useState(false);
|
|
const [contentInset, setContentInset] = useState<null | {
|
|
bottom: number,
|
|
left: number,
|
|
right: number,
|
|
top: number,
|
|
}>(null);
|
|
const [contentContainerStyle, setContentContainerStyle] = useState<null | {
|
|
backgroundColor: string,
|
|
}>(null);
|
|
const [contentInsetAdjustmentBehavior, setContentInsetAdjustmentBehavior] =
|
|
useState('never');
|
|
return (
|
|
<View>
|
|
{/* $FlowFixMe[incompatible-use] Natural Inference rollout. See
|
|
* https://fburl.com/workplace/6291gfvu */}
|
|
<ScrollView
|
|
style={[styles.scrollView, {height: 200}]}
|
|
canCancelContentTouches={canCancelContentTouches}
|
|
contentOffset={{x: 100, y: 0}}
|
|
contentContainerStyle={contentContainerStyle}
|
|
contentInset={contentInset}
|
|
contentInsetAdjustmentBehavior={contentInsetAdjustmentBehavior}
|
|
nestedScrollEnabled>
|
|
{ITEMS.map(createItemRow)}
|
|
</ScrollView>
|
|
<View>
|
|
{Platform.OS === 'ios' ? (
|
|
<>
|
|
<Button
|
|
onPress={() =>
|
|
setCanCancelContentTouches(!canCancelContentTouches)
|
|
}
|
|
label={
|
|
'canCancelContentTouches: ' + canCancelContentTouches.toString()
|
|
}
|
|
/>
|
|
<Button
|
|
onPress={() =>
|
|
contentInsetAdjustmentBehavior === 'never'
|
|
? setContentInsetAdjustmentBehavior('always')
|
|
: setContentInsetAdjustmentBehavior('never')
|
|
}
|
|
label={
|
|
contentInsetAdjustmentBehavior === 'never'
|
|
? "setContentInsetAdjustmentBehavior to 'always'"
|
|
: 'reset content inset adjustment behavior'
|
|
}
|
|
/>
|
|
</>
|
|
) : null}
|
|
<Button
|
|
onPress={() =>
|
|
contentContainerStyle === null
|
|
? setContentContainerStyle(styles.containerStyle)
|
|
: setContentContainerStyle(null)
|
|
}
|
|
label={
|
|
contentContainerStyle === null
|
|
? 'setContentContainerStyle'
|
|
: 'reset content container style'
|
|
}
|
|
/>
|
|
<Button
|
|
onPress={() =>
|
|
contentInset === null
|
|
? setContentInset({top: 10, bottom: 10, left: 10, right: 10})
|
|
: setContentInset(null)
|
|
}
|
|
label={
|
|
contentInset === null ? 'setContentInset' : 'reset content inset'
|
|
}
|
|
/>
|
|
</View>
|
|
</View>
|
|
);
|
|
};
|
|
|
|
const BouncesExample = () => {
|
|
const [bounces, setBounces] = useState(false);
|
|
const [bouncesZoom, setBouncesZoom] = useState(false);
|
|
return (
|
|
<View>
|
|
<ScrollView
|
|
style={[styles.scrollView, {height: 200}]}
|
|
bounces={bounces}
|
|
bouncesZoom={bouncesZoom}
|
|
nestedScrollEnabled>
|
|
{ITEMS.map(createItemRow)}
|
|
</ScrollView>
|
|
<View>
|
|
<Button
|
|
onPress={() => setBounces(!bounces)}
|
|
label={'Bounces: ' + bounces.toString()}
|
|
/>
|
|
<Button
|
|
onPress={() => setBouncesZoom(!bouncesZoom)}
|
|
label={'Bounces Zoom: ' + bouncesZoom.toString()}
|
|
/>
|
|
</View>
|
|
</View>
|
|
);
|
|
};
|
|
|
|
const BouncesExampleHorizontal = () => {
|
|
const [bounce, setBounce] = useState(false);
|
|
return (
|
|
<View>
|
|
<ScrollView
|
|
style={[styles.scrollView, {height: 200}]}
|
|
horizontal={true}
|
|
alwaysBounceHorizontal={bounce}
|
|
contentOffset={{x: 100, y: 0}}>
|
|
{ITEMS.map(createItemRow)}
|
|
</ScrollView>
|
|
<View>
|
|
<Button
|
|
onPress={() => setBounce(!bounce)}
|
|
label={'Always Bounce Horizontal: ' + bounce.toString()}
|
|
/>
|
|
</View>
|
|
</View>
|
|
);
|
|
};
|
|
|
|
const BouncesExampleVertical = () => {
|
|
const [bounce, setBounce] = useState(false);
|
|
return (
|
|
<View>
|
|
<ScrollView
|
|
style={[styles.scrollView, {height: 200}]}
|
|
alwaysBounceVertical={bounce}
|
|
contentOffset={{x: 100, y: 0}}
|
|
nestedScrollEnabled>
|
|
{ITEMS.map(createItemRow)}
|
|
</ScrollView>
|
|
<View>
|
|
<Button
|
|
onPress={() => setBounce(!bounce)}
|
|
label={'Always Bounce Vertical: ' + bounce.toString()}
|
|
/>
|
|
</View>
|
|
</View>
|
|
);
|
|
};
|
|
|
|
function ClippingExampleVertical() {
|
|
return (
|
|
<ScrollView
|
|
testID="clipping_example_vertical"
|
|
style={[
|
|
styles.scrollView,
|
|
{height: 200, borderRadius: 100, borderColor: 'red', borderWidth: 5},
|
|
]}
|
|
nestedScrollEnabled={true}>
|
|
{ITEMS.map(createItemRow)}
|
|
</ScrollView>
|
|
);
|
|
}
|
|
|
|
function ClippingExampleHorizontal() {
|
|
return (
|
|
<ScrollView
|
|
testID="clipping_example_horizontal"
|
|
horizontal={true}
|
|
style={[
|
|
styles.scrollView,
|
|
{height: 200, borderRadius: 100, borderColor: 'red', borderWidth: 5},
|
|
]}
|
|
nestedScrollEnabled={true}>
|
|
{ITEMS.map(createItemRow)}
|
|
</ScrollView>
|
|
);
|
|
}
|
|
|
|
function TouchableItem({index}: {index: number}) {
|
|
const [pressed, setPressed] = useState(false);
|
|
|
|
return (
|
|
<View
|
|
onTouchStart={() => setPressed(p => !p)}
|
|
testID={`touchable_item_${index}`}
|
|
style={{
|
|
position: 'relative',
|
|
flexDirection: 'row',
|
|
justifyContent: 'center',
|
|
alignItems: 'center',
|
|
flexGrow: 1,
|
|
flexShrink: 1,
|
|
flexBasis: '25%',
|
|
margin: 5,
|
|
backgroundColor: pressed ? 'gray' : 'lightgray',
|
|
}}>
|
|
<Text>Item {index}</Text>
|
|
</View>
|
|
);
|
|
}
|
|
|
|
function ChildrenWithTouchEventsOverflowingContainerHorizontal() {
|
|
return (
|
|
<ScrollView
|
|
testID="touchable_overflowing_container_horizontal"
|
|
horizontal={true}
|
|
style={[styles.scrollView, {height: 200, width: '100%'}]}
|
|
contentContainerStyle={{
|
|
backgroundColor: 'red',
|
|
}}
|
|
nestedScrollEnabled={true}>
|
|
<View
|
|
style={{
|
|
display: 'flex',
|
|
flexDirection: 'row',
|
|
justifyContent: 'flex-start',
|
|
alignItems: 'stretch',
|
|
minHeight: 45,
|
|
minWidth: '100%',
|
|
}}>
|
|
<TouchableItem index={1} />
|
|
<TouchableItem index={2} />
|
|
<TouchableItem index={3} />
|
|
</View>
|
|
</ScrollView>
|
|
);
|
|
}
|
|
|
|
class Item extends React.PureComponent<{
|
|
msg?: string,
|
|
style?: ViewStyleProp,
|
|
}> {
|
|
render(): $FlowFixMe {
|
|
return (
|
|
<View style={[styles.item, this.props.style]}>
|
|
<Text>{this.props.msg}</Text>
|
|
</View>
|
|
);
|
|
}
|
|
}
|
|
|
|
const ITEMS = [...Array(12)].map((_, i) => `Item ${i}`);
|
|
|
|
const createItemRow = (msg: string, index: number) => (
|
|
<Item key={index} msg={msg} />
|
|
);
|
|
|
|
const Button = (props: {
|
|
active?: boolean,
|
|
label: string,
|
|
onPress: () => void,
|
|
testID?: string,
|
|
}) => (
|
|
<TouchableOpacity
|
|
style={StyleSheet.compose(
|
|
styles.button,
|
|
props.active === true ? styles.activeButton : null,
|
|
)}
|
|
onPress={props.onPress}
|
|
testID={props.testID}>
|
|
<Text>{props.label}</Text>
|
|
</TouchableOpacity>
|
|
);
|
|
|
|
const styles = StyleSheet.create({
|
|
scrollView: {
|
|
backgroundColor: '#eeeeee',
|
|
height: 300,
|
|
},
|
|
horizontalScrollView: {
|
|
height: 106,
|
|
},
|
|
text: {
|
|
fontSize: 16,
|
|
fontWeight: 'bold',
|
|
margin: 5,
|
|
},
|
|
activeButton: {
|
|
backgroundColor: 'rgba(100,215,255,.3)',
|
|
},
|
|
button: {
|
|
margin: 5,
|
|
padding: 5,
|
|
alignItems: 'center',
|
|
backgroundColor: '#cccccc',
|
|
borderRadius: 3,
|
|
},
|
|
row: {
|
|
flexDirection: 'row',
|
|
justifyContent: 'space-around',
|
|
},
|
|
item: {
|
|
margin: 5,
|
|
padding: 5,
|
|
backgroundColor: '#cccccc',
|
|
borderRadius: 3,
|
|
minWidth: 96,
|
|
},
|
|
containerStyle: {
|
|
backgroundColor: '#aae3b6',
|
|
},
|
|
rowTitle: {
|
|
flex: 1,
|
|
fontWeight: 'bold',
|
|
alignSelf: 'center',
|
|
},
|
|
textInput: {
|
|
height: 40,
|
|
borderColor: 'gray',
|
|
borderWidth: 1,
|
|
},
|
|
});
|