@@ -1381,4 +1381,111 @@ describe('ReactUse', () => {
1381
1381
assertLog ( [ 'B' , 'A' , 'C' ] ) ;
1382
1382
expect ( root ) . toMatchRenderedOutput ( 'BAC' ) ;
1383
1383
} ) ;
1384
+
1385
+ test ( 'basic Context as node' , async ( ) => {
1386
+ const Context = React . createContext ( null ) ;
1387
+
1388
+ function Indirection ( { children} ) {
1389
+ Scheduler . log ( 'Indirection' ) ;
1390
+ return children ;
1391
+ }
1392
+
1393
+ function ParentOfContextNode ( ) {
1394
+ Scheduler . log ( 'ParentOfContextNode' ) ;
1395
+ return Context ;
1396
+ }
1397
+
1398
+ function Child ( { text} ) {
1399
+ useEffect ( ( ) => {
1400
+ Scheduler . log ( 'Mount' ) ;
1401
+ return ( ) => {
1402
+ Scheduler . log ( 'Unmount' ) ;
1403
+ } ;
1404
+ } , [ ] ) ;
1405
+ return < Text text = { text } /> ;
1406
+ }
1407
+
1408
+ function App ( { contextValue, children} ) {
1409
+ const memoizedChildren = useMemo (
1410
+ ( ) => (
1411
+ < Indirection >
1412
+ < ParentOfContextNode />
1413
+ </ Indirection >
1414
+ ) ,
1415
+ [ children ] ,
1416
+ ) ;
1417
+ return (
1418
+ < Context . Provider value = { contextValue } >
1419
+ { memoizedChildren }
1420
+ </ Context . Provider >
1421
+ ) ;
1422
+ }
1423
+
1424
+ // Initial render
1425
+ const root = ReactNoop . createRoot ( ) ;
1426
+ await act ( ( ) => {
1427
+ root . render ( < App contextValue = { < Child text = "A" /> } /> ) ;
1428
+ } ) ;
1429
+ assertLog ( [ 'Indirection' , 'ParentOfContextNode' , 'A' , 'Mount' ] ) ;
1430
+ expect ( root ) . toMatchRenderedOutput ( 'A' ) ;
1431
+
1432
+ // Update the child to a new value
1433
+ await act ( async ( ) => {
1434
+ root . render ( < App contextValue = { < Child text = "B" /> } /> ) ;
1435
+ } ) ;
1436
+ assertLog ( [
1437
+ // Notice that the <Indirection /> did not rerender, because the
1438
+ // update was sent via Context.
1439
+
1440
+ // TODO: We shouldn't have to re-render the parent of the context node.
1441
+ // This happens because we need to reconcile the parent's children again.
1442
+ // However, we should be able to skip directly to reconcilation without
1443
+ // evaluating the component. One way to do this might be to mark the
1444
+ // context dependency with a flag that says it was added
1445
+ // during reconcilation.
1446
+ 'ParentOfContextNode' ,
1447
+
1448
+ // Notice that this was an update, not a remount.
1449
+ 'B' ,
1450
+ ] ) ;
1451
+ expect ( root ) . toMatchRenderedOutput ( 'B' ) ;
1452
+
1453
+ // Delete the old child and replace it with a new one, by changing the key
1454
+ await act ( async ( ) => {
1455
+ root . render ( < App contextValue = { < Child key = "C" text = "C" /> } /> ) ;
1456
+ } ) ;
1457
+ assertLog ( [
1458
+ 'ParentOfContextNode' ,
1459
+
1460
+ // A new instance is mounted
1461
+ 'C' ,
1462
+ 'Unmount' ,
1463
+ 'Mount' ,
1464
+ ] ) ;
1465
+ } ) ;
1466
+
1467
+ test ( 'context as node, at the root' , async ( ) => {
1468
+ const Context = React . createContext ( < Text text = "Hi" /> ) ;
1469
+ const root = ReactNoop . createRoot ( ) ;
1470
+ await act ( async ( ) => {
1471
+ startTransition ( ( ) => {
1472
+ root . render ( Context ) ;
1473
+ } ) ;
1474
+ } ) ;
1475
+ assertLog ( [ 'Hi' ] ) ;
1476
+ expect ( root ) . toMatchRenderedOutput ( 'Hi' ) ;
1477
+ } ) ;
1478
+
1479
+ test ( 'promises that resolves to a context, rendered as a node' , async ( ) => {
1480
+ const Context = React . createContext ( < Text text = "Hi" /> ) ;
1481
+ const promise = Promise . resolve ( Context ) ;
1482
+ const root = ReactNoop . createRoot ( ) ;
1483
+ await act ( async ( ) => {
1484
+ startTransition ( ( ) => {
1485
+ root . render ( promise ) ;
1486
+ } ) ;
1487
+ } ) ;
1488
+ assertLog ( [ 'Hi' ] ) ;
1489
+ expect ( root ) . toMatchRenderedOutput ( 'Hi' ) ;
1490
+ } ) ;
1384
1491
} ) ;
0 commit comments