mirror of
https://github.com/facebook/react.git
synced 2025-11-01 09:12:30 +00:00
168 lines
4.8 KiB
JavaScript
168 lines
4.8 KiB
JavaScript
/**
|
|
* Copyright (c) Meta Platforms, Inc. and affiliates.
|
|
*
|
|
* This source code is licensed under the MIT license found in the
|
|
* LICENSE file in the root directory of this source tree.
|
|
*
|
|
* @flow
|
|
*/
|
|
|
|
import type {ReactContext} from 'shared/ReactTypes';
|
|
|
|
import * as React from 'react';
|
|
import {createContext, useCallback, useContext, useEffect} from 'react';
|
|
import {createResource} from '../../cache';
|
|
import {BridgeContext, StoreContext} from '../context';
|
|
import {TreeDispatcherContext, TreeStateContext} from './TreeContext';
|
|
import {backendToFrontendSerializedElementMapper} from 'react-devtools-shared/src/utils';
|
|
|
|
import type {OwnersList} from 'react-devtools-shared/src/backend/types';
|
|
import type {
|
|
Element,
|
|
SerializedElement,
|
|
} from 'react-devtools-shared/src/frontend/types';
|
|
import type {Resource, Thenable} from '../../cache';
|
|
|
|
type Context = (id: number) => Array<SerializedElement> | null;
|
|
|
|
const OwnersListContext: ReactContext<Context> = createContext<Context>(
|
|
((null: any): Context),
|
|
);
|
|
OwnersListContext.displayName = 'OwnersListContext';
|
|
|
|
type ResolveFn = (ownersList: Array<SerializedElement> | null) => void;
|
|
type InProgressRequest = {
|
|
promise: Thenable<Array<SerializedElement>>,
|
|
resolveFn: ResolveFn,
|
|
};
|
|
|
|
const inProgressRequests: WeakMap<Element, InProgressRequest> = new WeakMap();
|
|
const resource: Resource<
|
|
Element,
|
|
Element,
|
|
Array<SerializedElement>,
|
|
> = createResource(
|
|
(element: Element) => {
|
|
const request = inProgressRequests.get(element);
|
|
if (request != null) {
|
|
return request.promise;
|
|
}
|
|
|
|
let resolveFn:
|
|
| ResolveFn
|
|
| ((
|
|
result: Promise<Array<SerializedElement>> | Array<SerializedElement>,
|
|
) => void) = ((null: any): ResolveFn);
|
|
const promise = new Promise(resolve => {
|
|
resolveFn = resolve;
|
|
});
|
|
|
|
// $FlowFixMe[incompatible-call] found when upgrading Flow
|
|
inProgressRequests.set(element, {promise, resolveFn});
|
|
|
|
return (promise: $FlowFixMe);
|
|
},
|
|
(element: Element) => element,
|
|
{useWeakMap: true},
|
|
);
|
|
|
|
type Props = {
|
|
children: React$Node,
|
|
};
|
|
|
|
function useChangeOwnerAction(): (nextOwnerID: number) => void {
|
|
const bridge = useContext(BridgeContext);
|
|
const store = useContext(StoreContext);
|
|
const treeAction = useContext(TreeDispatcherContext);
|
|
|
|
return useCallback(
|
|
function changeOwnerAction(nextOwnerID: number) {
|
|
treeAction({type: 'SELECT_OWNER', payload: nextOwnerID});
|
|
|
|
const element = store.getElementByID(nextOwnerID);
|
|
if (element !== null) {
|
|
if (!inProgressRequests.has(element)) {
|
|
let resolveFn:
|
|
| ResolveFn
|
|
| ((
|
|
result:
|
|
| Promise<Array<SerializedElement>>
|
|
| Array<SerializedElement>,
|
|
) => void) = ((null: any): ResolveFn);
|
|
const promise = new Promise(resolve => {
|
|
resolveFn = resolve;
|
|
});
|
|
|
|
// $FlowFixMe[incompatible-call] found when upgrading Flow
|
|
inProgressRequests.set(element, {promise, resolveFn});
|
|
}
|
|
|
|
const rendererID = store.getRendererIDForElement(nextOwnerID);
|
|
if (rendererID !== null) {
|
|
bridge.send('getOwnersList', {id: nextOwnerID, rendererID});
|
|
}
|
|
}
|
|
},
|
|
[bridge, store],
|
|
);
|
|
}
|
|
|
|
function OwnersListContextController({children}: Props): React.Node {
|
|
const bridge = useContext(BridgeContext);
|
|
const store = useContext(StoreContext);
|
|
const {ownerID} = useContext(TreeStateContext);
|
|
|
|
const read = useCallback(
|
|
(id: number) => {
|
|
const element = store.getElementByID(id);
|
|
if (element !== null) {
|
|
return resource.read(element);
|
|
} else {
|
|
return null;
|
|
}
|
|
},
|
|
[store],
|
|
);
|
|
|
|
useEffect(() => {
|
|
const onOwnersList = (ownersList: OwnersList) => {
|
|
const id = ownersList.id;
|
|
|
|
const element = store.getElementByID(id);
|
|
if (element !== null) {
|
|
const request = inProgressRequests.get(element);
|
|
if (request != null) {
|
|
request.resolveFn(
|
|
ownersList.owners === null
|
|
? null
|
|
: ownersList.owners.map(backendToFrontendSerializedElementMapper),
|
|
);
|
|
}
|
|
}
|
|
};
|
|
|
|
bridge.addListener('ownersList', onOwnersList);
|
|
return () => bridge.removeListener('ownersList', onOwnersList);
|
|
}, [bridge, store]);
|
|
|
|
// This effect requests an updated owners list any time the selected owner changes
|
|
useEffect(() => {
|
|
if (ownerID !== null) {
|
|
const rendererID = store.getRendererIDForElement(ownerID);
|
|
if (rendererID !== null) {
|
|
bridge.send('getOwnersList', {id: ownerID, rendererID});
|
|
}
|
|
}
|
|
|
|
return () => {};
|
|
}, [bridge, ownerID, store]);
|
|
|
|
return (
|
|
<OwnersListContext.Provider value={read}>
|
|
{children}
|
|
</OwnersListContext.Provider>
|
|
);
|
|
}
|
|
|
|
export {OwnersListContext, OwnersListContextController, useChangeOwnerAction};
|