mirror of
https://github.com/status-im/react-native.git
synced 2025-01-16 12:34:17 +00:00
475f1b5796
Summary:Using TouchableNativeFeedback has been a problem for me because the ripples it makes don't follow the child view's border radii so the ripples stick out of the child view's rounded corners. This PR should fix this problem with a minor caveat: this only works for TouchableNativeFeedback.Ripple and not TouchableNativeFeedback.SelectableBackground. I searched how I could apply corner radius to selectableItemBackground and it doesn't seem to be possible (the prevalent advice is to create the ripple manually which is equivalent to using TNF.Ripple in our case), though I could be wrong. I added [an example to UIExplorer (TouchableExample)](http://i.imgur.com/CHY9xjW.png). This is my first PR to this repo so let me know if something's wrong. Cheers! Closes https://github.com/facebook/react-native/pull/6515 Differential Revision: D3126513 Pulled By: AaaChiuuu fb-gh-sync-id: 4a00e7177ee4ffd8dffeca143f4f43f08c99b5a1 fbshipit-source-id: 4a00e7177ee4ffd8dffeca143f4f43f08c99b5a1
445 lines
12 KiB
JavaScript
445 lines
12 KiB
JavaScript
/**
|
|
* The examples provided by Facebook are for non-commercial testing and
|
|
* evaluation purposes only.
|
|
*
|
|
* Facebook reserves all rights not expressly granted.
|
|
*
|
|
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS
|
|
* OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
|
* FITNESS FOR A PARTICULAR PURPOSE AND NON INFRINGEMENT. IN NO EVENT SHALL
|
|
* FACEBOOK BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN
|
|
* AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN
|
|
* CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
|
|
*
|
|
* @flow
|
|
*/
|
|
'use strict';
|
|
|
|
var React = require('react');
|
|
var ReactNative = require('react-native');
|
|
var {
|
|
PixelRatio,
|
|
Image,
|
|
StyleSheet,
|
|
Text,
|
|
TouchableHighlight,
|
|
TouchableOpacity,
|
|
UIManager,
|
|
Platform,
|
|
TouchableNativeFeedback,
|
|
View,
|
|
} = ReactNative;
|
|
|
|
exports.displayName = (undefined: ?string);
|
|
exports.description = 'Touchable and onPress examples.';
|
|
exports.title = '<Touchable*> and onPress';
|
|
exports.examples = [
|
|
{
|
|
title: '<TouchableHighlight>',
|
|
description: 'TouchableHighlight works by adding an extra view with a ' +
|
|
'black background under the single child view. This works best when the ' +
|
|
'child view is fully opaque, although it can be made to work as a simple ' +
|
|
'background color change as well with the activeOpacity and ' +
|
|
'underlayColor props.',
|
|
render: function() {
|
|
return (
|
|
<View>
|
|
<View style={styles.row}>
|
|
<TouchableHighlight
|
|
style={styles.wrapper}
|
|
onPress={() => console.log('stock THW image - highlight')}>
|
|
<Image
|
|
source={heartImage}
|
|
style={styles.image}
|
|
/>
|
|
</TouchableHighlight>
|
|
<TouchableHighlight
|
|
style={styles.wrapper}
|
|
activeOpacity={1}
|
|
animationVelocity={0}
|
|
underlayColor="rgb(210, 230, 255)"
|
|
onPress={() => console.log('custom THW text - highlight')}>
|
|
<View style={styles.wrapperCustom}>
|
|
<Text style={styles.text}>
|
|
Tap Here For Custom Highlight!
|
|
</Text>
|
|
</View>
|
|
</TouchableHighlight>
|
|
</View>
|
|
</View>
|
|
);
|
|
},
|
|
}, {
|
|
title: '<Text onPress={fn}> with highlight',
|
|
render: function(): ReactElement {
|
|
return <TextOnPressBox />;
|
|
},
|
|
}, {
|
|
title: 'Touchable feedback events',
|
|
description: '<Touchable*> components accept onPress, onPressIn, ' +
|
|
'onPressOut, and onLongPress as props.',
|
|
render: function(): ReactElement {
|
|
return <TouchableFeedbackEvents />;
|
|
},
|
|
}, {
|
|
title: 'Touchable delay for events',
|
|
description: '<Touchable*> components also accept delayPressIn, ' +
|
|
'delayPressOut, and delayLongPress as props. These props impact the ' +
|
|
'timing of feedback events.',
|
|
render: function(): ReactElement {
|
|
return <TouchableDelayEvents />;
|
|
},
|
|
}, {
|
|
title: '3D Touch / Force Touch',
|
|
description: 'iPhone 6s and 6s plus support 3D touch, which adds a force property to touches',
|
|
render: function(): ReactElement {
|
|
return <ForceTouchExample />;
|
|
},
|
|
platform: 'ios',
|
|
}, {
|
|
title: 'Touchable Hit Slop',
|
|
description: '<Touchable*> components accept hitSlop prop which extends the touch area ' +
|
|
'without changing the view bounds.',
|
|
render: function(): ReactElement {
|
|
return <TouchableHitSlop />;
|
|
},
|
|
}, {
|
|
title: 'Disabled Touchable*',
|
|
description: '<Touchable*> components accept disabled prop which prevents ' +
|
|
'any interaction with component',
|
|
render: function(): ReactElement {
|
|
return <TouchableDisabled />;
|
|
},
|
|
}];
|
|
|
|
var TextOnPressBox = React.createClass({
|
|
getInitialState: function() {
|
|
return {
|
|
timesPressed: 0,
|
|
};
|
|
},
|
|
textOnPress: function() {
|
|
this.setState({
|
|
timesPressed: this.state.timesPressed + 1,
|
|
});
|
|
},
|
|
render: function() {
|
|
var textLog = '';
|
|
if (this.state.timesPressed > 1) {
|
|
textLog = this.state.timesPressed + 'x text onPress';
|
|
} else if (this.state.timesPressed > 0) {
|
|
textLog = 'text onPress';
|
|
}
|
|
|
|
return (
|
|
<View>
|
|
<Text
|
|
style={styles.textBlock}
|
|
onPress={this.textOnPress}>
|
|
Text has built-in onPress handling
|
|
</Text>
|
|
<View style={styles.logBox}>
|
|
<Text>
|
|
{textLog}
|
|
</Text>
|
|
</View>
|
|
</View>
|
|
);
|
|
}
|
|
});
|
|
|
|
var TouchableFeedbackEvents = React.createClass({
|
|
getInitialState: function() {
|
|
return {
|
|
eventLog: [],
|
|
};
|
|
},
|
|
render: function() {
|
|
return (
|
|
<View testID="touchable_feedback_events">
|
|
<View style={[styles.row, {justifyContent: 'center'}]}>
|
|
<TouchableOpacity
|
|
style={styles.wrapper}
|
|
testID="touchable_feedback_events_button"
|
|
accessibilityLabel="touchable feedback events"
|
|
accessibilityTraits="button"
|
|
accessibilityComponentType="button"
|
|
onPress={() => this._appendEvent('press')}
|
|
onPressIn={() => this._appendEvent('pressIn')}
|
|
onPressOut={() => this._appendEvent('pressOut')}
|
|
onLongPress={() => this._appendEvent('longPress')}>
|
|
<Text style={styles.button}>
|
|
Press Me
|
|
</Text>
|
|
</TouchableOpacity>
|
|
</View>
|
|
<View testID="touchable_feedback_events_console" style={styles.eventLogBox}>
|
|
{this.state.eventLog.map((e, ii) => <Text key={ii}>{e}</Text>)}
|
|
</View>
|
|
</View>
|
|
);
|
|
},
|
|
_appendEvent: function(eventName) {
|
|
var limit = 6;
|
|
var eventLog = this.state.eventLog.slice(0, limit - 1);
|
|
eventLog.unshift(eventName);
|
|
this.setState({eventLog});
|
|
},
|
|
});
|
|
|
|
var TouchableDelayEvents = React.createClass({
|
|
getInitialState: function() {
|
|
return {
|
|
eventLog: [],
|
|
};
|
|
},
|
|
render: function() {
|
|
return (
|
|
<View testID="touchable_delay_events">
|
|
<View style={[styles.row, {justifyContent: 'center'}]}>
|
|
<TouchableOpacity
|
|
style={styles.wrapper}
|
|
testID="touchable_delay_events_button"
|
|
onPress={() => this._appendEvent('press')}
|
|
delayPressIn={400}
|
|
onPressIn={() => this._appendEvent('pressIn - 400ms delay')}
|
|
delayPressOut={1000}
|
|
onPressOut={() => this._appendEvent('pressOut - 1000ms delay')}
|
|
delayLongPress={800}
|
|
onLongPress={() => this._appendEvent('longPress - 800ms delay')}>
|
|
<Text style={styles.button}>
|
|
Press Me
|
|
</Text>
|
|
</TouchableOpacity>
|
|
</View>
|
|
<View style={styles.eventLogBox} testID="touchable_delay_events_console">
|
|
{this.state.eventLog.map((e, ii) => <Text key={ii}>{e}</Text>)}
|
|
</View>
|
|
</View>
|
|
);
|
|
},
|
|
_appendEvent: function(eventName) {
|
|
var limit = 6;
|
|
var eventLog = this.state.eventLog.slice(0, limit - 1);
|
|
eventLog.unshift(eventName);
|
|
this.setState({eventLog});
|
|
},
|
|
});
|
|
|
|
var ForceTouchExample = React.createClass({
|
|
getInitialState: function() {
|
|
return {
|
|
force: 0,
|
|
};
|
|
},
|
|
_renderConsoleText: function() {
|
|
return View.forceTouchAvailable ?
|
|
'Force: ' + this.state.force.toFixed(3) :
|
|
'3D Touch is not available on this device';
|
|
},
|
|
render: function() {
|
|
return (
|
|
<View testID="touchable_3dtouch_event">
|
|
<View style={styles.forceTouchBox} testID="touchable_3dtouch_output">
|
|
<Text>{this._renderConsoleText()}</Text>
|
|
</View>
|
|
<View style={[styles.row, {justifyContent: 'center'}]}>
|
|
<View
|
|
style={styles.wrapper}
|
|
testID="touchable_3dtouch_button"
|
|
onStartShouldSetResponder={() => true}
|
|
onResponderMove={(event) => this.setState({force: event.nativeEvent.force})}
|
|
onResponderRelease={(event) => this.setState({force: 0})}>
|
|
<Text style={styles.button}>
|
|
Press Me
|
|
</Text>
|
|
</View>
|
|
</View>
|
|
</View>
|
|
);
|
|
},
|
|
});
|
|
|
|
var TouchableHitSlop = React.createClass({
|
|
getInitialState: function() {
|
|
return {
|
|
timesPressed: 0,
|
|
};
|
|
},
|
|
onPress: function() {
|
|
this.setState({
|
|
timesPressed: this.state.timesPressed + 1,
|
|
});
|
|
},
|
|
render: function() {
|
|
var log = '';
|
|
if (this.state.timesPressed > 1) {
|
|
log = this.state.timesPressed + 'x onPress';
|
|
} else if (this.state.timesPressed > 0) {
|
|
log = 'onPress';
|
|
}
|
|
|
|
return (
|
|
<View testID="touchable_hit_slop">
|
|
<View style={[styles.row, {justifyContent: 'center'}]}>
|
|
<TouchableOpacity
|
|
onPress={this.onPress}
|
|
style={styles.hitSlopWrapper}
|
|
hitSlop={{top: 30, bottom: 30, left: 60, right: 60}}
|
|
testID="touchable_hit_slop_button">
|
|
<Text style={styles.hitSlopButton}>
|
|
Press Outside This View
|
|
</Text>
|
|
</TouchableOpacity>
|
|
</View>
|
|
<View style={styles.logBox}>
|
|
<Text>
|
|
{log}
|
|
</Text>
|
|
</View>
|
|
</View>
|
|
);
|
|
}
|
|
});
|
|
|
|
var TouchableDisabled = React.createClass({
|
|
render: function() {
|
|
return (
|
|
<View>
|
|
<TouchableOpacity disabled={true} style={[styles.row, styles.block]}>
|
|
<Text style={styles.disabledButton}>Disabled TouchableOpacity</Text>
|
|
</TouchableOpacity>
|
|
|
|
<TouchableOpacity disabled={false} style={[styles.row, styles.block]}>
|
|
<Text style={styles.button}>Enabled TouchableOpacity</Text>
|
|
</TouchableOpacity>
|
|
|
|
<TouchableHighlight
|
|
activeOpacity={1}
|
|
disabled={true}
|
|
animationVelocity={0}
|
|
underlayColor="rgb(210, 230, 255)"
|
|
style={[styles.row, styles.block]}
|
|
onPress={() => console.log('custom THW text - highlight')}>
|
|
<Text style={styles.disabledButton}>
|
|
Disabled TouchableHighlight
|
|
</Text>
|
|
</TouchableHighlight>
|
|
|
|
<TouchableHighlight
|
|
activeOpacity={1}
|
|
animationVelocity={0}
|
|
underlayColor="rgb(210, 230, 255)"
|
|
style={[styles.row, styles.block]}
|
|
onPress={() => console.log('custom THW text - highlight')}>
|
|
<Text style={styles.button}>
|
|
Disabled TouchableHighlight
|
|
</Text>
|
|
</TouchableHighlight>
|
|
|
|
{Platform.OS === 'android' &&
|
|
<TouchableNativeFeedback
|
|
style={[styles.row, styles.block]}
|
|
onPress={() => console.log('custom TNF has been clicked')}
|
|
background={TouchableNativeFeedback.SelectableBackground()}>
|
|
<View>
|
|
<Text style={[styles.button, styles.nativeFeedbackButton]}>
|
|
Enabled TouchableNativeFeedback
|
|
</Text>
|
|
</View>
|
|
</TouchableNativeFeedback>
|
|
}
|
|
|
|
{Platform.OS === 'android' &&
|
|
<TouchableNativeFeedback
|
|
disabled={true}
|
|
style={[styles.row, styles.block]}
|
|
onPress={() => console.log('custom TNF has been clicked')}
|
|
background={TouchableNativeFeedback.SelectableBackground()}>
|
|
<View>
|
|
<Text style={[styles.disabledButton, styles.nativeFeedbackButton]}>
|
|
Disabled TouchableNativeFeedback
|
|
</Text>
|
|
</View>
|
|
</TouchableNativeFeedback>
|
|
}
|
|
</View>
|
|
);
|
|
}
|
|
});
|
|
|
|
var heartImage = {uri: 'https://pbs.twimg.com/media/BlXBfT3CQAA6cVZ.png:small'};
|
|
|
|
var styles = StyleSheet.create({
|
|
row: {
|
|
justifyContent: 'center',
|
|
flexDirection: 'row',
|
|
},
|
|
icon: {
|
|
width: 24,
|
|
height: 24,
|
|
},
|
|
image: {
|
|
width: 50,
|
|
height: 50,
|
|
},
|
|
text: {
|
|
fontSize: 16,
|
|
},
|
|
block: {
|
|
padding: 10,
|
|
},
|
|
button: {
|
|
color: '#007AFF',
|
|
},
|
|
disabledButton: {
|
|
color: '#007AFF',
|
|
opacity: 0.5,
|
|
},
|
|
nativeFeedbackButton: {
|
|
textAlign: 'center',
|
|
margin: 10,
|
|
},
|
|
hitSlopButton: {
|
|
color: 'white',
|
|
},
|
|
wrapper: {
|
|
borderRadius: 8,
|
|
},
|
|
wrapperCustom: {
|
|
borderRadius: 8,
|
|
padding: 6,
|
|
},
|
|
hitSlopWrapper: {
|
|
backgroundColor: 'red',
|
|
marginVertical: 30,
|
|
},
|
|
logBox: {
|
|
padding: 20,
|
|
margin: 10,
|
|
borderWidth: StyleSheet.hairlineWidth,
|
|
borderColor: '#f0f0f0',
|
|
backgroundColor: '#f9f9f9',
|
|
},
|
|
eventLogBox: {
|
|
padding: 10,
|
|
margin: 10,
|
|
height: 120,
|
|
borderWidth: StyleSheet.hairlineWidth,
|
|
borderColor: '#f0f0f0',
|
|
backgroundColor: '#f9f9f9',
|
|
},
|
|
forceTouchBox: {
|
|
padding: 10,
|
|
margin: 10,
|
|
borderWidth: StyleSheet.hairlineWidth,
|
|
borderColor: '#f0f0f0',
|
|
backgroundColor: '#f9f9f9',
|
|
alignItems: 'center',
|
|
},
|
|
textBlock: {
|
|
fontWeight: '500',
|
|
color: 'blue',
|
|
},
|
|
});
|