1. /**
    
  2.  * Copyright (c) Meta Platforms, Inc. and affiliates.
    
  3.  *
    
  4.  * This source code is licensed under the MIT license found in the
    
  5.  * LICENSE file in the root directory of this source tree.
    
  6.  *
    
  7.  * @flow
    
  8.  */
    
  9. 
    
  10. import type {ReactContext} from 'shared/ReactTypes';
    
  11. 
    
  12. import * as React from 'react';
    
  13. import {createContext, useCallback, useContext, useEffect} from 'react';
    
  14. import {createResource} from '../../cache';
    
  15. import {BridgeContext, StoreContext} from '../context';
    
  16. import {TreeStateContext} from './TreeContext';
    
  17. import {separateDisplayNameAndHOCs} from 'react-devtools-shared/src/utils';
    
  18. 
    
  19. import type {OwnersList} from 'react-devtools-shared/src/backend/types';
    
  20. import type {
    
  21.   Element,
    
  22.   SerializedElement,
    
  23. } from 'react-devtools-shared/src/frontend/types';
    
  24. import type {Resource, Thenable} from '../../cache';
    
  25. 
    
  26. type Context = (id: number) => Array<SerializedElement> | null;
    
  27. 
    
  28. const OwnersListContext: ReactContext<Context> = createContext<Context>(
    
  29.   ((null: any): Context),
    
  30. );
    
  31. OwnersListContext.displayName = 'OwnersListContext';
    
  32. 
    
  33. type ResolveFn = (ownersList: Array<SerializedElement> | null) => void;
    
  34. type InProgressRequest = {
    
  35.   promise: Thenable<Array<SerializedElement>>,
    
  36.   resolveFn: ResolveFn,
    
  37. };
    
  38. 
    
  39. const inProgressRequests: WeakMap<Element, InProgressRequest> = new WeakMap();
    
  40. const resource: Resource<
    
  41.   Element,
    
  42.   Element,
    
  43.   Array<SerializedElement>,
    
  44. > = createResource(
    
  45.   (element: Element) => {
    
  46.     const request = inProgressRequests.get(element);
    
  47.     if (request != null) {
    
  48.       return request.promise;
    
  49.     }
    
  50. 
    
  51.     let resolveFn:
    
  52.       | ResolveFn
    
  53.       | ((
    
  54.           result: Promise<Array<SerializedElement>> | Array<SerializedElement>,
    
  55.         ) => void) = ((null: any): ResolveFn);
    
  56.     const promise = new Promise(resolve => {
    
  57.       resolveFn = resolve;
    
  58.     });
    
  59. 
    
  60.     // $FlowFixMe[incompatible-call] found when upgrading Flow
    
  61.     inProgressRequests.set(element, {promise, resolveFn});
    
  62. 
    
  63.     return (promise: $FlowFixMe);
    
  64.   },
    
  65.   (element: Element) => element,
    
  66.   {useWeakMap: true},
    
  67. );
    
  68. 
    
  69. type Props = {
    
  70.   children: React$Node,
    
  71. };
    
  72. 
    
  73. function OwnersListContextController({children}: Props): React.Node {
    
  74.   const bridge = useContext(BridgeContext);
    
  75.   const store = useContext(StoreContext);
    
  76.   const {ownerID} = useContext(TreeStateContext);
    
  77. 
    
  78.   const read = useCallback(
    
  79.     (id: number) => {
    
  80.       const element = store.getElementByID(id);
    
  81.       if (element !== null) {
    
  82.         return resource.read(element);
    
  83.       } else {
    
  84.         return null;
    
  85.       }
    
  86.     },
    
  87.     [store],
    
  88.   );
    
  89. 
    
  90.   useEffect(() => {
    
  91.     const onOwnersList = (ownersList: OwnersList) => {
    
  92.       const id = ownersList.id;
    
  93. 
    
  94.       const element = store.getElementByID(id);
    
  95.       if (element !== null) {
    
  96.         const request = inProgressRequests.get(element);
    
  97.         if (request != null) {
    
  98.           inProgressRequests.delete(element);
    
  99. 
    
  100.           request.resolveFn(
    
  101.             ownersList.owners === null
    
  102.               ? null
    
  103.               : ownersList.owners.map(owner => {
    
  104.                   const [displayNameWithoutHOCs, hocDisplayNames] =
    
  105.                     separateDisplayNameAndHOCs(owner.displayName, owner.type);
    
  106. 
    
  107.                   return {
    
  108.                     ...owner,
    
  109.                     displayName: displayNameWithoutHOCs,
    
  110.                     hocDisplayNames,
    
  111.                   };
    
  112.                 }),
    
  113.           );
    
  114.         }
    
  115.       }
    
  116.     };
    
  117. 
    
  118.     bridge.addListener('ownersList', onOwnersList);
    
  119.     return () => bridge.removeListener('ownersList', onOwnersList);
    
  120.   }, [bridge, store]);
    
  121. 
    
  122.   // This effect requests an updated owners list any time the selected owner changes
    
  123.   useEffect(() => {
    
  124.     if (ownerID !== null) {
    
  125.       const rendererID = store.getRendererIDForElement(ownerID);
    
  126.       if (rendererID !== null) {
    
  127.         bridge.send('getOwnersList', {id: ownerID, rendererID});
    
  128.       }
    
  129.     }
    
  130. 
    
  131.     return () => {};
    
  132.   }, [bridge, ownerID, store]);
    
  133. 
    
  134.   return (
    
  135.     <OwnersListContext.Provider value={read}>
    
  136.       {children}
    
  137.     </OwnersListContext.Provider>
    
  138.   );
    
  139. }
    
  140. 
    
  141. export {OwnersListContext, OwnersListContextController};