2015-02-19 20:10:52 -08:00
/ * *
2015-03-23 13:35:08 -07: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-02-19 20:10:52 -08:00
*
2015-03-26 10:06:50 -07:00
* @ flow
2018-01-14 19:32:26 -08:00
* @ format
2015-02-19 20:10:52 -08:00
* /
'use strict' ;
2018-03-14 14:04:19 -07:00
const ImageProps = require ( 'ImageProps' ) ;
2016-06-07 05:29:13 -07:00
const NativeModules = require ( 'NativeModules' ) ;
const React = require ( 'React' ) ;
2018-06-19 21:28:04 -07:00
const ReactNative = require ( 'ReactNative' ) ;
2016-06-07 05:29:13 -07:00
const StyleSheet = require ( 'StyleSheet' ) ;
const flattenStyle = require ( 'flattenStyle' ) ;
const requireNativeComponent = require ( 'requireNativeComponent' ) ;
const resolveAssetSource = require ( 'resolveAssetSource' ) ;
const ImageViewManager = NativeModules . ImageViewManager ;
Added getImageSize method
Summary:
public
This diff adds a `getSize()` method to `Image` to retrieve the width and height of an image prior to displaying it. This is useful when working with images from uncontrolled sources, and has been a much-requested feature.
In order to retrieve the image dimensions, the image may first need to be loaded or downloaded, after which it will be cached. This means that in principle you could use this method to preload images, however it is not optimized for that purpose, and may in future be implemented in a way that does not fully load/download the image data.
A fully supported way to preload images will be provided in a future diff.
The API (separate success and failure callbacks) is far from ideal, but until we agree on a unified standard, this was the most conventional way I could think of to implement it. If it returned a promise or something similar, it would be unique among all such APIS in the framework.
Please note that this has been a long time coming, in part due to much bikeshedding about what the API should look like, so while it's not unlikely that the API may change in future, I think having *some* way to do this is better than waiting until we can define the "perfect" way.
Reviewed By: vjeux
Differential Revision: D2797365
fb-gh-sync-id: 11eb1b8547773b1f8be0bc55ddf6dfedebf7fc0a
2015-12-31 18:50:26 -08:00
2018-06-01 12:37:22 -07:00
const RCTImageView = requireNativeComponent ( 'RCTImageView' ) ;
2018-08-14 16:31:04 -07:00
import type { ImageStyleProp } from 'StyleSheet' ;
2018-06-19 21:28:04 -07:00
import type { ImageProps as ImagePropsType } from 'ImageProps' ;
function getSize (
uri : string ,
success : ( width : number , height : number ) => void ,
failure ? : ( error : any ) => void ,
) {
ImageViewManager . getSize (
uri ,
success ,
failure ||
function ( ) {
console . warn ( 'Failed to get size for image: ' + uri ) ;
} ,
) ;
}
function prefetch ( url : string ) {
return ImageViewManager . prefetchImage ( url ) ;
}
2018-08-20 16:06:18 -07:00
async function queryCache ( urls : Array < string > ) : Promise < Map < string , 'memory' | 'disk' >> {
return await ImageViewManager . queryCache ( urls ) ;
}
2018-06-19 21:28:04 -07:00
declare class ImageComponentType extends ReactNative . NativeComponent <
ImagePropsType ,
> {
static getSize : typeof getSize ;
static prefetch : typeof prefetch ;
2018-08-20 16:06:18 -07:00
static queryCache : typeof queryCache ;
2018-06-19 21:28:04 -07:00
static resolveAssetSource : typeof resolveAssetSource ;
static propTypes : typeof ImageProps ;
}
2015-02-19 20:10:52 -08:00
/ * *
2015-04-27 13:55:01 -07:00
* A React component for displaying different types of images ,
2015-02-19 20:10:52 -08:00
* including network images , static resources , temporary local images , and
2015-03-09 09:28:51 -07:00
* images from local disk , such as the camera roll .
2015-02-19 20:10:52 -08:00
*
2018-01-29 16:10:49 -08:00
* See https : //facebook.github.io/react-native/docs/image.html
2015-02-19 20:10:52 -08:00
* /
2018-06-19 21:28:04 -07:00
let Image = (
props : ImagePropsType ,
forwardedRef : ? React . Ref < 'RCTImageView' > ,
) => {
const source = resolveAssetSource ( props . source ) || {
uri : undefined ,
width : undefined ,
height : undefined ,
} ;
let sources ;
2018-08-14 16:31:04 -07:00
let style : ImageStyleProp ;
2018-06-19 21:28:04 -07:00
if ( Array . isArray ( source ) ) {
2018-08-14 16:31:04 -07:00
// $FlowFixMe flattenStyle is not strong enough
2018-06-19 21:28:04 -07:00
style = flattenStyle ( [ styles . base , props . style ] ) || { } ;
sources = source ;
} else {
const { width , height , uri } = source ;
2018-08-14 16:31:04 -07:00
// $FlowFixMe flattenStyle is not strong enough
2018-06-19 21:28:04 -07:00
style = flattenStyle ( [ { width , height } , styles . base , props . style ] ) || { } ;
sources = [ source ] ;
if ( uri === '' ) {
console . warn ( 'source.uri should not be an empty string' ) ;
}
}
2015-02-19 20:10:52 -08:00
2018-06-21 11:56:37 -07:00
const resizeMode = props . resizeMode || style . resizeMode || 'cover' ;
2018-06-19 23:02:50 -07:00
const tintColor = style . tintColor ;
2015-02-19 20:10:52 -08:00
2018-06-19 21:28:04 -07:00
if ( props . src != null ) {
console . warn (
'The <Image> component requires a `source` property rather than `src`.' ,
) ;
}
2015-02-19 20:10:52 -08:00
2018-06-19 21:28:04 -07:00
if ( props . children != null ) {
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.' ,
) ;
}
return (
< RCTImageView
{ ... props }
ref = { forwardedRef }
style = { style }
resizeMode = { resizeMode }
tintColor = { tintColor }
source = { sources }
/ >
) ;
} ;
// $FlowFixMe - TODO T29156721 `React.forwardRef` is not defined in Flow, yet.
Image = React . forwardRef ( Image ) ;
2016-06-06 13:20:09 -07:00
2018-06-19 21:28:04 -07:00
/ * *
* Retrieve the width and height ( in pixels ) of an image prior to displaying it .
*
* See https : //facebook.github.io/react-native/docs/image.html#getsize
* /
Image . getSize = getSize ;
2016-07-28 13:58:50 -07:00
2018-06-19 21:28:04 -07:00
/ * *
* Prefetches a remote image for later use by downloading it to the disk
* cache .
*
* See https : //facebook.github.io/react-native/docs/image.html#prefetch
* /
Image . prefetch = prefetch ;
2016-03-04 07:31:38 -08:00
2018-08-20 16:06:18 -07:00
/ * *
* Performs cache interrogation .
*
* See https : //facebook.github.io/react-native/docs/image.html#querycache
* /
Image . queryCache = queryCache ;
2018-06-19 21:28:04 -07:00
/ * *
* Resolves an asset reference into an object .
*
* See https : //facebook.github.io/react-native/docs/image.html#resolveassetsource
* /
Image . resolveAssetSource = resolveAssetSource ;
2017-09-25 21:55:56 -07:00
2018-06-19 21:28:04 -07:00
Image . propTypes = ImageProps ;
2015-02-19 20:10:52 -08:00
2016-06-07 05:29:13 -07:00
const styles = StyleSheet . create ( {
2015-02-19 20:10:52 -08:00
base : {
overflow : 'hidden' ,
} ,
} ) ;
2018-06-19 21:28:04 -07:00
module . exports = ( Image : Class < ImageComponentType > ) ;