diff --git a/packages/react-native-web-examples/pages/image/index.js b/packages/react-native-web-examples/pages/image/index.js index 5cc756bf4..7d95ae1b7 100644 --- a/packages/react-native-web-examples/pages/image/index.js +++ b/packages/react-native-web-examples/pages/image/index.js @@ -15,6 +15,18 @@ const dataBase64Svg = ''; const dataSvg = 'data:image/svg+xml;utf8,'; +const sourceWithHeaders = { + uri: placeholder, + headers: { + 'x-token': '0012345' + } +}; +const sourceWithHeadersAndRedirect = { + uri: source, + headers: { + 'x-token': '0012345' + } +}; function Divider() { return ; @@ -114,6 +126,17 @@ export default function ImagePage() { /> + + + + With Headers + + + + Headers & Redirect + + + ); } diff --git a/packages/react-native-web/src/exports/Image/__tests__/__snapshots__/index-test.js.snap b/packages/react-native-web/src/exports/Image/__tests__/__snapshots__/index-test.js.snap index 04b41e3ef..f46206ac4 100644 --- a/packages/react-native-web/src/exports/Image/__tests__/__snapshots__/index-test.js.snap +++ b/packages/react-native-web/src/exports/Image/__tests__/__snapshots__/index-test.js.snap @@ -329,14 +329,14 @@ exports[`components/Image prop "style" removes other unsupported View styles 1`] >
{ beforeEach(() => { ImageUriCache._entries = {}; + ImageLoader.load = jest.fn((source, onLoad, onError) => { + onLoad(); + return { source, cancel: jest.fn() }; + }); window.Image = jest.fn(() => ({})); }); @@ -107,9 +111,6 @@ describe('components/Image', () => { describe('prop "onLoad"', () => { test('is called after image is loaded from network', () => { jest.useFakeTimers(); - ImageLoader.load = jest.fn().mockImplementation((_, onLoad, onError) => { - onLoad(); - }); const onLoadStartStub = jest.fn(); const onLoadStub = jest.fn(); const onLoadEndStub = jest.fn(); @@ -127,9 +128,6 @@ describe('components/Image', () => { test('is called after image is loaded from cache', () => { jest.useFakeTimers(); - ImageLoader.load = jest.fn().mockImplementation((_, onLoad, onError) => { - onLoad(); - }); const onLoadStartStub = jest.fn(); const onLoadStub = jest.fn(); const onLoadEndStub = jest.fn(); @@ -261,11 +259,6 @@ describe('components/Image', () => { test('is set immediately if the image was preloaded', () => { const uri = 'https://yahoo.com/favicon.ico'; - ImageLoader.load = jest - .fn() - .mockImplementationOnce((_, onLoad, onError) => { - onLoad(); - }); return Image.prefetch(uri).then(() => { const source = { uri }; const { container } = render(, { @@ -311,8 +304,9 @@ describe('components/Image', () => { let loadCallback; ImageLoader.load = jest .fn() - .mockImplementationOnce((_, onLoad, onError) => { + .mockImplementationOnce((source, onLoad, onError) => { loadCallback = onLoad; + return { source, cancel: jest.fn() }; }); const { container } = render( diff --git a/packages/react-native-web/src/exports/Image/index.js b/packages/react-native-web/src/exports/Image/index.js index d68898dea..fa28788fa 100644 --- a/packages/react-native-web/src/exports/Image/index.js +++ b/packages/react-native-web/src/exports/Image/index.js @@ -8,6 +8,7 @@ * @flow */ +import type { LoadRequest } from '../../modules/ImageLoader'; import type { ImageProps } from './types'; import * as React from 'react'; @@ -146,6 +147,15 @@ function resolveAssetUri(source): ?string { return uri; } +function hasSourceDiff(a, b) { + if (a.uri !== b.uri) return true; + + const headersA = a.headers || {}; + const headersB = b.headers || {}; + + return JSON.stringify(headersA) !== JSON.stringify(headersB); +} + interface ImageStatics { getSize: ( uri: string, @@ -201,7 +211,11 @@ const Image: React.AbstractComponent< const hasTextAncestor = React.useContext(TextAncestorContext); const hiddenImageRef = React.useRef(null); const filterRef = React.useRef(_filterId++); - const requestRef = React.useRef(null); + const requestRef = React.useRef({ + source: { uri: '' }, + cancel: () => {}, + requestId: -1 + }); const shouldDisplaySource = state === LOADED || (state === LOADING && defaultSource == null); const [flatStyle, _resizeMode, filter, tintColor] = getFlatStyle( @@ -258,16 +272,27 @@ const Image: React.AbstractComponent< // Image loading const uri = resolveAssetUri(source); React.useEffect(() => { - abortPendingRequest(); - if (uri != null) { + const nextSource = { uri, headers: undefined }; + if (source && source.headers != null) { + nextSource.headers = source.headers; + } + + if (!hasSourceDiff(nextSource, requestRef.current.source)) return; + + requestRef.current.cancel(); updateState(LOADING); if (onLoadStart) { onLoadStart(); } - requestRef.current = ImageLoader.load( - uri, + const makeRequest = nextSource.headers + ? ImageLoader.loadWithHeaders + : ImageLoader.load; + + requestRef.current = makeRequest( + // $FlowFixMe + nextSource, function load(e) { updateState(LOADED); if (onLoad) { @@ -291,17 +316,14 @@ const Image: React.AbstractComponent< } } ); + } else { + requestRef.current.source = { uri: '' }; + requestRef.current.cancel(); } + }, [uri, updateState, onError, onLoad, onLoadEnd, onLoadStart, source]); - function abortPendingRequest() { - if (requestRef.current != null) { - ImageLoader.abort(requestRef.current); - requestRef.current = null; - } - } - - return abortPendingRequest; - }, [uri, requestRef, updateState, onError, onLoad, onLoadEnd, onLoadStart]); + // Run the cleanup function on unmount + React.useEffect(() => requestRef.current.cancel, []); return ( { + image.onload = (nativeEvent) => { // avoid blocking the main thread - const onDecode = () => onLoad({ nativeEvent: e }); + const onDecode = () => { + // Append `source` to match RN's ImageLoadEvent interface + nativeEvent.source = { + uri: image.src, + width: image.naturalWidth, + height: image.naturalHeight + }; + + onLoad({ nativeEvent }); + }; if (typeof image.decode === 'function') { // Safari currently throws exceptions when decoding svgs. // We want to catch that error and allow the load handler @@ -134,14 +147,58 @@ const ImageLoader = { setTimeout(onDecode, 0); } }; - image.src = uri; + image.src = source.uri; requests[`${id}`] = image; - return id; + + return { + source, + cancel: () => ImageLoader.abort(id), + requestId: id + }; + }, + loadWithHeaders( + source: ImageSource, + onLoad: Function, + onError: Function + ): LoadRequest { + let loadRequest: LoadRequest; + let uri: string; + const abortCtrl = new AbortController(); + const request = new Request(source.uri, { + headers: source.headers, + signal: abortCtrl.signal + }); + request.headers.append('accept', 'image/*'); + + fetch(request) + .then((response) => response.blob()) + .then((blob) => { + uri = URL.createObjectURL(blob); + loadRequest = ImageLoader.load({ uri }, onLoad, onError); + }) + .catch((error) => { + if (error.name !== 'AbortError' && onError) { + onError(error); + } + }); + + return { + source, + get requestId() { + if (loadRequest) return loadRequest.requestId; + return -1; + }, + cancel: () => { + abortCtrl.abort(); + if (loadRequest) loadRequest.cancel(); + URL.revokeObjectURL(uri); + } + }; }, prefetch(uri: string): Promise { return new Promise((resolve, reject) => { ImageLoader.load( - uri, + { uri }, () => { // Add the uri to the cache so it can be immediately displayed when used // but also immediately remove it to correctly reflect that it has no active references @@ -164,4 +221,15 @@ const ImageLoader = { } }; +export type LoadRequest = {| + requestId: number, + source: ImageSource | { uri: string }, + cancel: Function +|}; + +type ImageSource = { + uri: string, + headers: { [key: string]: string } +}; + export default ImageLoader;