154 lines
4.5 KiB
TypeScript
154 lines
4.5 KiB
TypeScript
"use client";
|
|
|
|
import {
|
|
createContext,
|
|
useCallback,
|
|
useContext,
|
|
useEffect,
|
|
useRef,
|
|
useState,
|
|
type ReactNode,
|
|
} from "react";
|
|
import type {
|
|
CreateFlowState,
|
|
CreateFlowContextValue,
|
|
CreateFlowStep,
|
|
} from "../types";
|
|
import {
|
|
clearAnonymousCreateFlowStorage,
|
|
clearLegacyCreateFlowKeysOnce,
|
|
readAnonymousCreateFlowState,
|
|
writeAnonymousCreateFlowState,
|
|
} from "../utils/anonymousDraftStorage";
|
|
import {
|
|
clearCoreValueDetailsLocalStorage,
|
|
readCoreValueDetailsFromLocalStorage,
|
|
writeCoreValueDetailsToLocalStorage,
|
|
} from "../utils/coreValueDetailsLocalStorage";
|
|
|
|
const CreateFlowContext = createContext<CreateFlowContextValue | null>(null);
|
|
|
|
interface CreateFlowProviderProps {
|
|
children: ReactNode;
|
|
initialStep?: CreateFlowStep | null;
|
|
/**
|
|
* When true (signed-out, session resolved), load/sync `create-flow-anonymous` in localStorage.
|
|
* When false, in-memory only (authenticated fresh create).
|
|
*/
|
|
enableAnonymousPersistence?: boolean;
|
|
}
|
|
|
|
/**
|
|
* Create flow state. Anonymous users mirror state to localStorage; authenticated users stay in memory.
|
|
*/
|
|
export function CreateFlowProvider({
|
|
children,
|
|
initialStep = null,
|
|
enableAnonymousPersistence = false,
|
|
}: CreateFlowProviderProps) {
|
|
const [state, setState] = useState<CreateFlowState>(() => {
|
|
const base = enableAnonymousPersistence
|
|
? readAnonymousCreateFlowState()
|
|
: {};
|
|
const storedDetails = readCoreValueDetailsFromLocalStorage();
|
|
if (Object.keys(storedDetails).length === 0) return base;
|
|
return {
|
|
...base,
|
|
coreValueDetailsByChipId: {
|
|
...storedDetails,
|
|
...(base.coreValueDetailsByChipId ?? {}),
|
|
},
|
|
};
|
|
});
|
|
const [interactionTouched, setInteractionTouched] = useState(false);
|
|
const [currentStep] = useState<CreateFlowStep | null>(initialStep);
|
|
const prevPersistRef = useRef(enableAnonymousPersistence);
|
|
|
|
useEffect(() => {
|
|
clearLegacyCreateFlowKeysOnce();
|
|
}, []);
|
|
|
|
// Session resolved as guest after initial paint: hydrate from localStorage if still empty.
|
|
useEffect(() => {
|
|
if (!enableAnonymousPersistence) {
|
|
prevPersistRef.current = false;
|
|
return;
|
|
}
|
|
const wasOff = !prevPersistRef.current;
|
|
prevPersistRef.current = true;
|
|
if (!wasOff) return;
|
|
const from = readAnonymousCreateFlowState();
|
|
if (Object.keys(from).length === 0) return;
|
|
// eslint-disable-next-line react-hooks/set-state-in-effect -- hydrate anonymous draft when guest persistence turns on
|
|
setState((prev) => (Object.keys(prev).length > 0 ? prev : { ...from }));
|
|
}, [enableAnonymousPersistence]);
|
|
|
|
useEffect(() => {
|
|
if (!enableAnonymousPersistence) return;
|
|
writeAnonymousCreateFlowState(state);
|
|
}, [state, enableAnonymousPersistence]);
|
|
|
|
/** Meaning/signals for core values: survives refresh for signed-in users; merged with anonymous draft when both exist. */
|
|
useEffect(() => {
|
|
writeCoreValueDetailsToLocalStorage(state.coreValueDetailsByChipId);
|
|
}, [state.coreValueDetailsByChipId]);
|
|
|
|
const markCreateFlowInteraction = useCallback(() => {
|
|
setInteractionTouched(true);
|
|
}, []);
|
|
|
|
const updateState = useCallback((updates: Partial<CreateFlowState>) => {
|
|
setState((prevState) => {
|
|
const merged: CreateFlowState = { ...prevState, ...updates };
|
|
if (updates.communityStructureChipSnapshots !== undefined) {
|
|
merged.communityStructureChipSnapshots = {
|
|
...(prevState.communityStructureChipSnapshots ?? {}),
|
|
...updates.communityStructureChipSnapshots,
|
|
};
|
|
}
|
|
if (updates.coreValueDetailsByChipId !== undefined) {
|
|
merged.coreValueDetailsByChipId = {
|
|
...(prevState.coreValueDetailsByChipId ?? {}),
|
|
...updates.coreValueDetailsByChipId,
|
|
};
|
|
}
|
|
return merged;
|
|
});
|
|
}, []);
|
|
|
|
const replaceState = useCallback((next: CreateFlowState) => {
|
|
setState(next);
|
|
}, []);
|
|
|
|
const clearState = useCallback(() => {
|
|
setState({});
|
|
setInteractionTouched(false);
|
|
clearAnonymousCreateFlowStorage();
|
|
clearCoreValueDetailsLocalStorage();
|
|
}, []);
|
|
|
|
const contextValue: CreateFlowContextValue = {
|
|
state,
|
|
currentStep,
|
|
updateState,
|
|
replaceState,
|
|
clearState,
|
|
interactionTouched,
|
|
markCreateFlowInteraction,
|
|
};
|
|
|
|
return (
|
|
<CreateFlowContext.Provider value={contextValue}>
|
|
{children}
|
|
</CreateFlowContext.Provider>
|
|
);
|
|
}
|
|
|
|
export function useCreateFlow(): CreateFlowContextValue {
|
|
const context = useContext(CreateFlowContext);
|
|
if (!context) {
|
|
throw new Error("useCreateFlow must be used within CreateFlowProvider");
|
|
}
|
|
return context;
|
|
}
|