mirror of
https://github.com/infiniflow/ragflow.git
synced 2025-12-08 20:42:30 +08:00
### What problem does this PR solve? Feat: Add FileUploadDialog #3221 ### Type of change - [x] New Feature (non-breaking change which adds functionality)
68 lines
1.8 KiB
TypeScript
68 lines
1.8 KiB
TypeScript
import * as React from 'react';
|
|
|
|
import { useCallbackRef } from '@/hooks/use-callback-ref';
|
|
|
|
/**
|
|
* @see https://github.com/radix-ui/primitives/blob/main/packages/react/use-controllable-state/src/useControllableState.tsx
|
|
*/
|
|
|
|
type UseControllableStateParams<T> = {
|
|
prop?: T | undefined;
|
|
defaultProp?: T | undefined;
|
|
onChange?: (state: T) => void;
|
|
};
|
|
|
|
type SetStateFn<T> = (prevState?: T) => T;
|
|
|
|
function useUncontrolledState<T>({
|
|
defaultProp,
|
|
onChange,
|
|
}: Omit<UseControllableStateParams<T>, 'prop'>) {
|
|
const uncontrolledState = React.useState<T | undefined>(defaultProp);
|
|
const [value] = uncontrolledState;
|
|
const prevValueRef = React.useRef(value);
|
|
const handleChange = useCallbackRef(onChange);
|
|
|
|
React.useEffect(() => {
|
|
if (prevValueRef.current !== value) {
|
|
handleChange(value as T);
|
|
prevValueRef.current = value;
|
|
}
|
|
}, [value, prevValueRef, handleChange]);
|
|
|
|
return uncontrolledState;
|
|
}
|
|
|
|
function useControllableState<T>({
|
|
prop,
|
|
defaultProp,
|
|
onChange = () => {},
|
|
}: UseControllableStateParams<T>) {
|
|
const [uncontrolledProp, setUncontrolledProp] = useUncontrolledState({
|
|
defaultProp,
|
|
onChange,
|
|
});
|
|
const isControlled = prop !== undefined;
|
|
const value = isControlled ? prop : uncontrolledProp;
|
|
const handleChange = useCallbackRef(onChange);
|
|
|
|
const setValue: React.Dispatch<React.SetStateAction<T | undefined>> =
|
|
React.useCallback(
|
|
(nextValue) => {
|
|
if (isControlled) {
|
|
const setter = nextValue as SetStateFn<T>;
|
|
const value =
|
|
typeof nextValue === 'function' ? setter(prop) : nextValue;
|
|
if (value !== prop) handleChange(value as T);
|
|
} else {
|
|
setUncontrolledProp(nextValue);
|
|
}
|
|
},
|
|
[isControlled, prop, setUncontrolledProp, handleChange],
|
|
);
|
|
|
|
return [value, setValue] as const;
|
|
}
|
|
|
|
export { useControllableState };
|