diff --git a/packages/react-dom-bindings/src/client/ReactFiberConfigDOM.js b/packages/react-dom-bindings/src/client/ReactFiberConfigDOM.js index 62afafe11efc..bab0772ca57c 100644 --- a/packages/react-dom-bindings/src/client/ReactFiberConfigDOM.js +++ b/packages/react-dom-bindings/src/client/ReactFiberConfigDOM.js @@ -2173,6 +2173,7 @@ type PreloadOptions = { crossOrigin?: string, integrity?: string, type?: string, + fetchPriority?: 'high' | 'low' | 'auto', }; function preload(href: string, options: PreloadOptions) { if (!enableFloat) { @@ -2245,6 +2246,7 @@ function preloadPropsFromPreloadOptions( crossOrigin: as === 'font' ? '' : options.crossOrigin, integrity: options.integrity, type: options.type, + fetchPriority: options.fetchPriority, }; } @@ -2254,6 +2256,7 @@ type PreinitOptions = { crossOrigin?: string, integrity?: string, nonce?: string, + fetchPriority?: 'high' | 'low' | 'auto', }; function preinit(href: string, options: PreinitOptions) { if (!enableFloat) { @@ -2395,6 +2398,7 @@ function stylesheetPropsFromPreinitOptions( 'data-precedence': precedence, crossOrigin: options.crossOrigin, integrity: options.integrity, + fetchPriority: options.fetchPriority, }; } @@ -2408,6 +2412,7 @@ function scriptPropsFromPreinitOptions( crossOrigin: options.crossOrigin, integrity: options.integrity, nonce: options.nonce, + fetchPriority: options.fetchPriority, }; } diff --git a/packages/react-dom-bindings/src/server/ReactFizzConfigDOM.js b/packages/react-dom-bindings/src/server/ReactFizzConfigDOM.js index 9eeb12008299..85f9a2a82373 100644 --- a/packages/react-dom-bindings/src/server/ReactFizzConfigDOM.js +++ b/packages/react-dom-bindings/src/server/ReactFizzConfigDOM.js @@ -5102,6 +5102,7 @@ type PreloadOptions = { crossOrigin?: string, integrity?: string, type?: string, + fetchPriority?: 'high' | 'low' | 'auto', }; export function preload(href: string, options: PreloadOptions) { if (!enableFloat) { @@ -5247,6 +5248,7 @@ type PreinitOptions = { crossOrigin?: string, integrity?: string, nonce?: string, + fetchPriority?: 'high' | 'low' | 'auto', }; function preinit(href: string, options: PreinitOptions): void { if (!enableFloat) { @@ -5590,6 +5592,7 @@ function preloadPropsFromPreloadOptions( crossOrigin: as === 'font' ? '' : options.crossOrigin, integrity: options.integrity, type: options.type, + fetchPriority: options.fetchPriority, }; } @@ -5631,6 +5634,7 @@ function stylesheetPropsFromPreinitOptions( 'data-precedence': precedence, crossOrigin: options.crossOrigin, integrity: options.integrity, + fetchPriority: options.fetchPriority, }; } @@ -5662,6 +5666,7 @@ function scriptPropsFromPreinitOptions( crossOrigin: options.crossOrigin, integrity: options.integrity, nonce: options.nonce, + fetchPriority: options.fetchPriority, }; } diff --git a/packages/react-dom/src/ReactDOMDispatcher.js b/packages/react-dom/src/ReactDOMDispatcher.js index 0c2b85945006..2bd42537400a 100644 --- a/packages/react-dom/src/ReactDOMDispatcher.js +++ b/packages/react-dom/src/ReactDOMDispatcher.js @@ -14,6 +14,7 @@ export type PreloadOptions = { crossOrigin?: string, integrity?: string, type?: string, + fetchPriority?: 'high' | 'low' | 'auto', }; export type PreinitOptions = { as: string, @@ -21,6 +22,7 @@ export type PreinitOptions = { crossOrigin?: string, integrity?: string, nonce?: string, + fetchPriority?: 'high' | 'low' | 'auto', }; export type HostDispatcher = { diff --git a/packages/react-dom/src/__tests__/ReactDOMFloat-test.js b/packages/react-dom/src/__tests__/ReactDOMFloat-test.js index b0303165d0db..4b14030757ef 100644 --- a/packages/react-dom/src/__tests__/ReactDOMFloat-test.js +++ b/packages/react-dom/src/__tests__/ReactDOMFloat-test.js @@ -3789,6 +3789,7 @@ body { as: 'style', crossOrigin: 'use-credentials', integrity: 'some hash', + fetchPriority: 'low', }); return ( @@ -3909,6 +3910,113 @@ body { 'ReactDOM.preload(): For `href` "foo", The options provided conflict with props on a matching element. When the preload options disagree with the underlying resource it usually means the browser will not be able to use the preload when the resource is fetched, negating any benefit the preload would provide. React will preload the resource using props derived from the resource instead and ignore the options provided to the `ReactDOM.preload()` call. In general, preloading is useful when you expect to render a resource soon but have not yet done so. In this case since the underlying resource was already rendered the preload call may be extraneous. Try removing the call, otherwise try adjusting both the props on the and the options passed to `ReactDOM.preload()` to agree.\n "integrity" missing from options, underlying prop value: "some hash"\n "media" missing from options, underlying prop value: "print"\n "crossOrigin" option value: "use-credentials", missing from underlying props', ]); }); + + it('supports fetchPriority', async () => { + function Component({isServer}) { + ReactDOM.preload(isServer ? 'highserver' : 'highclient', { + as: 'script', + fetchPriority: 'high', + }); + ReactDOM.preload(isServer ? 'lowserver' : 'lowclient', { + as: 'style', + fetchPriority: 'low', + }); + ReactDOM.preload(isServer ? 'autoserver' : 'autoclient', { + as: 'style', + fetchPriority: 'auto', + }); + return 'hello'; + } + + await act(() => { + renderToPipeableStream( + + + + + , + ).pipe(writable); + }); + + expect(getMeaningfulChildren(document)).toEqual( + + + + + + + hello + , + ); + + ReactDOMClient.hydrateRoot( + document, + + + + + , + ); + await waitForAll([]); + expect(getMeaningfulChildren(document)).toEqual( + + + + + + + + + + hello + , + ); + }); }); describe('ReactDOM.preinit(href, { as: ... })', () => { @@ -4442,7 +4550,6 @@ body { hello , ); - await clientAct(() => { ReactDOMClient.hydrateRoot( document, @@ -4453,7 +4560,6 @@ body { , ); }); - expect(getMeaningfulChildren(document)).toEqual( @@ -4474,6 +4580,97 @@ body { , ); }); + + it('supports fetchPriority', async () => { + function Component({isServer}) { + ReactDOM.preinit(isServer ? 'highserver' : 'highclient', { + as: 'script', + fetchPriority: 'high', + }); + ReactDOM.preinit(isServer ? 'lowserver' : 'lowclient', { + as: 'style', + fetchPriority: 'low', + }); + ReactDOM.preinit(isServer ? 'autoserver' : 'autoclient', { + as: 'style', + fetchPriority: 'auto', + }); + return 'hello'; + } + + await act(() => { + renderToPipeableStream( + + + + + , + ).pipe(writable); + }); + + expect(getMeaningfulChildren(document)).toEqual( + + + + +