mirror of
https://github.com/zhigang1992/react-native-web.git
synced 2026-04-21 19:12:13 +08:00
563 lines
17 KiB
JavaScript
563 lines
17 KiB
JavaScript
/* eslint-disable react/jsx-no-bind, react/prefer-es6-class, react/prop-types */
|
|
|
|
/**
|
|
* Copyright (c) 2013-present, Facebook, Inc.
|
|
* All rights reserved.
|
|
*
|
|
* This source code is licensed under the BSD-style license found in the
|
|
* LICENSE file in the root directory of this source tree. An additional grant
|
|
* of patent rights can be found in the PATENTS file in the same directory.
|
|
*
|
|
* 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
|
|
*/
|
|
|
|
import React from 'react';
|
|
import UIExplorer from '../../UIExplorer';
|
|
import { storiesOf } from '@kadira/storybook';
|
|
import { ActivityIndicator, Image, StyleSheet, Text, View } from 'react-native';
|
|
|
|
const base64Icon =
|
|
'data:image/png;base64,iVBORw0KGgoAAAANSUhEUgAAAEsAAABLCAQAAACSR7JhAAADtUlEQVR4Ac3YA2Bj6QLH0XPT1Fzbtm29tW3btm3bfLZtv7e2ObZnms7d8Uw098tuetPzrxv8wiISrtVudrG2JXQZ4VOv+qUfmqCGGl1mqLhoA52oZlb0mrjsnhKpgeUNEs91Z0pd1kvihA3ULGVHiQO2narKSHKkEMulm9VgUyE60s1aWoMQUbpZOWE+kaqs4eLEjdIlZTcFZB0ndc1+lhB1lZrIuk5P2aib1NBpZaL+JaOGIt0ls47SKzLC7CqrlGF6RZ09HGoNy1lYl2aRSWL5GuzqWU1KafRdoRp0iOQEiDzgZPnG6DbldcomadViflnl/cL93tOoVbsOLVM2jylvdWjXolWX1hmfZbGR/wjypDjFLSZIRov09BgYmtUqPQPlQrPapecLgTIy0jMgPKtTeob2zWtrGH3xvjUkPCtNg/tm1rjwrMa+mdUkPd3hWbH0jArPGiU9ufCsNNWFZ40wpwn+62/66R2RUtoso1OB34tnLOcy7YB1fUdc9e0q3yru8PGM773vXsuZ5YIZX+5xmHwHGVvlrGPN6ZSiP1smOsMMde40wKv2VmwPPVXNut4sVpUreZiLBHi0qln/VQeI/LTMYXpsJtFiclUN+5HVZazim+Ky+7sAvxWnvjXrJFneVtLWLyPJu9K3cXLWeOlbMTlrIelbMDlrLenrjEQOtIF+fuI9xRp9ZBFp6+b6WT8RrxEpdK64BuvHgDk+vUy+b5hYk6zfyfs051gRoNO1usU12WWRWL73/MMEy9pMi9qIrR4ZpV16Rrvduxazmy1FSvuFXRkqTnE7m2kdb5U8xGjLw/spRr1uTov4uOgQE+0N/DvFrG/Jt7i/FzwxbA9kDanhf2w+t4V97G8lrT7wc08aA2QNUkuTfW/KimT01wdlfK4yEw030VfT0RtZbzjeMprNq8m8tnSTASrTLti64oBNdpmMQm0eEwvfPwRbUBywG5TzjPCsdwk3IeAXjQblLCoXnDVeoAz6SfJNk5TTzytCNZk/POtTSV40NwOFWzw86wNJRpubpXsn60NJFlHeqlYRbslqZm2jnEZ3qcSKgm0kTli3zZVS7y/iivZTweYXJ26Y+RTbV1zh3hYkgyFGSTKPfRVbRqWWVReaxYeSLarYv1Qqsmh1s95S7G+eEWK0f3jYKTbV6bOwepjfhtafsvUsqrQvrGC8YhmnO9cSCk3yuY984F1vesdHYhWJ5FvASlacshUsajFt2mUM9pqzvKGcyNJW0arTKN1GGGzQlH0tXwLDgQTurS8eIQAAAABJRU5ErkJggg==';
|
|
|
|
const IMAGE_PREFETCH_URL = 'http://origami.design/public/images/bird-logo.png?r=1&t=' + Date.now();
|
|
const prefetchTask = Image.prefetch(IMAGE_PREFETCH_URL);
|
|
|
|
class NetworkImageCallbackExample extends React.Component {
|
|
constructor(props, context) {
|
|
super(props, context);
|
|
this.state = {
|
|
events: [],
|
|
startLoadPrefetched: false,
|
|
mountTime: new Date()
|
|
};
|
|
}
|
|
|
|
componentWillMount() {
|
|
this.setState({ mountTime: new Date() });
|
|
}
|
|
|
|
render() {
|
|
const { mountTime } = this.state;
|
|
|
|
return (
|
|
<View>
|
|
<Image
|
|
onLoad={() => this._loadEventFired(`✔ onLoad (+${new Date() - mountTime}ms)`)}
|
|
onLoadEnd={() => {
|
|
this._loadEventFired(`✔ onLoadEnd (+${new Date() - mountTime}ms)`);
|
|
this.setState({ startLoadPrefetched: true }, () => {
|
|
prefetchTask.then(
|
|
() => {
|
|
this._loadEventFired(`✔ Prefetch OK (+${new Date() - mountTime}ms)`);
|
|
},
|
|
error => {
|
|
this._loadEventFired(`✘ Prefetch failed (+${new Date() - mountTime}ms)`);
|
|
console.log(error);
|
|
}
|
|
);
|
|
});
|
|
}}
|
|
onLoadStart={() => this._loadEventFired(`✔ onLoadStart (+${new Date() - mountTime}ms)`)}
|
|
source={this.props.source}
|
|
style={[styles.base, { overflow: 'visible' }]}
|
|
/>
|
|
{this.state.startLoadPrefetched
|
|
? <Image
|
|
onLoad={() =>
|
|
this._loadEventFired(`✔ (prefetched) onLoad (+${new Date() - mountTime}ms)`)}
|
|
onLoadEnd={() =>
|
|
this._loadEventFired(`✔ (prefetched) onLoadEnd (+${new Date() - mountTime}ms)`)}
|
|
onLoadStart={() =>
|
|
this._loadEventFired(`✔ (prefetched) onLoadStart (+${new Date() - mountTime}ms)`)}
|
|
source={this.props.prefetchedSource}
|
|
style={[styles.base, { overflow: 'visible' }]}
|
|
/>
|
|
: null}
|
|
<Text style={{ marginTop: 20 }}>
|
|
{this.state.events.join('\n')}
|
|
</Text>
|
|
</View>
|
|
);
|
|
}
|
|
|
|
_loadEventFired = event => {
|
|
this.setState(state => (state.events = [...state.events, event]));
|
|
};
|
|
}
|
|
|
|
class NetworkImageExample extends React.Component {
|
|
state = {
|
|
error: false,
|
|
loading: false,
|
|
progress: 0
|
|
};
|
|
|
|
render() {
|
|
const loader = this.state.loading
|
|
? <View style={styles.progress}>
|
|
<Text>{this.state.progress}%</Text>
|
|
<ActivityIndicator style={{ marginLeft: 5 }} />
|
|
</View>
|
|
: null;
|
|
return this.state.error
|
|
? <Text>{this.state.error}</Text>
|
|
: <Image
|
|
onError={e => this.setState({ error: e.nativeEvent.error, loading: false })}
|
|
onLoad={() => this.setState({ loading: false, error: false })}
|
|
onLoadStart={e => this.setState({ loading: true })}
|
|
onProgress={e =>
|
|
this.setState({
|
|
progress: Math.round(100 * e.nativeEvent.loaded / e.nativeEvent.total)
|
|
})}
|
|
source={this.props.source}
|
|
style={[styles.base, { overflow: 'visible' }]}
|
|
>
|
|
{loader}
|
|
</Image>;
|
|
}
|
|
}
|
|
|
|
class ImageSizeExample extends React.Component {
|
|
state = {
|
|
width: 0,
|
|
height: 0
|
|
};
|
|
|
|
componentDidMount() {
|
|
Image.getSize(this.props.source.uri, (width, height) => {
|
|
this.setState({ width, height });
|
|
});
|
|
}
|
|
|
|
render() {
|
|
return (
|
|
<View>
|
|
<Text>
|
|
Actual dimensions:{'\n'}
|
|
width: {this.state.width}, height: {this.state.height}
|
|
</Text>
|
|
<Image
|
|
source={this.props.source}
|
|
style={{
|
|
backgroundColor: '#eee',
|
|
height: 227,
|
|
marginTop: 10,
|
|
width: 323
|
|
}}
|
|
/>
|
|
</View>
|
|
);
|
|
}
|
|
}
|
|
|
|
/*
|
|
var MultipleSourcesExample = createReactClass({
|
|
getInitialState: function() {
|
|
return {
|
|
width: 30,
|
|
height: 30,
|
|
};
|
|
},
|
|
render: function() {
|
|
return (
|
|
<View style={styles.container}>
|
|
<View style={{flexDirection: 'row', justifyContent: 'space-between'}}>
|
|
<Text
|
|
style={styles.touchableText}
|
|
onPress={this.decreaseImageSize} >
|
|
Decrease image size
|
|
</Text>
|
|
<Text
|
|
style={styles.touchableText}
|
|
onPress={this.increaseImageSize} >
|
|
Increase image size
|
|
</Text>
|
|
</View>
|
|
<Text>Container image size: {this.state.width}x{this.state.height} </Text>
|
|
<View
|
|
style={[styles.imageContainer, {height: this.state.height, width: this.state.width}]} >
|
|
<Image
|
|
style={{flex: 1}}
|
|
source={[
|
|
{uri: 'http://facebook.github.io/react/img/logo_small.png', width: 38, height: 38},
|
|
{uri: 'http://facebook.github.io/react/img/logo_small_2x.png', width: 76, height: 76},
|
|
{uri: 'http://facebook.github.io/react/img/logo_og.png', width: 400, height: 400}
|
|
]}
|
|
/>
|
|
</View>
|
|
</View>
|
|
);
|
|
},
|
|
increaseImageSize: function() {
|
|
if (this.state.width >= 100) {
|
|
return;
|
|
}
|
|
this.setState({
|
|
width: this.state.width + 10,
|
|
height: this.state.height + 10,
|
|
});
|
|
},
|
|
decreaseImageSize: function() {
|
|
if (this.state.width <= 10) {
|
|
return;
|
|
}
|
|
this.setState({
|
|
width: this.state.width - 10,
|
|
height: this.state.height - 10,
|
|
});
|
|
},
|
|
});
|
|
*/
|
|
|
|
const examples = [
|
|
{
|
|
title: 'Plain network image',
|
|
description:
|
|
'If the `source` prop `uri` property is prefixed with ' +
|
|
'"http", then it will be downloaded from the network.',
|
|
render: function() {
|
|
return (
|
|
<Image
|
|
source={{
|
|
uri: 'http://facebook.github.io/react/img/logo_og.png',
|
|
width: 1200,
|
|
height: 630
|
|
}}
|
|
style={styles.base}
|
|
/>
|
|
);
|
|
}
|
|
},
|
|
{
|
|
title: 'Plain static image',
|
|
description:
|
|
'Static assets should be placed in the source code tree, and ' +
|
|
'required in the same way as JavaScript modules.',
|
|
render: function() {
|
|
return (
|
|
<View style={styles.horizontal}>
|
|
<Image source={require('./uie_thumb_normal@2x.png')} style={styles.icon} />
|
|
<Image source={require('./uie_thumb_selected@2x.png')} style={styles.icon} />
|
|
{/*<Image source={require('./uie_comment_normal.png')} style={styles.icon} />*/}
|
|
{/*<Image source={require('./uie_comment_highlighted.png')} style={styles.icon} />*/}
|
|
</View>
|
|
);
|
|
}
|
|
},
|
|
{
|
|
title: 'Image loading events',
|
|
render() {
|
|
return (
|
|
<NetworkImageCallbackExample
|
|
prefetchedSource={{ uri: IMAGE_PREFETCH_URL }}
|
|
source={{ uri: 'http://origami.design/public/images/bird-logo.png?r=1&t=' + Date.now() }}
|
|
/>
|
|
);
|
|
}
|
|
},
|
|
{
|
|
title: 'Error handler',
|
|
render() {
|
|
return (
|
|
<NetworkImageExample
|
|
source={{ uri: 'http://TYPO_ERROR_facebook.github.io/react/img/logo_og.png' }}
|
|
/>
|
|
);
|
|
}
|
|
},
|
|
{
|
|
title: 'defaultSource',
|
|
description: 'Show a placeholder image when a network image is loading',
|
|
render() {
|
|
return (
|
|
<Image
|
|
defaultSource={require('./bunny.png')}
|
|
source={{ uri: 'http://facebook.github.io/origami/public/images/birds.jpg' }}
|
|
style={styles.base}
|
|
/>
|
|
);
|
|
}
|
|
},
|
|
{
|
|
title: 'Border color',
|
|
render() {
|
|
return (
|
|
<View style={styles.horizontal}>
|
|
<Image
|
|
source={smallImage}
|
|
style={[
|
|
styles.base,
|
|
styles.background,
|
|
{
|
|
borderColor: '#f099f0',
|
|
borderWidth: 3
|
|
}
|
|
]}
|
|
/>
|
|
</View>
|
|
);
|
|
}
|
|
},
|
|
{
|
|
title: 'Border width',
|
|
render() {
|
|
return (
|
|
<View style={styles.horizontal}>
|
|
<Image
|
|
source={smallImage}
|
|
style={[
|
|
styles.base,
|
|
styles.background,
|
|
{
|
|
borderColor: '#f099f0',
|
|
borderWidth: 5
|
|
}
|
|
]}
|
|
/>
|
|
</View>
|
|
);
|
|
}
|
|
},
|
|
{
|
|
title: 'Border radius',
|
|
render() {
|
|
return (
|
|
<View style={styles.horizontal}>
|
|
<Image source={fullImage} style={[styles.base, { borderRadius: 5 }]} />
|
|
<Image
|
|
source={fullImage}
|
|
style={[styles.base, styles.leftMargin, { borderRadius: 19 }]}
|
|
/>
|
|
</View>
|
|
);
|
|
}
|
|
},
|
|
{
|
|
title: 'Background color',
|
|
render() {
|
|
return (
|
|
<View style={styles.horizontal}>
|
|
<Image source={smallImage} style={styles.base} />
|
|
<Image
|
|
source={smallImage}
|
|
style={[styles.base, styles.leftMargin, { backgroundColor: 'rgba(0, 0, 100, 0.25)' }]}
|
|
/>
|
|
<Image
|
|
source={smallImage}
|
|
style={[styles.base, styles.leftMargin, { backgroundColor: 'red' }]}
|
|
/>
|
|
<Image
|
|
source={smallImage}
|
|
style={[styles.base, styles.leftMargin, { backgroundColor: 'black' }]}
|
|
/>
|
|
</View>
|
|
);
|
|
}
|
|
},
|
|
{
|
|
title: 'Opacity',
|
|
render() {
|
|
return (
|
|
<View style={styles.horizontal}>
|
|
<Image source={fullImage} style={[styles.base, { opacity: 1 }]} />
|
|
<Image source={fullImage} style={[styles.base, styles.leftMargin, { opacity: 0.8 }]} />
|
|
<Image source={fullImage} style={[styles.base, styles.leftMargin, { opacity: 0.6 }]} />
|
|
<Image source={fullImage} style={[styles.base, styles.leftMargin, { opacity: 0.4 }]} />
|
|
<Image source={fullImage} style={[styles.base, styles.leftMargin, { opacity: 0.2 }]} />
|
|
<Image source={fullImage} style={[styles.base, styles.leftMargin, { opacity: 0 }]} />
|
|
</View>
|
|
);
|
|
}
|
|
},
|
|
{
|
|
title: 'Nesting',
|
|
render() {
|
|
return (
|
|
<Image source={fullImage} style={{ width: 60, height: 60, backgroundColor: 'transparent' }}>
|
|
<Text style={styles.nestedText}>
|
|
React
|
|
</Text>
|
|
</Image>
|
|
);
|
|
}
|
|
},
|
|
{
|
|
title: 'Resize mode',
|
|
description: 'The `resizeMode` style prop controls how the image is rendered within the frame.',
|
|
render() {
|
|
return (
|
|
<View>
|
|
{[smallImage, fullImage].map((image, index) => {
|
|
return (
|
|
<View key={index}>
|
|
<View style={styles.horizontal}>
|
|
<View>
|
|
<Text style={[styles.resizeModeText]}>
|
|
Contain
|
|
</Text>
|
|
<Image
|
|
resizeMode={Image.resizeMode.contain}
|
|
source={image}
|
|
style={styles.resizeMode}
|
|
/>
|
|
</View>
|
|
<View style={styles.leftMargin}>
|
|
<Text style={[styles.resizeModeText]}>
|
|
Cover
|
|
</Text>
|
|
<Image
|
|
resizeMode={Image.resizeMode.cover}
|
|
source={image}
|
|
style={styles.resizeMode}
|
|
/>
|
|
</View>
|
|
</View>
|
|
<View style={styles.horizontal}>
|
|
<View>
|
|
<Text style={[styles.resizeModeText]}>
|
|
Stretch
|
|
</Text>
|
|
<Image
|
|
resizeMode={Image.resizeMode.stretch}
|
|
source={image}
|
|
style={styles.resizeMode}
|
|
/>
|
|
</View>
|
|
<View style={styles.leftMargin}>
|
|
<Text style={[styles.resizeModeText]}>
|
|
Repeat
|
|
</Text>
|
|
<Image resizeMode={'repeat'} source={image} style={styles.resizeMode} />
|
|
</View>
|
|
<View style={styles.leftMargin}>
|
|
<Text style={[styles.resizeModeText]}>
|
|
Center
|
|
</Text>
|
|
<Image resizeMode={'center'} source={image} style={styles.resizeMode} />
|
|
</View>
|
|
</View>
|
|
</View>
|
|
);
|
|
})}
|
|
</View>
|
|
);
|
|
}
|
|
},
|
|
{
|
|
title: 'Animated GIF',
|
|
render() {
|
|
return (
|
|
<Image
|
|
source={{
|
|
uri:
|
|
'http://38.media.tumblr.com/9e9bd08c6e2d10561dd1fb4197df4c4e/tumblr_mfqekpMktw1rn90umo1_500.gif'
|
|
}}
|
|
style={styles.gif}
|
|
/>
|
|
);
|
|
}
|
|
},
|
|
{
|
|
title: 'Base64 image',
|
|
render() {
|
|
return <Image source={{ uri: base64Icon, scale: 3 }} style={styles.base64} />;
|
|
}
|
|
},
|
|
{
|
|
title: 'Image dimensions',
|
|
description:
|
|
'`Image.getSize` provides the dimensions of an image as soon as they are available (i.e., before loading is complete)',
|
|
render() {
|
|
return (
|
|
<ImageSizeExample
|
|
source={{
|
|
uri: 'https://upload.wikimedia.org/wikipedia/commons/d/d7/Chestnut-mandibled_Toucan.jpg'
|
|
}}
|
|
/>
|
|
);
|
|
}
|
|
}
|
|
];
|
|
|
|
const fullImage = { uri: 'http://facebook.github.io/react/img/logo_og.png' };
|
|
const smallImage = { uri: 'http://facebook.github.io/react/img/logo_small_2x.png' };
|
|
|
|
const styles = StyleSheet.create({
|
|
base: {
|
|
height: 38,
|
|
width: 38
|
|
},
|
|
progress: {
|
|
alignItems: 'center',
|
|
flex: 1,
|
|
flexDirection: 'row',
|
|
width: 100
|
|
},
|
|
leftMargin: {
|
|
marginLeft: 10
|
|
},
|
|
background: {
|
|
backgroundColor: '#222222'
|
|
},
|
|
sectionText: {
|
|
marginVertical: 6
|
|
},
|
|
nestedText: {
|
|
backgroundColor: 'transparent',
|
|
color: 'white',
|
|
marginLeft: 12,
|
|
marginTop: 20
|
|
},
|
|
resizeMode: {
|
|
borderColor: 'black',
|
|
borderWidth: 0.5,
|
|
height: 60,
|
|
width: 90
|
|
},
|
|
resizeModeText: {
|
|
fontSize: 11,
|
|
marginBottom: 3
|
|
},
|
|
icon: {
|
|
height: 15,
|
|
width: 15
|
|
},
|
|
horizontal: {
|
|
flexDirection: 'row'
|
|
},
|
|
gif: {
|
|
flex: 1,
|
|
height: 200
|
|
},
|
|
base64: {
|
|
flex: 1,
|
|
height: 50,
|
|
resizeMode: 'contain'
|
|
},
|
|
touchableText: {
|
|
color: 'blue',
|
|
fontWeight: '500'
|
|
}
|
|
});
|
|
|
|
storiesOf('Components', module).add('Image', () =>
|
|
<UIExplorer
|
|
description="An accessibile image component with support for image resizing, default image, and child content."
|
|
examples={examples}
|
|
title="Image"
|
|
url="https://github.com/necolas/react-native-web/blob/master/docs/components/Image.md"
|
|
/>
|
|
);
|