2015-09-14 15:35:58 +01:00
/ * *
* Copyright ( c ) 2015 - present , Facebook , Inc .
*
2018-02-16 18:24:55 -08:00
* This source code is licensed under the MIT license found in the
* LICENSE file in the root directory of this source tree .
2015-09-14 15:35:58 +01:00
*
* @ flow
2018-01-14 19:32:26 -08:00
* @ format
2015-09-14 15:35:58 +01:00
* /
2018-05-09 00:47:46 -07:00
2015-09-14 15:35:58 +01:00
'use strict' ;
2018-05-10 15:44:52 -07:00
const ImageResizeMode = require ( 'ImageResizeMode' ) ;
const ImageStylePropTypes = require ( 'ImageStylePropTypes' ) ;
const NativeMethodsMixin = require ( 'NativeMethodsMixin' ) ;
const NativeModules = require ( 'NativeModules' ) ;
const React = require ( 'React' ) ;
const PropTypes = require ( 'prop-types' ) ;
const ReactNativeViewAttributes = require ( 'ReactNativeViewAttributes' ) ;
const StyleSheet = require ( 'StyleSheet' ) ;
const StyleSheetPropType = require ( 'StyleSheetPropType' ) ;
2018-05-09 00:47:46 -07:00
const TextAncestor = require ( 'TextAncestor' ) ;
2018-05-10 15:44:52 -07:00
const ViewPropTypes = require ( 'ViewPropTypes' ) ;
2015-09-14 15:35:58 +01:00
2018-05-10 15:44:52 -07:00
const createReactClass = require ( 'create-react-class' ) ;
const flattenStyle = require ( 'flattenStyle' ) ;
const merge = require ( 'merge' ) ;
const requireNativeComponent = require ( 'requireNativeComponent' ) ;
const resolveAssetSource = require ( 'resolveAssetSource' ) ;
2015-09-14 15:35:58 +01:00
2018-05-10 15:44:52 -07:00
const { ImageLoader } = NativeModules ;
2016-04-13 07:29:10 -07:00
2016-07-18 12:07:29 -07:00
let _requestId = 1 ;
function generateRequestId ( ) {
return _requestId ++ ;
}
2015-09-14 15:35:58 +01:00
/ * *
2018-01-29 16:10:49 -08:00
* A React component for displaying different types of images ,
2015-09-14 15:35:58 +01:00
* including network images , static resources , temporary local images , and
2018-01-29 16:10:49 -08:00
* images from local disk , such as the camera roll .
2015-09-14 15:35:58 +01:00
*
2018-01-29 16:10:49 -08:00
* See https : //facebook.github.io/react-native/docs/image.html
2015-09-14 15:35:58 +01:00
* /
2018-05-10 15:44:52 -07:00
const Image = createReactClass ( {
2017-07-07 14:24:25 -07:00
displayName : 'Image' ,
2015-09-14 15:35:58 +01:00
propTypes : {
2017-03-24 00:22:57 -07:00
... ViewPropTypes ,
2016-02-09 19:15:52 -08:00
style : StyleSheetPropType ( ImageStylePropTypes ) ,
2018-01-14 19:32:26 -08:00
/ * *
2018-01-29 16:10:49 -08:00
* See https : //facebook.github.io/react-native/docs/image.html#source
2015-10-09 09:31:51 -07:00
* /
source : PropTypes . oneOfType ( [
PropTypes . shape ( {
uri : PropTypes . string ,
2017-02-18 04:33:59 -08:00
headers : PropTypes . objectOf ( PropTypes . string ) ,
2015-10-09 09:31:51 -07:00
} ) ,
// Opaque type returned by require('./image.jpg')
PropTypes . number ,
2016-06-13 14:04:19 -07:00
// Multiple sources
PropTypes . arrayOf (
PropTypes . shape ( {
uri : PropTypes . string ,
width : PropTypes . number ,
height : PropTypes . number ,
2017-10-02 23:19:31 -07:00
headers : PropTypes . objectOf ( PropTypes . string ) ,
2018-01-14 19:32:26 -08:00
} ) ,
) ,
2016-02-09 19:15:52 -08:00
] ) ,
2017-03-02 07:32:08 -08:00
/ * *
2018-01-14 19:32:26 -08:00
* blurRadius : the blur radius of the blur filter added to the image
2018-01-29 16:10:49 -08:00
*
* See https : //facebook.github.io/react-native/docs/image.html#blurradius
2018-01-14 19:32:26 -08:00
* /
2017-03-02 07:32:08 -08:00
blurRadius : PropTypes . number ,
2018-04-06 15:57:09 -07:00
/ * *
2018-04-19 02:30:39 -07:00
* See https : //facebook.github.io/react-native/docs/image.html#defaultsource
* /
2018-04-06 15:57:09 -07:00
defaultSource : PropTypes . number ,
2015-12-06 15:44:47 -08:00
/ * *
2018-01-29 16:10:49 -08:00
* See https : //facebook.github.io/react-native/docs/image.html#loadingindicatorsource
2015-12-06 15:44:47 -08:00
* /
loadingIndicatorSource : PropTypes . oneOfType ( [
PropTypes . shape ( {
uri : PropTypes . string ,
} ) ,
// Opaque type returned by require('./image.jpg')
PropTypes . number ,
] ) ,
2015-11-16 11:39:19 -08:00
progressiveRenderingEnabled : PropTypes . bool ,
fadeDuration : PropTypes . number ,
2015-11-25 17:06:59 -08:00
/ * *
* Invoked on load start
* /
onLoadStart : PropTypes . func ,
2016-11-14 19:42:34 -08:00
/ * *
* Invoked on load error
* /
onError : PropTypes . func ,
2015-11-25 17:06:59 -08:00
/ * *
* Invoked when load completes successfully
* /
onLoad : PropTypes . func ,
/ * *
* Invoked when load either succeeds or fails
* /
onLoadEnd : PropTypes . func ,
2015-09-14 15:35:58 +01:00
/ * *
* Used to locate this view in end - to - end tests .
* /
testID : PropTypes . string ,
2016-09-09 05:00:52 -07:00
/ * *
* The mechanism that should be used to resize the image when the image ' s dimensions
* differ from the image view ' s dimensions . Defaults to ` auto ` .
*
2018-01-29 16:10:49 -08:00
* See https : //facebook.github.io/react-native/docs/image.html#resizemethod
2016-09-09 05:00:52 -07:00
* /
resizeMethod : PropTypes . oneOf ( [ 'auto' , 'resize' , 'scale' ] ) ,
2016-08-01 00:49:14 -07:00
/ * *
* Determines how to resize the image when the frame doesn ' t match the raw
* image dimensions .
*
2018-01-29 16:10:49 -08:00
* See https : //facebook.github.io/react-native/docs/image.html#resizemode
2016-08-01 00:49:14 -07:00
* /
2018-04-19 02:30:39 -07:00
resizeMode : PropTypes . oneOf ( [
'cover' ,
'contain' ,
'stretch' ,
'repeat' ,
'center' ,
] ) ,
2015-09-14 15:35:58 +01:00
} ,
statics : {
resizeMode : ImageResizeMode ,
2016-05-20 18:40:59 -07:00
getSize (
url : string ,
success : ( width : number , height : number ) => void ,
2017-04-05 10:16:30 -07:00
failure ? : ( error : any ) => void ,
2016-05-20 18:40:59 -07:00
) {
return ImageLoader . getSize ( url )
. then ( function ( sizes ) {
success ( sizes . width , sizes . height ) ;
} )
2018-01-14 19:32:26 -08:00
. catch (
failure ||
function ( ) {
console . warn ( 'Failed to get size for image: ' + url ) ;
} ,
) ;
2016-05-20 18:40:59 -07:00
} ,
2016-04-13 07:29:10 -07:00
/ * *
* Prefetches a remote image for later use by downloading it to the disk
* cache
2018-01-29 16:10:49 -08:00
*
* See https : //facebook.github.io/react-native/docs/image.html#prefetch
2016-04-13 07:29:10 -07:00
* /
2016-07-18 12:07:29 -07:00
prefetch ( url : string , callback : ? Function ) {
const requestId = generateRequestId ( ) ;
callback && callback ( requestId ) ;
return ImageLoader . prefetchImage ( url , requestId ) ;
} ,
/ * *
2018-01-29 16:10:49 -08:00
* Abort prefetch request .
*
* See https : //facebook.github.io/react-native/docs/image.html#abortprefetch
2016-07-18 12:07:29 -07:00
* /
abortPrefetch ( requestId : number ) {
2016-07-23 00:16:32 -07:00
ImageLoader . abortRequest ( requestId ) ;
2016-04-13 07:29:10 -07:00
} ,
2016-08-31 05:06:10 -07:00
/ * *
* Perform cache interrogation .
*
2018-01-29 16:10:49 -08:00
* See https : //facebook.github.io/react-native/docs/image.html#querycache
2016-08-31 05:06:10 -07:00
* /
2018-01-14 19:32:26 -08:00
async queryCache (
urls : Array < string > ,
) : Promise < Map < string , 'memory' | 'disk' >> {
2016-08-31 05:06:10 -07:00
return await ImageLoader . queryCache ( urls ) ;
2016-11-14 21:01:02 -08:00
} ,
/ * *
2018-01-29 16:10:49 -08:00
* Resolves an asset reference into an object .
*
* See https : //facebook.github.io/react-native/docs/image.html#resolveassetsource
2016-11-14 21:01:02 -08:00
* /
resolveAssetSource : resolveAssetSource ,
2015-09-14 15:35:58 +01:00
} ,
mixins : [ NativeMethodsMixin ] ,
/ * *
* ` NativeMethodsMixin ` will look for this when invoking ` setNativeProps ` . We
2017-09-17 21:26:17 -07:00
* make ` this ` look like an actual native component class .
2015-09-14 15:35:58 +01:00
* /
viewConfig : {
uiViewClassName : 'RCTView' ,
2016-02-29 18:02:11 -08:00
validAttributes : ReactNativeViewAttributes . RCTView ,
2015-09-14 15:35:58 +01:00
} ,
render : function ( ) {
2016-06-13 14:04:19 -07:00
const source = resolveAssetSource ( this . props . source ) ;
2018-04-06 15:57:09 -07:00
const defaultSource = resolveAssetSource ( this . props . defaultSource ) ;
2018-01-14 19:32:26 -08:00
const loadingIndicatorSource = resolveAssetSource (
this . props . loadingIndicatorSource ,
) ;
2015-10-05 03:59:02 -07:00
2016-06-13 14:04:19 -07:00
// As opposed to the ios version, here we render `null` when there is no source, source.uri
// or source array.
2015-10-05 03:59:02 -07:00
if ( source && source . uri === '' ) {
console . warn ( 'source.uri should not be an empty string' ) ;
}
2016-03-04 07:31:38 -08:00
if ( this . props . src ) {
2018-01-14 19:32:26 -08:00
console . warn (
'The <Image> component requires a `source` property rather than `src`.' ,
) ;
2016-03-04 07:31:38 -08:00
}
2017-09-25 21:55:56 -07:00
if ( this . props . children ) {
2018-01-14 19:32:26 -08:00
throw new Error (
'The <Image> component cannot contain children. If you want to render content on top of the image, consider using the <ImageBackground> component or absolute positioning.' ,
) ;
2017-09-25 21:55:56 -07:00
}
2018-04-06 15:57:09 -07:00
if ( this . props . defaultSource && this . props . loadingIndicatorSource ) {
throw new Error (
'The <Image> component cannot have defaultSource and loadingIndicatorSource at the same time. Please use either defaultSource or loadingIndicatorSource.' ,
) ;
}
2018-05-09 00:47:46 -07:00
if ( ! source || ( ! source . uri && ! Array . isArray ( source ) ) ) {
return null ;
}
2015-09-14 15:35:58 +01:00
2018-05-09 00:47:46 -07:00
let style ;
let sources ;
if ( source . uri ) {
const { width , height } = source ;
style = flattenStyle ( [ { width , height } , styles . base , this . props . style ] ) ;
sources = [ { uri : source . uri } ] ;
} else {
style = flattenStyle ( [ styles . base , this . props . style ] ) ;
sources = source ;
2015-09-14 15:35:58 +01:00
}
2018-05-09 00:47:46 -07:00
const { onLoadStart , onLoad , onLoadEnd , onError } = this . props ;
const nativeProps = merge ( this . props , {
style ,
shouldNotifyLoadEvents : ! ! ( onLoadStart || onLoad || onLoadEnd || onError ) ,
src : sources ,
headers : source . headers ,
defaultSrc : defaultSource ? defaultSource . uri : null ,
loadingIndicatorSrc : loadingIndicatorSource
? loadingIndicatorSource . uri
: null ,
} ) ;
return (
< TextAncestor . Consumer >
{ hasTextAncestor =>
hasTextAncestor ? (
< RCTTextInlineImage { ... nativeProps } / >
) : (
< RKImage { ... nativeProps } / >
)
}
< / T e x t A n c e s t o r . C o n s u m e r >
) ;
2018-01-14 19:32:26 -08:00
} ,
2015-09-14 15:35:58 +01:00
} ) ;
2018-05-10 15:44:52 -07:00
const styles = StyleSheet . create ( {
2015-09-14 15:35:58 +01:00
base : {
overflow : 'hidden' ,
} ,
} ) ;
2018-05-10 15:44:52 -07:00
const cfg = {
2015-11-18 08:24:26 -08:00
nativeOnly : {
src : true ,
2017-02-18 04:33:59 -08:00
headers : true ,
2018-04-06 15:57:09 -07:00
defaultSrc : true ,
2015-12-06 15:44:47 -08:00
loadingIndicatorSrc : true ,
2015-11-25 17:06:59 -08:00
shouldNotifyLoadEvents : true ,
2015-11-18 08:24:26 -08:00
} ,
} ;
2018-05-10 15:44:52 -07:00
const RKImage = requireNativeComponent ( 'RCTImageView' , Image , cfg ) ;
const RCTTextInlineImage = requireNativeComponent (
2018-01-14 19:32:26 -08:00
'RCTTextInlineImage' ,
Image ,
cfg ,
) ;
2015-09-14 15:35:58 +01:00
module . exports = Image ;