Files
scrap/node_modules/next/dist/esm/client/components/use-reducer-with-devtools.js
2024-09-24 03:52:46 +00:00

134 lines
4.6 KiB
JavaScript

import React, { use, useContext } from "react";
import { useRef, useEffect, useCallback } from "react";
import { isThenable } from "./router-reducer/router-reducer-types";
import { ActionQueueContext } from "../../shared/lib/router/action-queue";
function normalizeRouterState(val) {
if (val instanceof Map) {
const obj = {};
for (const [key, value] of val.entries()){
if (typeof value === "function") {
obj[key] = "fn()";
continue;
}
if (typeof value === "object" && value !== null) {
if (value.$$typeof) {
obj[key] = value.$$typeof.toString();
continue;
}
if (value._bundlerConfig) {
obj[key] = "FlightData";
continue;
}
}
obj[key] = normalizeRouterState(value);
}
return obj;
}
if (typeof val === "object" && val !== null) {
const obj = {};
for(const key in val){
const value = val[key];
if (typeof value === "function") {
obj[key] = "fn()";
continue;
}
if (typeof value === "object" && value !== null) {
if (value.$$typeof) {
obj[key] = value.$$typeof.toString();
continue;
}
if (value.hasOwnProperty("_bundlerConfig")) {
obj[key] = "FlightData";
continue;
}
}
obj[key] = normalizeRouterState(value);
}
return obj;
}
if (Array.isArray(val)) {
return val.map(normalizeRouterState);
}
return val;
}
export function useUnwrapState(state) {
// reducer actions can be async, so sometimes we need to suspend until the state is resolved
if (isThenable(state)) {
const result = use(state);
return result;
}
return state;
}
function useReducerWithReduxDevtoolsNoop(initialState) {
return [
initialState,
()=>{},
()=>{}
];
}
function useReducerWithReduxDevtoolsImpl(initialState) {
const [state, setState] = React.useState(initialState);
const actionQueue = useContext(ActionQueueContext);
if (!actionQueue) {
throw new Error("Invariant: Missing ActionQueueContext");
}
const devtoolsConnectionRef = useRef();
const enabledRef = useRef();
useEffect(()=>{
if (devtoolsConnectionRef.current || enabledRef.current === false) {
return;
}
if (enabledRef.current === undefined && typeof window.__REDUX_DEVTOOLS_EXTENSION__ === "undefined") {
enabledRef.current = false;
return;
}
devtoolsConnectionRef.current = window.__REDUX_DEVTOOLS_EXTENSION__.connect({
instanceId: 8000,
name: "next-router"
});
if (devtoolsConnectionRef.current) {
devtoolsConnectionRef.current.init(normalizeRouterState(initialState));
if (actionQueue) {
actionQueue.devToolsInstance = devtoolsConnectionRef.current;
}
}
return ()=>{
devtoolsConnectionRef.current = undefined;
};
}, [
initialState,
actionQueue
]);
const dispatch = useCallback((action)=>{
if (!actionQueue.state) {
// we lazy initialize the mutable action queue state since the data needed
// to generate the state is not available when the actionQueue context is created
actionQueue.state = initialState;
}
actionQueue.dispatch(action, setState);
}, [
actionQueue,
initialState
]);
// Sync is called after a state update in the HistoryUpdater,
// for debugging purposes. Since the reducer state may be a Promise,
// we let the app router use() it and sync on the resolved value if
// something changed.
// Using the `state` here would be referentially unstable and cause
// undesirable re-renders and history updates.
const sync = useCallback((resolvedState)=>{
if (devtoolsConnectionRef.current) {
devtoolsConnectionRef.current.send({
type: "RENDER_SYNC"
}, normalizeRouterState(resolvedState));
}
}, []);
return [
state,
dispatch,
sync
];
}
export const useReducerWithReduxDevtools = typeof window !== "undefined" ? useReducerWithReduxDevtoolsImpl : useReducerWithReduxDevtoolsNoop;
//# sourceMappingURL=use-reducer-with-devtools.js.map