diff --git a/src/components/form/Form.tsx b/src/components/form/Form.tsx
index 906debbc9b..63394a8fa5 100644
--- a/src/components/form/Form.tsx
+++ b/src/components/form/Form.tsx
@@ -91,10 +91,7 @@ export function FormPage({ currentPageId }: { currentPageId: string | undefined
return (
<>
-
+
>
);
}
diff --git a/src/components/presentation/Presentation.tsx b/src/components/presentation/Presentation.tsx
index 6bad9e14b5..6727ba2a20 100644
--- a/src/components/presentation/Presentation.tsx
+++ b/src/components/presentation/Presentation.tsx
@@ -11,6 +11,7 @@ import { Header } from 'src/components/presentation/Header';
import { NavBar } from 'src/components/presentation/NavBar';
import classes from 'src/components/presentation/Presentation.module.css';
import { Progress } from 'src/components/presentation/Progress';
+import { createContext } from 'src/core/contexts/context';
import { RenderStart } from 'src/core/ui/RenderStart';
import { Footer } from 'src/features/footer/Footer';
import { useUiConfigContext } from 'src/features/form/layout/UiConfigContext';
@@ -45,38 +46,40 @@ export const PresentationComponent = ({ header, type, children, renderNavBar = t
return (
-
-
-
- {isProcessStepsArchived && instanceStatus?.substatus && (
- }
- description={}
- />
- )}
- {renderNavBar && }
-
-
-
-
+
+
+
+
+ {isProcessStepsArchived && instanceStatus?.substatus && (
+ }
+ description={}
+ />
+ )}
+ {renderNavBar && }
+
+
+
+
+
);
};
@@ -98,3 +101,10 @@ function ProgressBar({ type }: { type: ProcessTaskType | PresentationType }) {
);
}
+
+const { Provider: PresentationProvider, useHasProvider } = createContext({
+ name: 'Presentation',
+ required: true,
+});
+
+export const useHasPresentation = () => useHasProvider();
diff --git a/src/components/wrappers/ProcessWrapper.tsx b/src/components/wrappers/ProcessWrapper.tsx
index fe81c0fe3e..a18da97c8e 100644
--- a/src/components/wrappers/ProcessWrapper.tsx
+++ b/src/components/wrappers/ProcessWrapper.tsx
@@ -5,7 +5,7 @@ import { Route, Routes } from 'react-router-dom';
import Grid from '@material-ui/core/Grid';
import { Button } from 'src/app-components/button/Button';
-import { Form, FormFirstPage } from 'src/components/form/Form';
+import { Form } from 'src/components/form/Form';
import { PresentationComponent } from 'src/components/presentation/Presentation';
import classes from 'src/components/wrappers/ProcessWrapper.module.css';
import { Loader } from 'src/core/loading/Loader';
@@ -187,7 +187,7 @@ export const ProcessWrapper = () => {
}
/>
@@ -196,10 +196,6 @@ export const ProcessWrapper = () => {
}
/>
- }
- />
);
diff --git a/src/core/loading/Loader.tsx b/src/core/loading/Loader.tsx
index 022e28718a..7510c8b624 100644
--- a/src/core/loading/Loader.tsx
+++ b/src/core/loading/Loader.tsx
@@ -2,7 +2,7 @@ import React from 'react';
import { AltinnContentIconFormData } from 'src/components/atoms/AltinnContentIconFormData';
import { AltinnContentLoader } from 'src/components/molecules/AltinnContentLoader';
-import { PresentationComponent } from 'src/components/presentation/Presentation';
+import { PresentationComponent, useHasPresentation } from 'src/components/presentation/Presentation';
import { useTaskStore } from 'src/core/contexts/taskStoreContext';
import { LoadingProvider } from 'src/core/loading/LoadingContext';
import { Lang } from 'src/features/language/Lang';
@@ -11,12 +11,12 @@ import { ProcessTaskType } from 'src/types';
interface LoaderProps {
reason: string; // The reason is used by developers to identify the reason for the loader
details?: string;
- renderPresentation?: boolean;
}
-export const Loader = ({ renderPresentation = true, ...rest }: LoaderProps) => {
+export const Loader = (props: LoaderProps) => {
const overriddenDataModelUuid = useTaskStore((state) => state.overriddenDataModelUuid);
const overriddenTaskId = useTaskStore((state) => state.overriddenTaskId);
+ const hasPresentation = useHasPresentation();
if (overriddenDataModelUuid) {
return null;
@@ -25,23 +25,23 @@ export const Loader = ({ renderPresentation = true, ...rest }: LoaderProps) => {
if (overriddenTaskId) {
return null;
}
- if (renderPresentation) {
+ if (!hasPresentation) {
return (
-
+
}
type={ProcessTaskType.Unknown}
renderNavBar={false}
>
-
+
);
}
return (
-
-
+
+
);
};
diff --git a/src/features/formData/FormDataWrite.tsx b/src/features/formData/FormDataWrite.tsx
index 73e9cd368d..8d354d0e39 100644
--- a/src/features/formData/FormDataWrite.tsx
+++ b/src/features/formData/FormDataWrite.tsx
@@ -971,11 +971,4 @@ export const FD = {
return useCallback((dataElementId: string) => map[dataElementId], [map]);
},
-
- /**
- * This lets you set to a function that will be called as soon as the saving operation finishes.
- * Beware that this is not a subscription service, so you can easily overwrite an existing callback here. This
- * is only meant to be used in NodesContext.
- */
- useSetOnSaveFinished: () => useSelector((s) => s.setOnSaveFinished),
};
diff --git a/src/features/formData/FormDataWriteStateMachine.tsx b/src/features/formData/FormDataWriteStateMachine.tsx
index e1feb5c69e..258fba9e1f 100644
--- a/src/features/formData/FormDataWriteStateMachine.tsx
+++ b/src/features/formData/FormDataWriteStateMachine.tsx
@@ -80,11 +80,6 @@ type FormDataState = {
// This contains the validation issues we receive from the server last time we saved the data model.
validationIssues: BackendValidationIssueGroups | undefined;
- // This may contain a callback function that will be called whenever the save finishes.
- // Should only be set from NodesContext.
- onSaveFinished: ((result: FDSaveFinished) => void) | undefined;
- setOnSaveFinished: (callback: (result: FDSaveFinished) => void) => void;
-
// This is used to track which component is currently blocking the auto-saving feature. If this is set to a string
// value, auto-saving will be disabled, even if the autoSaving flag is set to true. This is useful when you want
// to temporarily disable auto-saving, for example when clicking a CustomButton and waiting for the server to
@@ -238,7 +233,6 @@ function makeActions(
const { validationIssues, savedData, newDataModels, instance } = toProcess;
state.manualSaveRequested = false;
state.validationIssues = validationIssues;
- state.onSaveFinished?.(toProcess);
if (instance && changeInstance) {
changeInstance(() => instance);
@@ -548,11 +542,6 @@ export const createFormDataWriteStore = (
debounceTimeout: DEFAULT_DEBOUNCE_TIMEOUT,
manualSaveRequested: false,
validationIssues: undefined,
- onSaveFinished: undefined,
- setOnSaveFinished: (callback) =>
- set((state) => {
- state.onSaveFinished = callback;
- }),
...actions,
};
}),
diff --git a/src/features/routing/AppRoutingContext.tsx b/src/features/routing/AppRoutingContext.tsx
index 8f54cd2524..47fe582ec4 100644
--- a/src/features/routing/AppRoutingContext.tsx
+++ b/src/features/routing/AppRoutingContext.tsx
@@ -1,6 +1,7 @@
import React, { useEffect } from 'react';
import { matchPath, useLocation, useNavigate as useNativeNavigate } from 'react-router-dom';
import type { MutableRefObject, PropsWithChildren } from 'react';
+import type { NavigateOptions } from 'react-router-dom';
import { createStore } from 'zustand';
@@ -30,17 +31,18 @@ interface Context {
updateHash: (hash: string) => void;
effectCallback: NavigationEffectCb | null;
setEffectCallback: (cb: NavigationEffectCb | null) => void;
- navigateRef: MutableRefObject>;
+ navigateRef: MutableRefObject;
}
+export type SimpleNavigate = (target: string, options?: NavigateOptions) => void;
+
function newStore({ initialLocation }: { initialLocation: string | undefined }) {
return createStore((set) => ({
hash: initialLocation ? initialLocation : `${window.location.hash}`,
updateHash: (hash: string) => set({ hash }),
effectCallback: null,
setEffectCallback: (effectCallback: NavigationEffectCb) => set({ effectCallback }),
- // eslint-disable-next-line @typescript-eslint/no-explicit-any
- navigateRef: { current: undefined as any },
+ navigateRef: { current: undefined },
}));
}
@@ -139,11 +141,13 @@ export const useIsSubformPage = () =>
useSelector((s) => {
const path = getPath(s.hash);
const matches = matchers.map((matcher) => matchPath(matcher, path));
- return !!paramFrom(matches, 'mainPageKey');
+ const mainPageKey = paramFrom(matches, 'mainPageKey');
+ const subformPageKey = paramFrom(matches, 'pageKey');
+ return !!(mainPageKey && subformPageKey);
});
// Use this instead of the native one to avoid re-rendering whenever the route changes
-export const useNavigate = () => useSelector((ctx) => ctx.navigateRef).current;
+export const useNavigate = () => useStaticSelector((ctx) => ctx.navigateRef).current!;
const matchers: string[] = [
'/instance/:partyId/:instanceGuid',
@@ -198,8 +202,20 @@ function UpdateHash() {
}
function UpdateNavigate() {
- const navigateRef = useSelector((ctx) => ctx.navigateRef);
- navigateRef.current = useNativeNavigate();
+ const store = useStore();
+ const navigateRef = useStaticSelector((ctx) => ctx.navigateRef);
+ const nativeNavigate = useNativeNavigate();
+
+ navigateRef.current = (target, options) => {
+ if (target && !target.startsWith('/')) {
+ // Used for relative navigation, e.g. navigating to a subform page
+ const currentPath = getPath(store.getState().hash).replace(/\/$/, '');
+ const newTarget = `${currentPath}/${target}`;
+ nativeNavigate(newTarget, options);
+ return;
+ }
+ nativeNavigate(target, options);
+ };
return null;
}
diff --git a/src/features/validation/StoreValidationsInNode.tsx b/src/features/validation/StoreValidationsInNode.tsx
index 9d2a445f73..dd6dcffa0d 100644
--- a/src/features/validation/StoreValidationsInNode.tsx
+++ b/src/features/validation/StoreValidationsInNode.tsx
@@ -10,7 +10,7 @@ import { GeneratorCondition, StageFormValidation } from 'src/utils/layout/genera
import { NodesInternal } from 'src/utils/layout/NodesContext';
import type { AnyValidation, AttachmentValidation } from 'src/features/validation/index';
import type { CompCategory } from 'src/layout/common';
-import type { TypesFromCategory } from 'src/layout/layout';
+import type { CompIntermediate, TypesFromCategory } from 'src/layout/layout';
import type { LayoutNode } from 'src/utils/layout/LayoutNode';
export function StoreValidationsInNode() {
@@ -29,9 +29,9 @@ type Node = LayoutNode !deepEqual(data.validations, validations));
@@ -56,15 +56,6 @@ function StoreValidationsInNodeWorker() {
visibilityToSet !== undefined,
);
- const shouldSetProcessedLast = NodesInternal.useNodeData(
- node,
- (data) => data.validationsProcessedLast !== processedLast,
- );
- NodesStateQueue.useSetNodeProp(
- { node, prop: 'validationsProcessedLast', value: processedLast, force: true },
- shouldSetProcessedLast,
- );
-
return null;
}
@@ -91,3 +82,7 @@ function useUpdatedValidations(validations: AnyValidation[], node: Node) {
return copy;
});
}
+
+export function shouldValidateNode(item: CompIntermediate): boolean {
+ return !('renderAsSummary' in item && item.renderAsSummary);
+}
diff --git a/src/features/validation/ValidationPlugin.tsx b/src/features/validation/ValidationPlugin.tsx
index d57d30dc70..088ed45c04 100644
--- a/src/features/validation/ValidationPlugin.tsx
+++ b/src/features/validation/ValidationPlugin.tsx
@@ -1,22 +1,16 @@
import { CG } from 'src/codegen/CG';
import { NodeDefPlugin } from 'src/utils/layout/plugins/NodeDefPlugin';
import type { ComponentConfig } from 'src/codegen/ComponentConfig';
-import type { ComponentValidation, ValidationsProcessedLast } from 'src/features/validation/index';
+import type { ComponentValidation } from 'src/features/validation/index';
import type { CompCategory } from 'src/layout/common';
import type { TypesFromCategory } from 'src/layout/layout';
-import type { NodesContext } from 'src/utils/layout/NodesContext';
-import type {
- DefPluginExtraState,
- DefPluginState,
- DefPluginStateFactoryProps,
-} from 'src/utils/layout/plugins/NodeDefPlugin';
+import type { DefPluginExtraState, DefPluginStateFactoryProps } from 'src/utils/layout/plugins/NodeDefPlugin';
interface Config {
componentType: TypesFromCategory;
extraState: {
validations: ComponentValidation[];
validationVisibility: number;
- validationsProcessedLast: ValidationsProcessedLast;
};
}
@@ -49,7 +43,6 @@ export class ValidationPlugin extends NodeDefPlugin {
return {
validations: [],
validationVisibility: 0,
- validationsProcessedLast: { initial: undefined, incremental: undefined },
};
}
@@ -61,11 +54,4 @@ export class ValidationPlugin extends NodeDefPlugin {
return `<${StoreValidationsInNode} />`;
}
-
- stateIsReady(state: DefPluginState, fullState: NodesContext): boolean {
- return (
- state.validationsProcessedLast.initial === fullState.validationsProcessedLast.initial &&
- state.validationsProcessedLast.incremental === fullState.validationsProcessedLast.incremental
- );
- }
}
diff --git a/src/features/validation/backendValidation/BackendValidation.tsx b/src/features/validation/backendValidation/BackendValidation.tsx
index 3edaa35f85..39dda82b88 100644
--- a/src/features/validation/backendValidation/BackendValidation.tsx
+++ b/src/features/validation/backendValidation/BackendValidation.tsx
@@ -13,7 +13,6 @@ import {
useShouldValidateInitial,
} from 'src/features/validation/backendValidation/backendValidationUtils';
import { Validation } from 'src/features/validation/validationContext';
-import { NodesStore } from 'src/utils/layout/NodesContext';
const emptyObject = {};
const emptyArray = [];
@@ -91,15 +90,3 @@ export function BackendValidation({ dataTypes }: { dataTypes: string[] }) {
return null;
}
-
-export function MaintainInitialValidationsInNodesContext() {
- const enabled = useShouldValidateInitial();
- const { data: initialValidations } = useBackendValidationQuery(enabled);
- const setInitialValidations = NodesStore.useSelector((state) => state.setLatestInitialValidations);
-
- useEffect(() => {
- setInitialValidations(initialValidations);
- }, [initialValidations, setInitialValidations]);
-
- return null;
-}
diff --git a/src/features/validation/nodeValidation/useNodeValidation.ts b/src/features/validation/nodeValidation/useNodeValidation.ts
index 910e2401a1..cc902348b4 100644
--- a/src/features/validation/nodeValidation/useNodeValidation.ts
+++ b/src/features/validation/nodeValidation/useNodeValidation.ts
@@ -1,119 +1,59 @@
-import { useMemo } from 'react';
-
-import { useApplicationMetadata } from 'src/features/applicationMetadata/ApplicationMetadataProvider';
-import { useAttachmentsSelector } from 'src/features/attachments/hooks';
-import { useLayoutSets } from 'src/features/form/layoutSets/LayoutSetsProvider';
-import { FD } from 'src/features/formData/FormDataWrite';
-import { useLaxDataElementsSelector } from 'src/features/instance/InstanceContext';
-import { useCurrentLanguage } from 'src/features/language/LanguageProvider';
import { Validation } from 'src/features/validation/validationContext';
import { implementsValidateComponent, implementsValidateEmptyField } from 'src/layout';
+import { GeneratorInternal } from 'src/utils/layout/generator/GeneratorContext';
import { GeneratorData } from 'src/utils/layout/generator/GeneratorDataSources';
-import { NodesInternal } from 'src/utils/layout/NodesContext';
-import type {
- AnyValidation,
- BaseValidation,
- ValidationDataSources,
- ValidationsProcessedLast,
-} from 'src/features/validation';
+import type { AnyValidation, BaseValidation } from 'src/features/validation';
import type { CompDef, ValidationFilter } from 'src/layout';
import type { IDataModelReference } from 'src/layout/common.generated';
import type { LayoutNode } from 'src/utils/layout/LayoutNode';
import type { NodeDataSelector } from 'src/utils/layout/NodesContext';
+const emptyArray: AnyValidation[] = [];
+
/**
* Runs validations defined in the component classes. This runs from the node generator, and will collect all
* validations for a node and return them.
*/
-export function useNodeValidation(
- node: LayoutNode,
- shouldValidate: boolean,
-): { validations: AnyValidation[]; processedLast: ValidationsProcessedLast } {
- const dataModelSelector = Validation.useDataModelSelector();
- const validationDataSources = GeneratorData.useValidationDataSources();
- const nodeDataSelector = NodesInternal.useNodeDataSelector();
-
+export function useNodeValidation(node: LayoutNode, shouldValidate: boolean): AnyValidation[] {
+ const registry = GeneratorInternal.useRegistry();
+ const dataSources = GeneratorData.useValidationDataSources();
const getDataElementIdForDataType = GeneratorData.useGetDataElementIdForDataType();
- const processedLast = GeneratorData.useValidationsProcessedLast();
+ const dataModelBindings = GeneratorInternal.useIntermediateItem()?.dataModelBindings;
+ const bindings = Object.entries((dataModelBindings ?? {}) as Record);
- return {
- processedLast,
- validations: useMemo(() => {
- const validations: AnyValidation[] = [];
- if (!shouldValidate) {
- return validations;
- }
+ return Validation.useFullState((state) => {
+ if (!shouldValidate) {
+ return emptyArray;
+ }
- if (implementsValidateEmptyField(node.def)) {
- // eslint-disable-next-line @typescript-eslint/no-explicit-any
- validations.push(...node.def.runEmptyFieldValidation(node as any, validationDataSources));
- }
+ const validations: AnyValidation[] = [];
+ if (implementsValidateEmptyField(node.def)) {
+ // eslint-disable-next-line @typescript-eslint/no-explicit-any
+ validations.push(...node.def.runEmptyFieldValidation(node as any, dataSources));
+ }
- if (implementsValidateComponent(node.def)) {
- // eslint-disable-next-line @typescript-eslint/no-explicit-any
- validations.push(...node.def.runComponentValidation(node as any, validationDataSources));
- }
+ if (implementsValidateComponent(node.def)) {
+ // eslint-disable-next-line @typescript-eslint/no-explicit-any
+ validations.push(...node.def.runComponentValidation(node as any, dataSources));
+ }
- const dataModelBindings = validationDataSources.nodeDataSelector(
- (picker) => picker(node)?.layout.dataModelBindings,
- [node],
- );
- for (const [bindingKey, { dataType, field }] of Object.entries(
- (dataModelBindings ?? {}) as Record,
- )) {
- const dataElementId = getDataElementIdForDataType(dataType) ?? dataType; // stateless does not have dataElementId
- const fieldValidations = dataModelSelector(
- (dataModels) => dataModels[dataElementId]?.[field],
- [dataType, field],
- );
- if (fieldValidations) {
- validations.push(...fieldValidations.map((v) => ({ ...v, bindingKey })));
- }
+ for (const [bindingKey, { dataType, field }] of bindings) {
+ const dataElementId = getDataElementIdForDataType(dataType) ?? dataType; // stateless does not have dataElementId
+ const fieldValidations = state.state.dataModels[dataElementId]?.[field];
+ if (fieldValidations) {
+ validations.push(...fieldValidations.map((v) => ({ ...v, bindingKey })));
}
+ }
- return filter(validations, node, nodeDataSelector);
- }, [shouldValidate, node, validationDataSources, nodeDataSelector, getDataElementIdForDataType, dataModelSelector]),
- };
-}
+ const result = filter(validations, node, dataSources.nodeDataSelector);
+ registry.current.validationsProcessed[node.id] = state.processedLast;
-/**
- * Hook providing validation data sources
- */
-function useValidationDataSources(): ValidationDataSources {
- const formDataSelector = FD.useDebouncedSelector();
- const invalidDataSelector = FD.useInvalidDebouncedSelector();
- const attachmentsSelector = useAttachmentsSelector();
- const currentLanguage = useCurrentLanguage();
- const nodeSelector = NodesInternal.useNodeDataSelector();
- const applicationMetadata = useApplicationMetadata();
- const dataElementsSelector = useLaxDataElementsSelector();
- const layoutSets = useLayoutSets();
- const dataElementHasErrorsSelector = Validation.useDataElementHasErrorsSelector();
+ if (result.length === 0) {
+ return emptyArray;
+ }
- return useMemo(
- () => ({
- formDataSelector,
- invalidDataSelector,
- attachmentsSelector,
- currentLanguage,
- nodeDataSelector: nodeSelector,
- applicationMetadata,
- dataElementsSelector,
- layoutSets,
- dataElementHasErrorsSelector,
- }),
- [
- formDataSelector,
- invalidDataSelector,
- attachmentsSelector,
- currentLanguage,
- nodeSelector,
- applicationMetadata,
- dataElementsSelector,
- layoutSets,
- dataElementHasErrorsSelector,
- ],
- );
+ return result;
+ });
}
/**
diff --git a/src/features/validation/nodeValidation/waitForNodesToValidate.ts b/src/features/validation/nodeValidation/waitForNodesToValidate.ts
new file mode 100644
index 0000000000..09229564b7
--- /dev/null
+++ b/src/features/validation/nodeValidation/waitForNodesToValidate.ts
@@ -0,0 +1,54 @@
+import { useCallback } from 'react';
+
+import { shouldValidateNode } from 'src/features/validation/StoreValidationsInNode';
+import { GeneratorInternal } from 'src/utils/layout/generator/GeneratorContext';
+import { NodesStore } from 'src/utils/layout/NodesContext';
+import type { ValidationsProcessedLast } from 'src/features/validation';
+
+export function useWaitForNodesToValidate() {
+ const registry = GeneratorInternal.useRegistry();
+ const nodesStore = NodesStore.useStore();
+
+ return useCallback(
+ async (processedLast: ValidationsProcessedLast): Promise => {
+ let callbackId: ReturnType | undefined;
+ const request = window.requestIdleCallback || window.requestAnimationFrame;
+ const cancel = window.cancelIdleCallback || window.cancelAnimationFrame;
+
+ function check(): boolean {
+ const allNodeData = nodesStore.getState().nodeData;
+ const nodeIds = Object.keys(allNodeData);
+ for (const nodeId of nodeIds) {
+ const nodeData = allNodeData[nodeId];
+ if (!nodeData || !('validations' in nodeData) || !shouldValidateNode(nodeData.layout)) {
+ // Node does not support validation
+ continue;
+ }
+
+ const lastValidations = registry.current.validationsProcessed[nodeId];
+ const initialIsLatest = lastValidations?.initial === processedLast.initial;
+ const incrementalIsLatest = lastValidations?.incremental === processedLast.incremental;
+ if (!(lastValidations && initialIsLatest && incrementalIsLatest)) {
+ return false;
+ }
+ }
+
+ return true;
+ }
+
+ return new Promise((resolve) => {
+ function checkAndResolve() {
+ if (check()) {
+ resolve();
+ callbackId && cancel(callbackId);
+ } else {
+ callbackId = request(checkAndResolve);
+ }
+ }
+
+ checkAndResolve();
+ });
+ },
+ [nodesStore, registry],
+ );
+}
diff --git a/src/features/validation/validationContext.tsx b/src/features/validation/validationContext.tsx
index 0d52480958..c4dc098623 100644
--- a/src/features/validation/validationContext.tsx
+++ b/src/features/validation/validationContext.tsx
@@ -31,6 +31,7 @@ import {
useShouldValidateInitial,
} from 'src/features/validation/backendValidation/backendValidationUtils';
import { InvalidDataValidation } from 'src/features/validation/invalidDataValidation/InvalidDataValidation';
+import { useWaitForNodesToValidate } from 'src/features/validation/nodeValidation/waitForNodesToValidate';
import { SchemaValidation } from 'src/features/validation/schemaValidation/SchemaValidation';
import { hasValidationErrors, mergeFieldValidations, selectValidations } from 'src/features/validation/utils';
import { useAsRef } from 'src/hooks/useAsRef';
@@ -146,6 +147,7 @@ function initialCreateStore() {
const {
Provider,
useSelector,
+ useMemoSelector,
useLaxShallowSelector,
useSelectorAsRef,
useStore,
@@ -188,6 +190,7 @@ function useWaitForValidation(): WaitForValidation {
const hasWritableDataTypes = !!DataModels.useWritableDataTypes()?.length;
const enabled = useShouldValidateInitial();
const getCachedInitialValidations = useGetCachedInitialValidations();
+ const waitForNodesToValidate = useWaitForNodesToValidate();
return useCallback(
async (forceSave = true) => {
@@ -201,21 +204,22 @@ function useWaitForValidation(): WaitForValidation {
await waitForNodesReady();
const validationsFromSave = await waitForSave(forceSave);
// If validationsFromSave is not defined, we check if initial validations are done processing
- const lastInitialValidations = await waitForState((state, setReturnValue) => {
+ await waitForState(async (state) => {
const { isFetching, cachedInitialValidations } = getCachedInitialValidations();
- if (
+ const validationsReady =
state.processedLast.incremental === validationsFromSave &&
state.processedLast.initial === cachedInitialValidations &&
- !isFetching
- ) {
- setReturnValue(cachedInitialValidations);
+ !isFetching;
+
+ if (validationsReady) {
+ await waitForNodesToValidate(state.processedLast);
return true;
}
return false;
});
- await waitForNodesReady({ initial: lastInitialValidations, incremental: validationsFromSave });
+ await waitForNodesReady();
},
[
enabled,
@@ -223,6 +227,7 @@ function useWaitForValidation(): WaitForValidation {
hasWritableDataTypes,
waitForAttachments,
waitForNodesReady,
+ waitForNodesToValidate,
waitForSave,
waitForState,
],
@@ -334,8 +339,6 @@ export type ValidationDataModelSelector = ReturnType;
export const Validation = {
- useFullStateRef: () => useSelectorAsRef((state) => state.state),
-
// Selectors. These are memoized, so they won't cause a re-render unless the selected fields change.
useSelector: () => useDS((state) => state),
useDataModelSelector: () => useDS((state) => state.state.dataModels),
@@ -377,8 +380,12 @@ export const Validation = {
useUpdateDataModelValidations: () => useSelector((state) => state.updateDataModelValidations),
useUpdateBackendValidations: () => useSelector((state) => state.updateBackendValidations),
- useProcessedLast: () => useSelector((state) => state.processedLast),
- useProcessedLastRef: () => useSelectorAsRef((state) => state.processedLast),
+ useFullState: (selector: (state: ValidationContext & Internals) => U): U =>
+ useMemoSelector((state) => selector(state)),
+ useGetProcessedLast: () => {
+ const store = useStore();
+ return useCallback(() => store.getState().processedLast, [store]);
+ },
useRef: () => useSelectorAsRef((state) => state),
useLaxRef: () => useLaxSelectorAsRef((state) => state),
diff --git a/src/hooks/useWaitForState.ts b/src/hooks/useWaitForState.ts
index 349d860492..d6a6d8b819 100644
--- a/src/hooks/useWaitForState.ts
+++ b/src/hooks/useWaitForState.ts
@@ -6,9 +6,9 @@ import type { StoreApi } from 'zustand';
import { ContextNotProvided } from 'src/core/contexts/context';
export type WaitForState = (callback: Callback) => Promise;
-type Callback = (state: T, setReturnValue: (val: RetVal) => void) => boolean;
+type Callback = (state: T, setReturnValue: (val: RetVal) => void) => boolean | Promise;
// eslint-disable-next-line @typescript-eslint/no-explicit-any
-type Subscriber = (state: any) => boolean;
+type Subscriber = (state: any) => boolean | Promise;
// If ContextNotProvided is a valid state, it will possibly be provided as a direct
// input (not wrapped in a ref or store)
@@ -27,8 +27,13 @@ export function useWaitForState(state: ValidInputs[0]): WaitForSta
// Call subscribers on every re-render/state change if state is a ref
if (isRef(state)) {
for (const subscriber of subscribersRef.current) {
- if (subscriber(state.current)) {
+ const result = subscriber(state.current);
+ if (result === true) {
subscribersRef.current.delete(subscriber);
+ } else if (result instanceof Promise) {
+ result.then((res) => {
+ res && subscribersRef.current.delete(subscriber);
+ });
}
}
}
@@ -36,9 +41,9 @@ export function useWaitForState(state: ValidInputs[0]): WaitForSta
useEffect(() => {
// Call subscribers on every state change if state is a zustand store
if (isStore(state)) {
- return state.subscribe((state) => {
+ return state.subscribe(async (state) => {
for (const subscriber of subscribersRef.current) {
- if (subscriber(state)) {
+ if (await subscriber(state)) {
subscribersRef.current.delete(subscriber);
}
}
@@ -56,8 +61,8 @@ export function useWaitForState(state: ValidInputs[0]): WaitForSta
}
if (state === ContextNotProvided) {
- subscribersRef.current.add((state) => {
- if (callback(state, setReturnValue)) {
+ subscribersRef.current.add(async (state) => {
+ if (await callback(state, setReturnValue)) {
resolve(returnValue as RetVal);
return true;
}
@@ -69,13 +74,20 @@ export function useWaitForState(state: ValidInputs[0]): WaitForSta
const currentState = isRef(state) ? state.current : state.getState();
// If state is already correct, resolve immediately
- if (callback(currentState, setReturnValue)) {
+ const result = callback(currentState, setReturnValue);
+
+ if (result === true) {
resolve(returnValue as RetVal);
return;
}
+ if (result instanceof Promise) {
+ result.then((res) => {
+ res && resolve(returnValue as RetVal);
+ });
+ }
- subscribersRef.current.add((state) => {
- if (callback(state, setReturnValue)) {
+ subscribersRef.current.add(async (state) => {
+ if (await callback(state, setReturnValue)) {
resolve(returnValue as RetVal);
return true;
}
diff --git a/src/layout/Subform/SubformComponent.tsx b/src/layout/Subform/SubformComponent.tsx
index 5db47207f5..48de463d6d 100644
--- a/src/layout/Subform/SubformComponent.tsx
+++ b/src/layout/Subform/SubformComponent.tsx
@@ -1,5 +1,4 @@
import React, { useState } from 'react';
-import { useNavigate } from 'react-router-dom';
import { Spinner, Table } from '@digdir/designsystemet-react';
import { Grid } from '@material-ui/core';
@@ -14,7 +13,7 @@ import { useFormDataQuery } from 'src/features/formData/useFormDataQuery';
import { useStrictDataElements, useStrictInstanceId } from 'src/features/instance/InstanceContext';
import { Lang } from 'src/features/language/Lang';
import { useLanguage } from 'src/features/language/useLanguage';
-import { useIsSubformPage } from 'src/features/routing/AppRoutingContext';
+import { useIsSubformPage, useNavigate } from 'src/features/routing/AppRoutingContext';
import { useAddEntryMutation, useDeleteEntryMutation } from 'src/features/subformData/useSubformMutations';
import { isSubformValidation } from 'src/features/validation';
import { useComponentValidationsForNode } from 'src/features/validation/selectors/componentValidationsForNode';
diff --git a/src/layout/Subform/SubformWrapper.tsx b/src/layout/Subform/SubformWrapper.tsx
index 57086eb08b..7a773cecd4 100644
--- a/src/layout/Subform/SubformWrapper.tsx
+++ b/src/layout/Subform/SubformWrapper.tsx
@@ -1,11 +1,11 @@
import React, { useEffect } from 'react';
-import { Form, FormFirstPage } from 'src/components/form/Form';
+import { Form } from 'src/components/form/Form';
import { useTaskStore } from 'src/core/contexts/taskStoreContext';
import { Loader } from 'src/core/loading/Loader';
import { FormProvider } from 'src/features/form/FormContext';
import { useDataTypeFromLayoutSet } from 'src/features/form/layout/LayoutsContext';
-import { useIsCurrentView, useNavigationParam } from 'src/features/routing/AppRoutingContext';
+import { useNavigationParam } from 'src/features/routing/AppRoutingContext';
import { useNavigatePage } from 'src/hooks/useNavigatePage';
import { useNodeItem } from 'src/utils/layout/useNodeItem';
import type { LayoutNode } from 'src/utils/layout/LayoutNode';
@@ -19,21 +19,10 @@ export function SubformWrapper({ node }: { node: LayoutNode<'Subform'> }) {
return (
-
+
);
}
-
-function SubformForm() {
- const hasFormPage = !useIsCurrentView(undefined);
-
- if (hasFormPage) {
- return ;
- }
-
- return ;
-}
-
export const RedirectBackToMainForm = () => {
const mainPageKey = useNavigationParam('mainPageKey');
const { navigateToPage } = useNavigatePage();
diff --git a/src/layout/Subform/Summary/SubformSummaryTable.tsx b/src/layout/Subform/Summary/SubformSummaryTable.tsx
index 75e5b37d15..9d2cb2a1c4 100644
--- a/src/layout/Subform/Summary/SubformSummaryTable.tsx
+++ b/src/layout/Subform/Summary/SubformSummaryTable.tsx
@@ -1,5 +1,4 @@
import React from 'react';
-import { useNavigate } from 'react-router-dom';
import { Paragraph, Spinner, Table } from '@digdir/designsystemet-react';
import { Grid } from '@material-ui/core';
@@ -13,7 +12,7 @@ import { useStrictDataElements, useStrictInstanceId } from 'src/features/instanc
import { Lang } from 'src/features/language/Lang';
import { useLanguage } from 'src/features/language/useLanguage';
import { usePdfModeActive } from 'src/features/pdf/PDFWrapper';
-import { useIsSubformPage } from 'src/features/routing/AppRoutingContext';
+import { useIsSubformPage, useNavigate } from 'src/features/routing/AppRoutingContext';
import { isSubformValidation } from 'src/features/validation';
import { useComponentValidationsForNode } from 'src/features/validation/selectors/componentValidationsForNode';
import { ComponentStructureWrapper } from 'src/layout/ComponentStructureWrapper';
diff --git a/src/utils/layout/NodesContext.tsx b/src/utils/layout/NodesContext.tsx
index d844ceef6f..dd4d06e902 100644
--- a/src/utils/layout/NodesContext.tsx
+++ b/src/utils/layout/NodesContext.tsx
@@ -17,11 +17,8 @@ import { UpdateAttachmentsForCypress } from 'src/features/attachments/UpdateAtta
import { HiddenComponentsProvider } from 'src/features/form/dynamics/HiddenComponentsProvider';
import { useLayouts } from 'src/features/form/layout/LayoutsContext';
import { useLaxLayoutSettings, useLayoutSettings } from 'src/features/form/layoutSettings/LayoutSettingsContext';
-import { FD } from 'src/features/formData/FormDataWrite';
import { OptionsStorePlugin } from 'src/features/options/OptionsStorePlugin';
import { useIsCurrentView } from 'src/features/routing/AppRoutingContext';
-import { MaintainInitialValidationsInNodesContext } from 'src/features/validation/backendValidation/BackendValidation';
-import { useGetCachedInitialValidations } from 'src/features/validation/backendValidation/backendValidationQuery';
import { ExpressionValidation } from 'src/features/validation/expressionValidation/ExpressionValidation';
import {
LoadingBlockerWaitForValidation,
@@ -51,7 +48,6 @@ import { LayoutPages } from 'src/utils/layout/LayoutPages';
import { RepeatingChildrenStorePlugin } from 'src/utils/layout/plugins/RepeatingChildrenStorePlugin';
import { TraversalTask } from 'src/utils/layout/useNodeTraversal';
import type { AttachmentsStorePluginConfig } from 'src/features/attachments/AttachmentsStorePlugin';
-import type { FDSaveFinished } from 'src/features/formData/FormDataWriteStateMachine';
import type { OptionsStorePluginConfig } from 'src/features/options/OptionsStorePlugin';
import type { ValidationsProcessedLast } from 'src/features/validation';
import type { ValidationStorePluginConfig } from 'src/features/validation/ValidationStorePlugin';
@@ -129,7 +125,6 @@ export interface SetPagePropRequest {
export enum NodesReadiness {
Ready = 'READY',
NotReady = 'NOT READY',
- WaitingUntilLastSaveHasProcessed = 'WAIT FOR SAVE',
}
export type NodesContext = {
@@ -144,7 +139,6 @@ export type NodesContext = {
prevNodeData: { [key: string]: NodeData } | undefined; // Earlier node data from before the state became non-ready
hiddenViaRules: { [key: string]: true | undefined };
hiddenViaRulesRan: boolean;
- validationsProcessedLast: ValidationsProcessedLast;
layouts: ILayouts | undefined; // Used to detect if the layouts have changed
stages: GeneratorStagesContext;
@@ -158,8 +152,6 @@ export type NodesContext = {
addPage: (pageKey: string) => void;
setPageProps: (requests: SetPagePropRequest[]) => void;
markReady: (reason: string, readiness?: NodesReadiness) => void;
- onSaveFinished: (result: FDSaveFinished) => void;
- setLatestInitialValidations: (validations: ValidationsProcessedLast['initial']) => void;
reset: (layouts: ILayouts, validationsProcessedLast: ValidationsProcessedLast) => void;
@@ -349,28 +341,6 @@ export function createNodesDataStore({ registry, validationsProcessedLast }: Cre
}),
markReady: (reason, readiness = NodesReadiness.Ready) =>
set((state) => setReadiness({ state, target: readiness, reason })),
- onSaveFinished: (result) =>
- set((state) => {
- if (state.readiness !== NodesReadiness.WaitingUntilLastSaveHasProcessed) {
- generatorLog('logReadiness', `Marking state as NOT READY: processing save results`);
- }
-
- return {
- readiness: NodesReadiness.WaitingUntilLastSaveHasProcessed,
- prevNodeData: state.nodeData,
- validationsProcessedLast: {
- ...state.validationsProcessedLast,
- incremental: result.validationIssues,
- },
- };
- }),
- setLatestInitialValidations: (validations) =>
- set((state) => ({
- validationsProcessedLast: {
- ...state.validationsProcessedLast,
- initial: validations,
- },
- })),
reset: (layouts, validationsProcessedLast: ValidationsProcessedLast) =>
set(() => {
@@ -518,12 +488,12 @@ const {
export const NodesProvider = ({ children }: React.PropsWithChildren) => {
const registry = useRegistry();
- const processedLast = Validation.useProcessedLastRef();
+ const getProcessedLast = Validation.useGetProcessedLast();
return (
@@ -551,7 +521,7 @@ function ProvideGlobalContext({ children, registry }: PropsWithChildren<{ regist
const layouts = Store.useSelector((s) => s.layouts);
const markNotReady = NodesInternal.useMarkNotReady();
const reset = Store.useSelector((s) => s.reset);
- const processedLast = Validation.useProcessedLastRef();
+ const getProcessedLast = Validation.useGetProcessedLast();
const pagesRef = useRef();
if (!pagesRef.current) {
pagesRef.current = new LayoutPages();
@@ -561,9 +531,9 @@ function ProvideGlobalContext({ children, registry }: PropsWithChildren<{ regist
if (layouts !== latestLayouts) {
markNotReady('new layouts');
pagesRef.current = new LayoutPages();
- reset(latestLayouts, processedLast.current);
+ reset(latestLayouts, getProcessedLast());
}
- }, [latestLayouts, layouts, markNotReady, reset, processedLast]);
+ }, [latestLayouts, layouts, markNotReady, reset, getProcessedLast]);
const layoutMap = useMemo(() => {
const out: { [id: string]: CompExternal } = {};
@@ -647,16 +617,6 @@ function IndicateReadiness() {
);
}
-function MarkAsReady() {
- return (
- <>
-
-
-
- >
- );
-}
-
/**
* Some selectors (like NodeTraversal) only re-runs when the data store is 'ready', and when nodes start being added
* or removed, the store is marked as not ready. This component will mark the store as ready when all nodes are added,
@@ -665,49 +625,27 @@ function MarkAsReady() {
* This causes the node traversal selectors to re-run only when all nodes in a new repeating group row (and similar)
* have been added.
*/
-function InnerMarkAsReady() {
+function MarkAsReady() {
+ const store = Store.useStore();
const markReady = Store.useSelector((s) => s.markReady);
const readiness = Store.useSelector((s) => s.readiness);
const hiddenViaRulesRan = Store.useSelector((s) => s.hiddenViaRulesRan);
const stagesFinished = GeneratorStages.useIsFinished();
- const hasUnsavedChanges = FD.useHasUnsavedChanges();
const registry = GeneratorInternal.useRegistry();
- const getCachedInitialValidations = useGetCachedInitialValidations();
// Even though the getAwaitingCommits() function works on refs in the GeneratorStages context, the effects of such
// commits always changes the NodesContext. Thus our useSelector() re-runs and re-renders this components when
// commits are done.
const getAwaitingCommits = useGetAwaitingCommits();
- const savingOk = readiness === NodesReadiness.WaitingUntilLastSaveHasProcessed ? !hasUnsavedChanges : true;
- const checkNodeStates = stagesFinished && savingOk && hiddenViaRulesRan && readiness !== NodesReadiness.Ready;
+ const checkNodeStates = stagesFinished && hiddenViaRulesRan && readiness !== NodesReadiness.Ready;
const nodeStateReady = Store.useSelector((state) => {
if (!checkNodeStates) {
return false;
}
- const { cachedInitialValidations, isFetching } = getCachedInitialValidations();
- if (isFetching || cachedInitialValidations !== state.validationsProcessedLast.initial) {
- generatorLog('logReadiness', `Initial validations are still being fetched, waiting...`);
- return false;
- }
-
- for (const nodeData of Object.values(state.nodeData)) {
- const def = getComponentDef(nodeData.layout.type) as LayoutComponent;
- const nodeReady = def.stateIsReady(nodeData);
- const pluginsReady = def.pluginStateIsReady(nodeData, state);
- if (!nodeReady || !pluginsReady) {
- generatorLog(
- 'logReadiness',
- `Node ${nodeData.layout.id} is not ready yet because of ` +
- `${nodeReady ? 'plugins' : pluginsReady ? 'node' : 'both node and plugins'}`,
- );
- return false;
- }
- }
-
- return true;
+ return areAllNodesReady(state);
});
const maybeReady = checkNodeStates && nodeStateReady;
@@ -719,22 +657,40 @@ function InnerMarkAsReady() {
// isn't ready.
return setIdleInterval(registry, () => {
const awaiting = getAwaitingCommits();
- if (awaiting === 0) {
- markReady('idle and nothing to commit');
- return true;
+ if (awaiting > 0) {
+ generatorLog('logReadiness', `Not quite ready yet (waiting for ${awaiting} commits)`);
+ return false;
}
- generatorLog('logReadiness', `Not quite ready yet (waiting for ${awaiting} commits)`);
- return false;
+ markReady('idle, nothing to commit');
+ return true;
});
}
return () => undefined;
- }, [maybeReady, getAwaitingCommits, markReady, registry]);
+ }, [maybeReady, getAwaitingCommits, markReady, registry, store]);
return null;
}
+function areAllNodesReady(state: NodesContext) {
+ for (const nodeData of Object.values(state.nodeData)) {
+ const def = getComponentDef(nodeData.layout.type) as LayoutComponent;
+ const nodeReady = def.stateIsReady(nodeData);
+ const pluginsReady = def.pluginStateIsReady(nodeData, state);
+ if (!nodeReady || !pluginsReady) {
+ generatorLog(
+ 'logReadiness',
+ `Node ${nodeData.layout.id} is not ready yet because of ` +
+ `${nodeReady ? 'plugins' : pluginsReady ? 'node' : 'both node and plugins'}`,
+ );
+ return false;
+ }
+ }
+
+ return true;
+}
+
const IDLE_COUNTDOWN = 3;
/**
@@ -775,19 +731,6 @@ function setIdleInterval(registry: MutableRefObject, fn: () => boolean
return () => (id === undefined ? undefined : cancel(id));
}
-function RegisterOnSaveFinished() {
- const setOnSaveFinished = FD.useSetOnSaveFinished();
- const ourOnSaveFinished = Store.useSelector((s) => s.onSaveFinished);
-
- useEffect(() => {
- setOnSaveFinished((result) => {
- ourOnSaveFinished(result);
- });
- }, [ourOnSaveFinished, setOnSaveFinished]);
-
- return null;
-}
-
function BlockUntilLoaded({ children }: PropsWithChildren) {
const hasBeenReady = useRef(false);
const ready = Store.useSelector((state) => {
@@ -1147,29 +1090,17 @@ export const NodesInternal = {
const store = Store.useLaxStore();
const waitForState = useWaitForState(store);
const waitForCommits = Store.useSelector((s) => s.waitForCommits);
- return useCallback(
- async (lastValidations?: ValidationsProcessedLast) => {
- await waitForState((state) => {
- if (state === ContextNotProvided) {
- return true;
- }
- const initialIsLatest =
- lastValidations?.initial === undefined ||
- lastValidations.initial === state.validationsProcessedLast.initial;
- const incrementalIsLatest =
- lastValidations?.incremental === undefined ||
- lastValidations.incremental === state.validationsProcessedLast.incremental;
- if (!incrementalIsLatest || !initialIsLatest) {
- return false;
- }
- return state.readiness === NodesReadiness.Ready && state.hiddenViaRulesRan;
- });
- if (waitForCommits) {
- await waitForCommits();
+ return useCallback(async () => {
+ await waitForState((state) => {
+ if (state === ContextNotProvided) {
+ return true;
}
- },
- [waitForState, waitForCommits],
- );
+ return state.readiness === NodesReadiness.Ready && state.hiddenViaRulesRan;
+ });
+ if (waitForCommits) {
+ await waitForCommits();
+ }
+ }, [waitForState, waitForCommits]);
},
useMarkNotReady() {
const markReady = Store.useSelector((s) => s.markReady);
diff --git a/src/utils/layout/generator/GeneratorDataSources.tsx b/src/utils/layout/generator/GeneratorDataSources.tsx
index aaae8f43d1..7c00edf8db 100644
--- a/src/utils/layout/generator/GeneratorDataSources.tsx
+++ b/src/utils/layout/generator/GeneratorDataSources.tsx
@@ -29,10 +29,7 @@ const { Provider, hooks } = createHookContext({
useReadableDataTypes: () => DataModels.useReadableDataTypes(),
useExternalApis: () => useExternalApis(useApplicationMetadata().externalApiIds ?? []),
useIsForcedVisibleByDevTools: () => useDevToolsStore((state) => state.isOpen && state.hiddenComponents !== 'hide'),
-
useGetDataElementIdForDataType: () => DataModels.useGetDataElementIdForDataType(),
- useValidationsProcessedLast: () => Validation.useProcessedLast(),
-
useCommitWhenFinished: () => useCommitWhenFinished(),
});
@@ -42,10 +39,7 @@ export const GeneratorData = {
useValidationDataSources,
useDefaultDataType: hooks.useDefaultDataType,
useIsForcedVisibleByDevTools: hooks.useIsForcedVisibleByDevTools,
-
useGetDataElementIdForDataType: hooks.useGetDataElementIdForDataType,
- useValidationsProcessedLast: hooks.useValidationsProcessedLast,
-
useCommitWhenFinished: hooks.useCommitWhenFinished,
};
diff --git a/src/utils/layout/generator/GeneratorStages.tsx b/src/utils/layout/generator/GeneratorStages.tsx
index 43643ad3f6..dd7d2b89c7 100644
--- a/src/utils/layout/generator/GeneratorStages.tsx
+++ b/src/utils/layout/generator/GeneratorStages.tsx
@@ -53,7 +53,9 @@ export type Registry = {
toCommit: RegistryCommitQueues;
toCommitCount: number;
commitTimeout: ReturnType | null;
- validations: ValidationsProcessedLast;
+ validationsProcessed: {
+ [nodeId: string]: ValidationsProcessedLast;
+ };
};
/**
@@ -238,10 +240,7 @@ export function useRegistry() {
setPageProps: [],
},
commitTimeout: null,
- validations: {
- initial: undefined,
- incremental: undefined,
- },
+ validationsProcessed: {},
});
}