@dnb/eufemia
Version:
DNB Eufemia Design System UI Library
197 lines (196 loc) • 7.83 kB
JavaScript
"use client";
var _BubbleValidation;
import React, { useCallback, useContext, useMemo, useReducer, useRef, useState } from 'react';
import useMountEffect from "../../../../shared/helpers/useMountEffect.js";
import pointer from "../../utils/json-pointer/index.js";
import { isZodSchema } from "../../utils/zod.js";
import { extractZodSubSchema } from "./extractZodSubSchema.js";
import { extendDeep } from "../../../../shared/component-helper.js";
import { isAsync } from "../../../../shared/helpers/isAsync.js";
import useDataValue from "../../hooks/useDataValue.js";
import { Context as DataContext, Provider } from "../../DataContext/index.js";
import SectionContext from "../Section/SectionContext.js";
import useReportError from "./useReportError.js";
import IsolationCommitButton from "./IsolationCommitButton.js";
import IsolationResetButton from "./IsolationResetButton.js";
import { clearedData } from "../../DataContext/Provider/index.js";
import { createDataReference } from "./IsolationDataReference.js";
import IsolatedContainer, { isolationError } from "./IsolatedContainer.js";
import IsolationContext from "./IsolationContext.js";
import { structuredClone } from "../../../../shared/helpers/structuredClone.js";
function IsolationProvider(props) {
var _IsolatedContainer;
const [dataReferenceFallback] = useState(() => {
if (!props?.dataReference) {
return createDataReference();
}
});
const {
children,
onPathChange,
onCommit: onCommitProp,
onClear: onClearProp,
transformOnCommit: transformOnCommitProp,
commitHandleRef,
bubbleValidation,
preventUncommittedChanges,
data,
defaultData,
dataReference = dataReferenceFallback,
resetDataAfterCommit
} = props;
const [, forceUpdate] = useReducer(() => ({}), {});
const internalDataRef = useRef();
const localDataRef = useRef({});
const dataContextRef = useRef(null);
const outerContext = useContext(DataContext);
const {
path: pathSection
} = useContext(SectionContext) || {};
const {
handlePathChange: handlePathChangeOuter,
data: dataOuter
} = outerContext || {};
const {
moveValueToPath
} = useDataValue();
const onPathChangeHandler = useCallback(async (path, value) => {
if (localDataRef.current === clearedData) {
localDataRef.current = {};
}
pointer.set(localDataRef.current, path, value);
if (pathSection) {
path = path.replace(pathSection, '');
}
return await onPathChange?.(path, value);
}, [onPathChange, pathSection]);
const onUpdateDataValueHandler = useCallback(async (path, value, {
preventUpdate = undefined
} = {}) => {
if (internalDataRef.current === clearedData) {
internalDataRef.current = {};
}
pointer.set(internalDataRef.current, path, value);
if (!preventUpdate) {
forceUpdate();
}
}, []);
const removeSectionPath = useCallback(data => {
return pathSection && pointer.has(data, pathSection) ? pointer.get(data, pathSection) : data;
}, [pathSection]);
const getMountedData = useCallback(data => {
const mounterData = {};
dataContextRef.current?.mountedFieldsRef.current.forEach((field, path) => {
if (field.isMounted && pointer.has(data, path)) {
pointer.set(mounterData, path, pointer.get(data, path));
}
});
return mounterData;
}, []);
useMountEffect(() => {
localDataRef.current = getMountedData(internalDataRef.current);
});
useMemo(() => {
if (localDataRef.current === clearedData) {
return;
}
let localData = data !== null && data !== void 0 ? data : defaultData;
if (localData && pathSection && !pointer.has(localDataRef.current, pathSection)) {
localData = moveValueToPath(pathSection, localData);
}
internalDataRef.current = Object.assign({}, localData || structuredClone(dataOuter) || {}, localDataRef.current);
}, [data, defaultData, pathSection, dataOuter, moveValueToPath]);
const onCommit = useCallback(async (data, additionalArgs) => {
var _props$path;
const mountedData = getMountedData(data);
const path = (_props$path = props.path) !== null && _props$path !== void 0 ? _props$path : '/';
const outerData = props.path && pointer.has(dataOuter, path) ? pointer.get(dataOuter, path) : dataOuter;
localDataRef.current = mountedData;
let isolatedData = structuredClone(mountedData);
if (typeof transformOnCommitProp === 'function') {
isolatedData = transformOnCommitProp(isolatedData, outerData);
}
let stop = false;
additionalArgs.preventCommit = () => stop = true;
const commitData = removeSectionPath(isolatedData);
const result = isAsync(onCommitProp) ? await onCommitProp?.(commitData, additionalArgs) : onCommitProp?.(commitData, additionalArgs);
if (stop) {
return;
}
await handlePathChangeOuter?.(path, Array.isArray(isolatedData) ? isolatedData : extendDeep({}, outerData, isolatedData));
return result;
}, [getMountedData, props.path, dataOuter, transformOnCommitProp, handlePathChangeOuter, onCommitProp, removeSectionPath]);
const setIsolatedData = useCallback(data => {
localDataRef.current = data;
internalDataRef.current = data;
}, []);
const onClear = useCallback(() => {
setIsolatedData(clearedData);
forceUpdate();
onClearProp?.();
}, [onClearProp, setIsolatedData]);
const providerProps = {
...props,
[defaultData ? 'defaultData' : 'data']: internalDataRef.current,
onUpdateDataValue: onUpdateDataValueHandler,
onPathChange: onPathChangeHandler,
onCommit,
onClear,
isolate: true,
schema: props?.schema || outerContext?.props?.schema,
ajvInstance: props?.ajvInstance || outerContext?.props?.ajvInstance
};
if (props?.path && props?.path !== '/' && !props?.schema && outerContext?.props?.schema) {
if (isZodSchema(outerContext.props.schema)) {
providerProps.schema = extractZodSubSchema(outerContext.props.schema, props.path);
} else {
providerProps.schema = {
$defs: {
root: outerContext.props.schema
},
$ref: `#/$defs/root${props.path.split('/').join('/properties/')}`
};
}
}
return React.createElement(Provider, providerProps, React.createElement(IsolationContext.Provider, {
value: {
preventUncommittedChanges,
dataReference,
resetDataAfterCommit,
outerContext,
setIsolatedData
}
}, React.createElement(DataContext.Consumer, null, dataContext => {
dataContextRef.current = dataContext;
if (commitHandleRef) {
commitHandleRef.current = dataContext?.handleSubmit;
}
return _IsolatedContainer || (_IsolatedContainer = React.createElement(IsolatedContainer, null, children, " "));
}), bubbleValidation && (_BubbleValidation || (_BubbleValidation = React.createElement(BubbleValidation, null)))));
}
function BubbleValidation() {
const innerContext = useContext(DataContext);
const {
outerContext
} = useContext(IsolationContext);
const {
setShowAllErrors
} = innerContext;
const setShowAllErrorsNested = useCallback(showAllErrors => {
setShowAllErrors?.(showAllErrors);
}, [setShowAllErrors]);
const {
addSetShowAllErrorsRef
} = outerContext || {};
if (!addSetShowAllErrorsRef?.current?.includes(setShowAllErrorsNested)) {
addSetShowAllErrorsRef?.current.push(setShowAllErrorsNested);
}
useReportError(innerContext.hasErrors() ? isolationError : undefined, outerContext, 'isolation');
return null;
}
IsolationProvider.CommitButton = IsolationCommitButton;
IsolationProvider.ResetButton = IsolationResetButton;
IsolationProvider.createDataReference = createDataReference;
IsolationProvider._supportsSpacingProps = undefined;
export default IsolationProvider;
//# sourceMappingURL=Isolation.js.map