From b0d76463309b8450fb614410c3440c02ef45b0f2 Mon Sep 17 00:00:00 2001 From: Diana Nanyanzi Date: Thu, 19 Dec 2024 17:23:41 +0300 Subject: [PATCH 1/5] feat: add modal and fields for create functionality --- src/components/modals/CreateModal.tsx | 61 +++++++++++++++++++++++++++ src/components/modals/Fields.tsx | 41 ++++++++++++++++++ 2 files changed, 102 insertions(+) create mode 100644 src/components/modals/CreateModal.tsx create mode 100644 src/components/modals/Fields.tsx diff --git a/src/components/modals/CreateModal.tsx b/src/components/modals/CreateModal.tsx new file mode 100644 index 0000000..eed466c --- /dev/null +++ b/src/components/modals/CreateModal.tsx @@ -0,0 +1,61 @@ +import { + Modal, + ModalTitle, + ModalContent, + ModalActions, + Button, + ButtonStrip, + ReactFinalForm, +} from '@dhis2/ui' +import React from 'react' +import i18n from '../../locales' + +const { Form } = ReactFinalForm + +interface FieldValues { + key: string + namespace?: string +} + +interface CreateModalProps { + closeModal: () => void + handleCreate: (FieldValues) => void + children: React.ReactNode + title: string +} + +const CreateModal = ({ + handleCreate, + closeModal, + children, + title, +}: CreateModalProps) => { + const onSubmit = (values) => { + handleCreate(values) + } + + return ( + +
+ {({ handleSubmit }) => ( + + {title} + {children} + + + + + + +
+ )} + +
+ ) +} + +export default CreateModal diff --git a/src/components/modals/Fields.tsx b/src/components/modals/Fields.tsx new file mode 100644 index 0000000..80a8873 --- /dev/null +++ b/src/components/modals/Fields.tsx @@ -0,0 +1,41 @@ +import { + ReactFinalForm, + InputFieldFF, + hasValue, + composeValidators, + alphaNumeric, +} from '@dhis2/ui' +import React from 'react' +import i18n from '../../locales' + +const { Field } = ReactFinalForm + +export const KeysField = ({ initialFocus }: { initialFocus?: boolean }) => { + return ( + + ) +} + +export const NamespaceField = ({ + initialFocus, +}: { + initialFocus?: boolean +}) => { + return ( + + ) +} From 35dd3b0ec2c93be3d0f65d242c05f5cfcaa6f1a9 Mon Sep 17 00:00:00 2001 From: Diana Nanyanzi Date: Thu, 19 Dec 2024 17:24:21 +0300 Subject: [PATCH 2/5] feat: create new namespace functionality --- .../sections/NamespaceDataSection.tsx | 36 ++++++++++++++++--- 1 file changed, 32 insertions(+), 4 deletions(-) diff --git a/src/components/sections/NamespaceDataSection.tsx b/src/components/sections/NamespaceDataSection.tsx index 56eed5d..d7ffb41 100644 --- a/src/components/sections/NamespaceDataSection.tsx +++ b/src/components/sections/NamespaceDataSection.tsx @@ -1,10 +1,13 @@ -import { useDataQuery } from '@dhis2/app-runtime' +import { useDataEngine, useDataQuery } from '@dhis2/app-runtime' import { IconAdd24, colors } from '@dhis2/ui' -import React from 'react' +import React, { useState } from 'react' +import { useNavigate, useParams } from 'react-router-dom' import classes from '../../App.module.css' import i18n from '../../locales' import ErrorNotice from '../error/ErrorNotice' import CenteredLoader from '../loader/Loader' +import CreateModal from '../modals/CreateModal' +import { KeysField, NamespaceField } from '../modals/Fields' import ItemsTable from '../table/ItemsTable' import CreateButton from './CreateButton' import SearchField from './SearchField' @@ -14,7 +17,22 @@ interface QueryResults { } const NamespaceDataSection = ({ query }) => { - const { error, loading, data } = useDataQuery(query) + const engine = useDataEngine() + const navigate = useNavigate() + const { store } = useParams() + const [openModal, setOpenModal] = useState(false) + + const { error, loading, data, refetch } = useDataQuery(query) + + const handleCreate = async (values) => { + await engine.mutate({ + type: 'create', + resource: `${store}/${values?.namespace}/${values?.key}`, + data: () => ({}), + }) + refetch() + navigate(`edit/${values?.namespace}`) + } if (error) { return @@ -30,13 +48,23 @@ const NamespaceDataSection = ({ query }) => { console.log('create new namespace')} + handleClick={() => setOpenModal(true)} icon={} />
{data && }
+ {openModal && ( + setOpenModal(false)} + handleCreate={handleCreate} + > + + + + )} ) } From 4577d52fc550c50fa4e9a543536e6b59cf426d87 Mon Sep 17 00:00:00 2001 From: Diana Nanyanzi Date: Thu, 19 Dec 2024 17:24:52 +0300 Subject: [PATCH 3/5] feat: create new key functionality --- src/components/sections/KeysDataSection.tsx | 36 ++++++++++++++++++--- 1 file changed, 32 insertions(+), 4 deletions(-) diff --git a/src/components/sections/KeysDataSection.tsx b/src/components/sections/KeysDataSection.tsx index 0de43c6..d61f326 100644 --- a/src/components/sections/KeysDataSection.tsx +++ b/src/components/sections/KeysDataSection.tsx @@ -1,12 +1,14 @@ -import { useDataQuery } from '@dhis2/app-runtime' +import { useDataEngine, useDataQuery } from '@dhis2/app-runtime' import { IconAdd16, colors } from '@dhis2/ui' -import React, { useEffect } from 'react' +import React, { useEffect, useState } from 'react' import { useParams } from 'react-router-dom' import classes from '../../App.module.css' import i18n from '../../locales' import ErrorNotice from '../error/ErrorNotice' import PanelHeader from '../header/PanelHeader' import CenteredLoader from '../loader/Loader' +import CreateModal from '../modals/CreateModal' +import { KeysField } from '../modals/Fields' import ItemsTable from '../table/ItemsTable' import CreateButton from './CreateButton' import SearchField from './SearchField' @@ -16,7 +18,10 @@ interface QueryResults { } const KeysDataSection = ({ query }) => { - const { namespace: currentNamespace } = useParams() + const engine = useDataEngine() + const { store, namespace: currentNamespace } = useParams() + + const [openModal, setOpenModal] = useState(false) const { error, loading, data, refetch } = useDataQuery( query, @@ -27,6 +32,20 @@ const KeysDataSection = ({ query }) => { } ) + const handleCreate = async ({ key }) => { + await engine.mutate( + { + type: 'create', + resource: `${store}/${currentNamespace}/${key}`, + data: () => ({}), + }, + { + onComplete: () => setOpenModal(false), + } + ) + refetch({ id: currentNamespace }) + } + useEffect(() => { refetch({ id: currentNamespace }) }, [currentNamespace]) @@ -47,7 +66,7 @@ const KeysDataSection = ({ query }) => { console.log('create new key')} + handleClick={() => setOpenModal(true)} icon={} /> @@ -57,6 +76,15 @@ const KeysDataSection = ({ query }) => {
{data && }
+ {openModal && ( + setOpenModal(false)} + handleCreate={handleCreate} + > + + + )} ) } From 1ae85dfb8a38274d33abb60aa4a8a631b9b2d972 Mon Sep 17 00:00:00 2001 From: Diana Nanyanzi Date: Thu, 19 Dec 2024 19:46:53 +0300 Subject: [PATCH 4/5] chore: type fixes --- i18n/en.pot | 53 +++++++++++++++++++-------- src/components/modals/CreateModal.tsx | 4 +- src/components/pages/Namespaces.tsx | 5 --- 3 files changed, 39 insertions(+), 23 deletions(-) diff --git a/i18n/en.pot b/i18n/en.pot index 2dc9dcc..bc65235 100644 --- a/i18n/en.pot +++ b/i18n/en.pot @@ -5,27 +5,39 @@ msgstr "" "Content-Type: text/plain; charset=utf-8\n" "Content-Transfer-Encoding: 8bit\n" "Plural-Forms: nplurals=2; plural=(n != 1)\n" -"POT-Creation-Date: 2024-12-18T09:41:03.249Z\n" -"PO-Revision-Date: 2024-12-18T09:41:03.250Z\n" +"POT-Creation-Date: 2024-12-19T16:36:13.574Z\n" +"PO-Revision-Date: 2024-12-19T16:36:13.575Z\n" msgid "An error has occurred" msgstr "An error has occurred" +msgid "404 Page Not Found" +msgstr "404 Page Not Found" + msgid "Back to datastore" msgstr "Back to datastore" -msgid "Back to all namespaces" -msgstr "Back to all namespaces" +msgid "An error has occurred. Try again" +msgstr "An error has occurred. Try again" -msgid "Search namespaces" -msgstr "Search namespaces" +msgid "Cancel" +msgstr "Cancel" -msgid "New Namespace" -msgstr "New Namespace" +msgid "Add" +msgstr "Add" + +msgid "Key" +msgstr "Key" msgid "Namespace" msgstr "Namespace" +msgid "Back to all namespaces" +msgstr "Back to all namespaces" + +msgid "Configure Namespaces" +msgstr "Configure Namespaces" + msgid "Close" msgstr "Close" @@ -38,8 +50,14 @@ msgstr "Save changes" msgid "Choose a key to start editing" msgstr "Choose a key to start editing" -msgid "Keys" -msgstr "Keys" +msgid "DataStore" +msgstr "DataStore" + +msgid "UserDataStore" +msgstr "UserDataStore" + +msgid "keys" +msgstr "keys" msgid "New Key" msgstr "New Key" @@ -47,14 +65,17 @@ msgstr "New Key" msgid "Search keys" msgstr "Search keys" -msgid "Key" -msgstr "Key" +msgid "Add New Key" +msgstr "Add New Key" -msgid "DataStore" -msgstr "DataStore" +msgid "Search namespaces" +msgstr "Search namespaces" -msgid "UserDataStore" -msgstr "UserDataStore" +msgid "New Namespace" +msgstr "New Namespace" + +msgid "Add New Namespace" +msgstr "Add New Namespace" msgid "Search" msgstr "Search" diff --git a/src/components/modals/CreateModal.tsx b/src/components/modals/CreateModal.tsx index eed466c..b0ddb2e 100644 --- a/src/components/modals/CreateModal.tsx +++ b/src/components/modals/CreateModal.tsx @@ -19,7 +19,7 @@ interface FieldValues { interface CreateModalProps { closeModal: () => void - handleCreate: (FieldValues) => void + handleCreate: ({ key, namespace }: FieldValues) => void children: React.ReactNode title: string } @@ -35,7 +35,7 @@ const CreateModal = ({ } return ( - +
{({ handleSubmit }) => ( diff --git a/src/components/pages/Namespaces.tsx b/src/components/pages/Namespaces.tsx index a7494bb..c1ec27d 100644 --- a/src/components/pages/Namespaces.tsx +++ b/src/components/pages/Namespaces.tsx @@ -18,11 +18,6 @@ const dataStoreNamespacesQuery = { }, } -export type FieldValues = { - namespace?: string - key?: string -} - const NamespacesPage = () => { const navigate = useNavigate() const { store } = useParams() From 2b9d8c1565cb9d3e0a5147b82b967c5b3023548d Mon Sep 17 00:00:00 2001 From: Diana Nanyanzi Date: Fri, 20 Dec 2024 00:20:07 +0300 Subject: [PATCH 5/5] feat: add alerts for error and success during creation process --- src/components/sections/KeysDataSection.tsx | 33 +++++++++--- .../sections/NamespaceDataSection.tsx | 52 +++++++++++++++---- src/hooks/useCustomAlert.tsx | 15 ++++++ 3 files changed, 83 insertions(+), 17 deletions(-) create mode 100644 src/hooks/useCustomAlert.tsx diff --git a/src/components/sections/KeysDataSection.tsx b/src/components/sections/KeysDataSection.tsx index d61f326..a05a934 100644 --- a/src/components/sections/KeysDataSection.tsx +++ b/src/components/sections/KeysDataSection.tsx @@ -3,6 +3,7 @@ import { IconAdd16, colors } from '@dhis2/ui' import React, { useEffect, useState } from 'react' import { useParams } from 'react-router-dom' import classes from '../../App.module.css' +import useCustomAlert from '../../hooks/useCustomAlert' import i18n from '../../locales' import ErrorNotice from '../error/ErrorNotice' import PanelHeader from '../header/PanelHeader' @@ -21,7 +22,9 @@ const KeysDataSection = ({ query }) => { const engine = useDataEngine() const { store, namespace: currentNamespace } = useParams() - const [openModal, setOpenModal] = useState(false) + const [openCreateModal, setOpenCreateModal] = useState(false) + + const { showError, showSuccess } = useCustomAlert() const { error, loading, data, refetch } = useDataQuery( query, @@ -40,10 +43,28 @@ const KeysDataSection = ({ query }) => { data: () => ({}), }, { - onComplete: () => setOpenModal(false), + onComplete() { + setOpenCreateModal(false) + showSuccess( + i18n.t("Key '{{key}}' added successfully", { + key, + }) + ) + refetch({ id: currentNamespace }) + }, + onError(error) { + showError( + i18n.t( + 'There was a problem adding this key - {{error}}', + { + error: error.message, + interpolation: { escapeValue: false }, + } + ) + ) + }, } ) - refetch({ id: currentNamespace }) } useEffect(() => { @@ -66,7 +87,7 @@ const KeysDataSection = ({ query }) => { setOpenModal(true)} + handleClick={() => setOpenCreateModal(true)} icon={} /> @@ -76,10 +97,10 @@ const KeysDataSection = ({ query }) => {
{data && }
- {openModal && ( + {openCreateModal && ( setOpenModal(false)} + closeModal={() => setOpenCreateModal(false)} handleCreate={handleCreate} > diff --git a/src/components/sections/NamespaceDataSection.tsx b/src/components/sections/NamespaceDataSection.tsx index d7ffb41..34a6a4d 100644 --- a/src/components/sections/NamespaceDataSection.tsx +++ b/src/components/sections/NamespaceDataSection.tsx @@ -3,6 +3,7 @@ import { IconAdd24, colors } from '@dhis2/ui' import React, { useState } from 'react' import { useNavigate, useParams } from 'react-router-dom' import classes from '../../App.module.css' +import useCustomAlert from '../../hooks/useCustomAlert' import i18n from '../../locales' import ErrorNotice from '../error/ErrorNotice' import CenteredLoader from '../loader/Loader' @@ -20,18 +21,47 @@ const NamespaceDataSection = ({ query }) => { const engine = useDataEngine() const navigate = useNavigate() const { store } = useParams() - const [openModal, setOpenModal] = useState(false) + const [openCreateModal, setOpenCreateModal] = useState(false) + + const { showError, showSuccess } = useCustomAlert() const { error, loading, data, refetch } = useDataQuery(query) const handleCreate = async (values) => { - await engine.mutate({ - type: 'create', - resource: `${store}/${values?.namespace}/${values?.key}`, - data: () => ({}), - }) - refetch() - navigate(`edit/${values?.namespace}`) + await engine.mutate( + { + type: 'create', + resource: `${store}/${values?.namespace}/${values?.key}`, + data: () => ({}), + }, + { + onComplete() { + showSuccess( + i18n.t( + "Namespace '{{namespace}}' and key '{{key}}' added successfully!", + { + namespace: values.namespace, + key: values.key, + } + ) + ) + refetch() + navigate(`edit/${values?.namespace}`) + setOpenCreateModal(false) + }, + onError(error) { + showError( + i18n.t( + 'There was a problem adding this namespace - {{error}}', + { + error: error.message, + interpolation: { escapeValue: false }, + } + ) + ) + }, + } + ) } if (error) { @@ -48,17 +78,17 @@ const NamespaceDataSection = ({ query }) => { setOpenModal(true)} + handleClick={() => setOpenCreateModal(true)} icon={} />
{data && }
- {openModal && ( + {openCreateModal && ( setOpenModal(false)} + closeModal={() => setOpenCreateModal(false)} handleCreate={handleCreate} > diff --git a/src/hooks/useCustomAlert.tsx b/src/hooks/useCustomAlert.tsx new file mode 100644 index 0000000..4b01b94 --- /dev/null +++ b/src/hooks/useCustomAlert.tsx @@ -0,0 +1,15 @@ +import { useAlert } from '@dhis2/app-runtime' + +const useCustomAlert = () => { + const { show } = useAlert( + ({ message }) => message, + ({ isError }) => + isError ? { critical: true } : { success: true, duration: 3000 } + ) + return { + showSuccess: (message) => show({ message }), + showError: (message) => show({ message, isError: true }), + } +} + +export default useCustomAlert