Skip to content
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.

Commit 2202d34

Browse files
committedOct 3, 2019
[Selective Hydration] ReactDOM.unstable_scheduleHydration(domNode)
Adds an API to explicitly prioritize hydrating the path to a particular DOM node without relying on events to do it. The API uses the current scheduler priority to schedule it. For the same priority, the last one wins. This allows a similar effect as continuous events. This is useful for example to hydrate based on scroll position. I considered having an API that explicitly overrides the current target(s). However that makes it difficult to coordinate across components in an app. This just hydrates on target at a time but if it is blocked on I/O we could consider increasing priority of later targets too.
1 parent de2edc2 commit 2202d34

File tree

4 files changed

+180
-2
lines changed

4 files changed

+180
-2
lines changed
 

‎packages/react-dom/src/__tests__/ReactDOMServerSelectiveHydration-test.internal.js

+50
Original file line numberDiff line numberDiff line change
@@ -440,4 +440,54 @@ describe('ReactDOMServerSelectiveHydration', () => {
440440

441441
document.body.removeChild(container);
442442
});
443+
444+
it('hydrates the last explicitly hydrated target at higher priority', async () => {
445+
function Child({text}) {
446+
Scheduler.unstable_yieldValue(text);
447+
return <span>{text}</span>;
448+
}
449+
450+
function App() {
451+
Scheduler.unstable_yieldValue('App');
452+
return (
453+
<div>
454+
<Suspense fallback="Loading...">
455+
<Child text="A" />
456+
</Suspense>
457+
<Suspense fallback="Loading...">
458+
<Child text="B" />
459+
</Suspense>
460+
<Suspense fallback="Loading...">
461+
<Child text="C" />
462+
</Suspense>
463+
</div>
464+
);
465+
}
466+
467+
let finalHTML = ReactDOMServer.renderToString(<App />);
468+
469+
expect(Scheduler).toHaveYielded(['App', 'A', 'B', 'C']);
470+
471+
let container = document.createElement('div');
472+
container.innerHTML = finalHTML;
473+
474+
let spanB = container.getElementsByTagName('span')[1];
475+
let spanC = container.getElementsByTagName('span')[2];
476+
477+
// A and D will be suspended. We'll click on D which should take
478+
// priority, after we unsuspend.
479+
let root = ReactDOM.unstable_createRoot(container, {hydrate: true});
480+
root.render(<App />);
481+
482+
// Nothing has been hydrated so far.
483+
expect(Scheduler).toHaveYielded([]);
484+
485+
// Increase priority of B and then C.
486+
ReactDOM.unstable_scheduleHydration(spanB);
487+
ReactDOM.unstable_scheduleHydration(spanC);
488+
489+
// We should prioritize hydrating C first because the last added
490+
// gets highest priority followed by the next added.
491+
expect(Scheduler).toFlushAndYield(['App', 'C', 'B', 'A']);
492+
});
443493
});

‎packages/react-dom/src/client/ReactDOM.js

+11-1
Original file line numberDiff line numberDiff line change
@@ -42,6 +42,7 @@ import {
4242
attemptSynchronousHydration,
4343
attemptUserBlockingHydration,
4444
attemptContinuousHydration,
45+
attemptHydrationAtCurrentPriority,
4546
} from 'react-reconciler/inline.dom';
4647
import {createPortal as createPortalImpl} from 'shared/ReactPortal';
4748
import {canUseDOM} from 'shared/ExecutionEnvironment';
@@ -81,8 +82,10 @@ import {
8182
setAttemptSynchronousHydration,
8283
setAttemptUserBlockingHydration,
8384
setAttemptContinuousHydration,
85+
setAttemptHydrationAtCurrentPriority,
86+
eagerlyTrapReplayableEvents,
87+
queueExplicitHydrationTarget,
8488
} from '../events/ReactDOMEventReplaying';
85-
import {eagerlyTrapReplayableEvents} from '../events/ReactDOMEventReplaying';
8689
import {
8790
ELEMENT_NODE,
8891
COMMENT_NODE,
@@ -94,6 +97,7 @@ import {ROOT_ATTRIBUTE_NAME} from '../shared/DOMProperty';
9497
setAttemptSynchronousHydration(attemptSynchronousHydration);
9598
setAttemptUserBlockingHydration(attemptUserBlockingHydration);
9699
setAttemptContinuousHydration(attemptContinuousHydration);
100+
setAttemptHydrationAtCurrentPriority(attemptHydrationAtCurrentPriority);
97101

98102
const ReactCurrentOwner = ReactSharedInternals.ReactCurrentOwner;
99103

@@ -841,6 +845,12 @@ const ReactDOM: Object = {
841845
unstable_createSyncRoot: createSyncRoot,
842846
unstable_flushControlled: flushControlled,
843847

848+
unstable_scheduleHydration(target: Node) {
849+
if (target) {
850+
queueExplicitHydrationTarget(target);
851+
}
852+
},
853+
844854
__SECRET_INTERNALS_DO_NOT_USE_OR_YOU_WILL_BE_FIRED: {
845855
// Keep in sync with ReactDOMUnstableNativeDependencies.js
846856
// ReactTestUtils.js, and ReactTestUtilsAct.js. This is an array for better minification.

‎packages/react-dom/src/events/ReactDOMEventReplaying.js

+107-1
Original file line numberDiff line numberDiff line change
@@ -11,15 +11,23 @@ import type {AnyNativeEvent} from 'legacy-events/PluginModuleType';
1111
import type {Container, SuspenseInstance} from '../client/ReactDOMHostConfig';
1212
import type {DOMTopLevelEventType} from 'legacy-events/TopLevelEventTypes';
1313
import type {EventSystemFlags} from 'legacy-events/EventSystemFlags';
14+
import type {FiberRoot} from 'react-reconciler/src/ReactFiberRoot';
1415

1516
import {
1617
enableFlareAPI,
1718
enableSelectiveHydration,
1819
} from 'shared/ReactFeatureFlags';
1920
import {
21+
unstable_runWithPriority as runWithPriority,
2022
unstable_scheduleCallback as scheduleCallback,
2123
unstable_NormalPriority as NormalPriority,
24+
unstable_getCurrentPriorityLevel as getCurrentPriorityLevel,
2225
} from 'scheduler';
26+
import {
27+
getNearestMountedFiber,
28+
getContainerFromFiber,
29+
getSuspenseInstanceFromFiber,
30+
} from 'react-reconciler/reflection';
2331
import {
2432
attemptToDispatchEvent,
2533
trapEventForResponderEventSystem,
@@ -28,8 +36,12 @@ import {
2836
getListeningSetForElement,
2937
listenToTopLevel,
3038
} from './ReactBrowserEventEmitter';
31-
import {getInstanceFromNode} from '../client/ReactDOMComponentTree';
39+
import {
40+
getInstanceFromNode,
41+
getClosestInstanceFromNode,
42+
} from '../client/ReactDOMComponentTree';
3243
import {unsafeCastDOMTopLevelTypeToString} from 'legacy-events/TopLevelEventTypes';
44+
import {HostRoot, SuspenseComponent} from 'shared/ReactWorkTags';
3345

3446
let attemptSynchronousHydration: (fiber: Object) => void;
3547

@@ -49,6 +61,14 @@ export function setAttemptContinuousHydration(fn: (fiber: Object) => void) {
4961
attemptContinuousHydration = fn;
5062
}
5163

64+
let attemptHydrationAtCurrentPriority: (fiber: Object) => void;
65+
66+
export function setAttemptHydrationAtCurrentPriority(
67+
fn: (fiber: Object) => void,
68+
) {
69+
attemptHydrationAtCurrentPriority = fn;
70+
}
71+
5272
// TODO: Upgrade this definition once we're on a newer version of Flow that
5373
// has this definition built-in.
5474
type PointerEvent = Event & {
@@ -124,6 +144,13 @@ let queuedPointers: Map<number, QueuedReplayableEvent> = new Map();
124144
let queuedPointerCaptures: Map<number, QueuedReplayableEvent> = new Map();
125145
// We could consider replaying selectionchange and touchmoves too.
126146

147+
type QueuedHydrationTarget = {|
148+
blockedOn: null | Container | SuspenseInstance,
149+
target: Node,
150+
priority: number,
151+
|};
152+
let queuedExplicitHydrationTargets: Array<QueuedHydrationTarget> = [];
153+
127154
export function hasQueuedDiscreteEvents(): boolean {
128155
return queuedDiscreteEvents.length > 0;
129156
}
@@ -422,6 +449,64 @@ export function queueIfContinuousEvent(
422449
return false;
423450
}
424451

452+
// Check if this target is unblocked. Returns true if it's unblocked.
453+
function attemptExplicitHydrationTarget(
454+
queuedTarget: QueuedHydrationTarget,
455+
): void {
456+
// TODO: This function shares a lot of logic with attemptToDispatchEvent.
457+
// Try to unify them. It's a bit tricky since it would require two return
458+
// values.
459+
let targetInst = getClosestInstanceFromNode(queuedTarget.target);
460+
if (targetInst !== null) {
461+
let nearestMounted = getNearestMountedFiber(targetInst);
462+
if (nearestMounted !== null) {
463+
const tag = nearestMounted.tag;
464+
if (tag === SuspenseComponent) {
465+
let instance = getSuspenseInstanceFromFiber(nearestMounted);
466+
if (instance !== null) {
467+
// We're blocked on hydrating this boundary.
468+
// Increase its priority.
469+
queuedTarget.blockedOn = instance;
470+
runWithPriority(queuedTarget.priority, () => {
471+
attemptHydrationAtCurrentPriority(nearestMounted);
472+
});
473+
return;
474+
}
475+
} else if (tag === HostRoot) {
476+
const root: FiberRoot = nearestMounted.stateNode;
477+
if (root.hydrate) {
478+
queuedTarget.blockedOn = getContainerFromFiber(nearestMounted);
479+
// We don't currently have a way to increase the priority of
480+
// a root other than sync.
481+
return;
482+
}
483+
}
484+
}
485+
}
486+
queuedTarget.blockedOn = null;
487+
}
488+
489+
export function queueExplicitHydrationTarget(target: Node): void {
490+
if (enableSelectiveHydration) {
491+
let priority = getCurrentPriorityLevel();
492+
const queuedTarget: QueuedHydrationTarget = {
493+
blockedOn: null,
494+
target: target,
495+
priority: priority,
496+
};
497+
let i = 0;
498+
for (; i < queuedExplicitHydrationTargets.length; i++) {
499+
if (priority <= queuedExplicitHydrationTargets[i].priority) {
500+
break;
501+
}
502+
}
503+
queuedExplicitHydrationTargets.splice(i, 0, queuedTarget);
504+
if (i === 0) {
505+
attemptExplicitHydrationTarget(queuedTarget);
506+
}
507+
}
508+
}
509+
425510
function attemptReplayContinuousQueuedEvent(
426511
queuedEvent: QueuedReplayableEvent,
427512
): boolean {
@@ -544,4 +629,25 @@ export function retryIfBlockedOn(
544629
scheduleCallbackIfUnblocked(queuedEvent, unblocked);
545630
queuedPointers.forEach(unblock);
546631
queuedPointerCaptures.forEach(unblock);
632+
633+
for (let i = 0; i < queuedExplicitHydrationTargets.length; i++) {
634+
let queuedTarget = queuedExplicitHydrationTargets[i];
635+
if (queuedTarget.blockedOn === unblocked) {
636+
queuedTarget.blockedOn = null;
637+
}
638+
}
639+
640+
while (queuedExplicitHydrationTargets.length > 0) {
641+
let nextExplicitTarget = queuedExplicitHydrationTargets[0];
642+
if (nextExplicitTarget.blockedOn !== null) {
643+
// We're still blocked.
644+
break;
645+
} else {
646+
attemptExplicitHydrationTarget(nextExplicitTarget);
647+
if (nextExplicitTarget.blockedOn === null) {
648+
// We're unblocked.
649+
queuedExplicitHydrationTargets.shift();
650+
}
651+
}
652+
}
547653
}

‎packages/react-reconciler/src/ReactFiberReconciler.js

+12
Original file line numberDiff line numberDiff line change
@@ -438,6 +438,18 @@ export function attemptContinuousHydration(fiber: Fiber): void {
438438
markRetryTimeIfNotHydrated(fiber, expTime);
439439
}
440440

441+
export function attemptHydrationAtCurrentPriority(fiber: Fiber): void {
442+
if (fiber.tag !== SuspenseComponent) {
443+
// We ignore HostRoots here because we can't increase
444+
// their priority other than synchronously flush it.
445+
return;
446+
}
447+
const currentTime = requestCurrentTime();
448+
const expTime = computeExpirationForFiber(currentTime, fiber, null);
449+
scheduleWork(fiber, expTime);
450+
markRetryTimeIfNotHydrated(fiber, expTime);
451+
}
452+
441453
export {findHostInstance};
442454

443455
export {findHostInstanceWithWarning};

0 commit comments

Comments
 (0)
Please sign in to comment.