): JsonPatchOperation[] {
+ return Object.keys(dirtyFields).map((name) => ({
+ op: get(name, originalValue) ? 'replace' : 'add',
+ path: `/${name.replace(/[.]/g, '/')}`,
+ value: get(name, values) || '',
+ }))
+}
diff --git a/src/pages/dataElementGroupSets/edit/index.ts b/src/pages/dataElementGroupSets/edit/index.ts
new file mode 100644
index 00000000..0069ca37
--- /dev/null
+++ b/src/pages/dataElementGroupSets/edit/index.ts
@@ -0,0 +1 @@
+export { createJsonPatchOperations } from './createJsonPatchOperations'
diff --git a/src/pages/dataElementGroupSets/fields/CodeField.tsx b/src/pages/dataElementGroupSets/fields/CodeField.tsx
new file mode 100644
index 00000000..fedc4eb8
--- /dev/null
+++ b/src/pages/dataElementGroupSets/fields/CodeField.tsx
@@ -0,0 +1,25 @@
+import i18n from '@dhis2/d2-i18n'
+import { InputFieldFF } from '@dhis2/ui'
+import React from 'react'
+import { Field as FieldRFF } from 'react-final-form'
+import { useCheckMaxLengthFromSchema } from '../../../lib'
+import type { SchemaName } from '../../../types'
+
+export function CodeField() {
+ const validate = useCheckMaxLengthFromSchema(
+ 'dataElement' as SchemaName,
+ 'code'
+ )
+
+ return (
+
+ )
+}
diff --git a/src/pages/dataElementGroupSets/fields/CompulsoryField.tsx b/src/pages/dataElementGroupSets/fields/CompulsoryField.tsx
new file mode 100644
index 00000000..922e6412
--- /dev/null
+++ b/src/pages/dataElementGroupSets/fields/CompulsoryField.tsx
@@ -0,0 +1,17 @@
+import i18n from '@dhis2/d2-i18n'
+import { CheckboxFieldFF } from '@dhis2/ui'
+import React from 'react'
+import { Field as FieldRFF } from 'react-final-form'
+
+export function CompulsoryField() {
+ return (
+
+ )
+}
diff --git a/src/pages/dataElementGroupSets/fields/DataDimensionField.tsx b/src/pages/dataElementGroupSets/fields/DataDimensionField.tsx
new file mode 100644
index 00000000..ec7ebc51
--- /dev/null
+++ b/src/pages/dataElementGroupSets/fields/DataDimensionField.tsx
@@ -0,0 +1,17 @@
+import i18n from '@dhis2/d2-i18n'
+import { CheckboxFieldFF } from '@dhis2/ui'
+import React from 'react'
+import { Field as FieldRFF } from 'react-final-form'
+
+export function DataDimensionField() {
+ return (
+
+ )
+}
diff --git a/src/pages/dataElementGroupSets/fields/DataElementGroupsField.module.css b/src/pages/dataElementGroupSets/fields/DataElementGroupsField.module.css
new file mode 100644
index 00000000..f40ea288
--- /dev/null
+++ b/src/pages/dataElementGroupSets/fields/DataElementGroupsField.module.css
@@ -0,0 +1,8 @@
+.dataElementsOptionsFooter {
+ padding: var(--spacers-dp8) 0;
+}
+
+.dataElementsPickedHeader {
+ padding: var(--spacers-dp8) 0;
+ margin: 0;
+}
diff --git a/src/pages/dataElementGroupSets/fields/DataElementGroupsField.tsx b/src/pages/dataElementGroupSets/fields/DataElementGroupsField.tsx
new file mode 100644
index 00000000..cce4dcf8
--- /dev/null
+++ b/src/pages/dataElementGroupSets/fields/DataElementGroupsField.tsx
@@ -0,0 +1,72 @@
+import i18n from '@dhis2/d2-i18n'
+import { ButtonStrip, Button, Field } from '@dhis2/ui'
+import React, { useRef } from 'react'
+import { useField } from 'react-final-form'
+import { useHref } from 'react-router'
+import { DataElementGroupsTransfer } from '../../../components'
+import classes from './DataElementGroupsField.module.css'
+
+/**
+ *
+ * DataElementGroups
+ *
+ */
+export function DataElementGroupsField() {
+ const name = 'dataElementGroups'
+ const { input, meta } = useField(name, {
+ multiple: true,
+ format: (dataElementGroups: { id: string }[]) =>
+ dataElementGroups?.map((dataElementGroups) => dataElementGroups.id),
+ parse: (ids: string[]) => ids.map((id) => ({ id })),
+ validateFields: [],
+ })
+
+ const newDataElementGroupsLink = useHref('/dataElementGroups/new')
+ const dataElementGroupsHandle = useRef({
+ refetch: () => {
+ throw new Error('Not initialized')
+ },
+ })
+
+ const rightHeader = (
+
+ {i18n.t('Selected data element groups')}
+
+ )
+
+ const leftFooter = (
+
+
+
+
+
+
+
+ )
+
+ return (
+
+ input.onChange(selected)}
+ rightHeader={rightHeader}
+ leftFooter={leftFooter}
+ />
+
+ )
+}
diff --git a/src/pages/dataElementGroupSets/fields/DescriptionField.tsx b/src/pages/dataElementGroupSets/fields/DescriptionField.tsx
new file mode 100644
index 00000000..c11298b8
--- /dev/null
+++ b/src/pages/dataElementGroupSets/fields/DescriptionField.tsx
@@ -0,0 +1,28 @@
+import i18n from '@dhis2/d2-i18n'
+import { TextAreaFieldFF } from '@dhis2/ui'
+import React from 'react'
+import { Field as FieldRFF } from 'react-final-form'
+import { useCheckMaxLengthFromSchema } from '../../../lib'
+import type { SchemaName } from '../../../types'
+
+export function DescriptionField() {
+ const validate = useCheckMaxLengthFromSchema(
+ 'dataElement' as SchemaName,
+ 'formName'
+ )
+
+ return (
+
+ )
+}
diff --git a/src/pages/dataElementGroupSets/fields/NameField.tsx b/src/pages/dataElementGroupSets/fields/NameField.tsx
new file mode 100644
index 00000000..b6c38724
--- /dev/null
+++ b/src/pages/dataElementGroupSets/fields/NameField.tsx
@@ -0,0 +1,64 @@
+import i18n from '@dhis2/d2-i18n'
+import { InputFieldFF } from '@dhis2/ui'
+import React, { useMemo } from 'react'
+import { Field as FieldRFF, useField } from 'react-final-form'
+import { useParams } from 'react-router-dom'
+import {
+ composeAsyncValidators,
+ required,
+ useCheckMaxLengthFromSchema,
+ useIsFieldValueUnique,
+} from '../../../lib'
+import { SchemaName } from '../../../types'
+import type { FormValues } from '../form'
+
+function useValidator() {
+ const params = useParams()
+ const dataElementId = params.id as string
+ const checkIsValueTaken = useIsFieldValueUnique({
+ model: 'dataElements',
+ field: 'name',
+ id: dataElementId,
+ })
+
+ const checkMaxLength = useCheckMaxLengthFromSchema(
+ SchemaName.dataElement,
+ 'name'
+ )
+
+ return useMemo(
+ () =>
+ composeAsyncValidators([
+ checkIsValueTaken,
+ checkMaxLength,
+ required,
+ ]),
+ [checkIsValueTaken, checkMaxLength]
+ )
+}
+
+export function NameField() {
+ const validator = useValidator()
+ const { meta } = useField('name', {
+ subscription: { validating: true },
+ })
+
+ return (
+
+ loading={meta.validating}
+ component={InputFieldFF}
+ dataTest="dataelementsformfields-name"
+ required
+ inputWidth="400px"
+ label={i18n.t('{{fieldLabel}} (required)', {
+ fieldLabel: i18n.t('Name'),
+ })}
+ name="name"
+ helpText={i18n.t(
+ 'A data element name should be concise and easy to recognize.'
+ )}
+ validate={(name?: string) => validator(name)}
+ validateFields={[]}
+ />
+ )
+}
diff --git a/src/pages/dataElementGroupSets/fields/ShortNameField.tsx b/src/pages/dataElementGroupSets/fields/ShortNameField.tsx
new file mode 100644
index 00000000..7e1845d5
--- /dev/null
+++ b/src/pages/dataElementGroupSets/fields/ShortNameField.tsx
@@ -0,0 +1,62 @@
+import i18n from '@dhis2/d2-i18n'
+import { InputFieldFF } from '@dhis2/ui'
+import React, { useMemo } from 'react'
+import { Field as FieldRFF, useField } from 'react-final-form'
+import { useParams } from 'react-router-dom'
+import {
+ composeAsyncValidators,
+ required,
+ useCheckMaxLengthFromSchema,
+ useIsFieldValueUnique,
+} from '../../../lib'
+import type { SchemaName } from '../../../types'
+import type { FormValues } from '../form'
+
+function useValidator() {
+ const params = useParams()
+ const dataElementId = params.id as string
+ const checkIsValueTaken = useIsFieldValueUnique({
+ model: 'dataElements',
+ field: 'name',
+ id: dataElementId,
+ })
+
+ const checkMaxLength = useCheckMaxLengthFromSchema(
+ 'dataElement' as SchemaName,
+ 'formName'
+ )
+
+ return useMemo(
+ () =>
+ composeAsyncValidators([
+ checkIsValueTaken,
+ checkMaxLength,
+ required,
+ ]),
+ [checkIsValueTaken, checkMaxLength]
+ )
+}
+
+export function ShortNameField() {
+ const validator = useValidator()
+ const { meta } = useField('shortName', {
+ subscription: { validating: true },
+ })
+
+ return (
+
+ loading={meta.validating}
+ component={InputFieldFF}
+ dataTest="dataelementsformfields-shortname"
+ required
+ inputWidth="400px"
+ label={i18n.t('{{fieldLabel}} (required)', {
+ fieldLabel: i18n.t('Short name'),
+ })}
+ name="shortName"
+ helpText={i18n.t('Often used in reports where space is limited')}
+ validate={(name?: string) => validator(name)}
+ validateFields={[]}
+ />
+ )
+}
diff --git a/src/pages/dataElementGroupSets/fields/index.ts b/src/pages/dataElementGroupSets/fields/index.ts
new file mode 100644
index 00000000..a25ce40d
--- /dev/null
+++ b/src/pages/dataElementGroupSets/fields/index.ts
@@ -0,0 +1,7 @@
+export { CodeField } from './CodeField'
+export { CompulsoryField } from './CompulsoryField'
+export { DataDimensionField } from './DataDimensionField'
+export { DataElementGroupsField } from './DataElementGroupsField'
+export { DescriptionField } from './DescriptionField'
+export { NameField } from './NameField'
+export { ShortNameField } from './ShortNameField'
diff --git a/src/pages/dataElementGroupSets/form/DataElementGroupSetFormFields.tsx b/src/pages/dataElementGroupSets/form/DataElementGroupSetFormFields.tsx
new file mode 100644
index 00000000..af1911e6
--- /dev/null
+++ b/src/pages/dataElementGroupSets/form/DataElementGroupSetFormFields.tsx
@@ -0,0 +1,73 @@
+import i18n from '@dhis2/d2-i18n'
+import React from 'react'
+import {
+ StandardFormSection,
+ StandardFormSectionTitle,
+ StandardFormSectionDescription,
+ StandardFormField,
+} from '../../../components'
+import {
+ CodeField,
+ CompulsoryField,
+ DataDimensionField,
+ DataElementGroupsField,
+ DescriptionField,
+ NameField,
+ ShortNameField,
+} from '../fields'
+
+export function DataElementGroupSetFormFields() {
+ return (
+ <>
+
+
+ {i18n.t('Basic information')}
+
+
+
+ {i18n.t(
+ 'Set up the information for this data element group'
+ )}
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+ {i18n.t('Data elements')}
+
+
+
+ {i18n.t('@TODO')}
+
+
+
+
+
+
+ >
+ )
+}
diff --git a/src/pages/dataElementGroupSets/form/dataElementGroupSetSchema.ts b/src/pages/dataElementGroupSets/form/dataElementGroupSetSchema.ts
new file mode 100644
index 00000000..ea1c9afe
--- /dev/null
+++ b/src/pages/dataElementGroupSets/form/dataElementGroupSetSchema.ts
@@ -0,0 +1,11 @@
+import { z } from 'zod'
+
+export const dataElementGroupSetSchema = z
+ .object({
+ name: z.string().trim(),
+ shortName: z.string().trim(),
+ code: z.string().trim(),
+ description: z.string().trim(),
+ dataElements: z.array(z.object({ id: z.string() })),
+ })
+ .partial()
diff --git a/src/pages/dataElementGroupSets/form/index.ts b/src/pages/dataElementGroupSets/form/index.ts
new file mode 100644
index 00000000..e12100af
--- /dev/null
+++ b/src/pages/dataElementGroupSets/form/index.ts
@@ -0,0 +1,3 @@
+export { DataElementGroupSetFormFields } from './DataElementGroupSetFormFields'
+export type { FormValues } from './types'
+export { validate } from './validate'
diff --git a/src/pages/dataElementGroupSets/form/types.ts b/src/pages/dataElementGroupSets/form/types.ts
new file mode 100644
index 00000000..a9466da4
--- /dev/null
+++ b/src/pages/dataElementGroupSets/form/types.ts
@@ -0,0 +1,3 @@
+import { DataElementGroupSet } from '../../../types/generated'
+
+export type FormValues = DataElementGroupSet
diff --git a/src/pages/dataElementGroupSets/form/validate.ts b/src/pages/dataElementGroupSets/form/validate.ts
new file mode 100644
index 00000000..efe2d856
--- /dev/null
+++ b/src/pages/dataElementGroupSets/form/validate.ts
@@ -0,0 +1,27 @@
+import { setIn } from 'final-form'
+import { dataElementGroupSetSchema } from './dataElementGroupSetSchema'
+import type { FormValues } from './types'
+
+// @TODO: Figure out if there's a utility for this? I couldn't find one
+function segmentsToPath(segments: Array) {
+ return segments.reduce((path, segment) => {
+ return typeof segment === 'number'
+ ? `${path}[${segment}]`
+ : `${path}.${segment}`
+ }) as string
+}
+
+export function validate(values: FormValues) {
+ const zodResult = dataElementGroupSetSchema.safeParse(values)
+
+ if (zodResult.success !== false) {
+ return undefined
+ }
+
+ const allFormErrors = zodResult.error.issues.reduce((formErrors, error) => {
+ const errorPath = segmentsToPath(error.path)
+ return setIn(formErrors, errorPath, error.message)
+ }, {})
+
+ return allFormErrors
+}