From 81944cae0bd873257f84288a1817d48c3423d304 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Dafydd=20Ll=C5=B7r=20Pearson?= Date: Sat, 18 Nov 2023 17:14:24 +0000 Subject: [PATCH] feat: Basic db writing --- .../@planx/components/Checklist/Public.tsx | 4 +-- .../src/@planx/components/Send/Editor.tsx | 4 +-- .../FlowEditor/lib/analyticsProvider.tsx | 27 +++++++++++++++++++ editor.planx.uk/src/ui/ErrorWrapper.tsx | 22 +++++++++------ 4 files changed, 45 insertions(+), 12 deletions(-) diff --git a/editor.planx.uk/src/@planx/components/Checklist/Public.tsx b/editor.planx.uk/src/@planx/components/Checklist/Public.tsx index f944eac47d..e26d49f642 100644 --- a/editor.planx.uk/src/@planx/components/Checklist/Public.tsx +++ b/editor.planx.uk/src/@planx/components/Checklist/Public.tsx @@ -4,7 +4,7 @@ import type { Checklist, Group } from "@planx/components/Checklist/model"; import ImageButton from "@planx/components/shared/Buttons/ImageButton"; import Card from "@planx/components/shared/Preview/Card"; import QuestionHeader from "@planx/components/shared/Preview/QuestionHeader"; -import { useFormik } from "formik"; +import { getIn, useFormik } from "formik"; import React, { useState } from "react"; import ChecklistItem from "ui/ChecklistItem"; import ErrorWrapper from "ui/ErrorWrapper"; @@ -139,7 +139,7 @@ const ChecklistComponent: React.FC = ({ img={img} /> - + {options ? ( options.map((option) => diff --git a/editor.planx.uk/src/@planx/components/Send/Editor.tsx b/editor.planx.uk/src/@planx/components/Send/Editor.tsx index ef464f5931..3d29215891 100644 --- a/editor.planx.uk/src/@planx/components/Send/Editor.tsx +++ b/editor.planx.uk/src/@planx/components/Send/Editor.tsx @@ -2,7 +2,7 @@ import Warning from "@mui/icons-material/Warning"; import Box from "@mui/material/Box"; import Grid from "@mui/material/Grid"; import Typography from "@mui/material/Typography"; -import { useFormik } from "formik"; +import { getIn, useFormik } from "formik"; import React from "react"; import ChecklistItem from "ui/ChecklistItem"; import ErrorWrapper from "ui/ErrorWrapper"; @@ -89,7 +89,7 @@ const SendComponent: React.FC = (props) => { /> - + {options.map((option) => ( diff --git a/editor.planx.uk/src/pages/FlowEditor/lib/analyticsProvider.tsx b/editor.planx.uk/src/pages/FlowEditor/lib/analyticsProvider.tsx index 53567b2f63..c0338e85ad 100644 --- a/editor.planx.uk/src/pages/FlowEditor/lib/analyticsProvider.tsx +++ b/editor.planx.uk/src/pages/FlowEditor/lib/analyticsProvider.tsx @@ -48,6 +48,7 @@ const analyticsContext = createContext<{ backwardsNavigationType: BackwardsNavigationInitiatorType, ) => Promise; node: Store.node | null; + trackInputErrors: (error: string) => Promise; }>({ createAnalytics: () => Promise.resolve(), trackHelpClick: () => Promise.resolve(), @@ -55,6 +56,7 @@ const analyticsContext = createContext<{ trackFlowDirectionChange: () => Promise.resolve(), trackBackwardsNavigationByNodeId: () => Promise.resolve(), node: null, + trackInputErrors: () => Promise.resolve(), }); const { Provider } = analyticsContext; @@ -140,6 +142,7 @@ export const AnalyticsProvider: React.FC<{ children: React.ReactNode }> = ({ trackFlowDirectionChange, trackBackwardsNavigationByNodeId, node, + trackInputErrors, }} > {children} @@ -401,6 +404,30 @@ export const AnalyticsProvider: React.FC<{ children: React.ReactNode }> = ({ }; return nodeMetadata; } + + /** + * Capture user input errors caught by ErrorWrapper component + */ + async function trackInputErrors(error: string) { + if (shouldTrackAnalytics && lastAnalyticsLogId) { + await publicClient.mutate({ + mutation: gql` + mutation TrackInputErrors($id: bigint!, $error: jsonb) { + update_analytics_logs_by_pk( + pk_columns: { id: $id } + _append: { input_errors: $error } + ) { + id + } + } + `, + variables: { + id: lastAnalyticsLogId, + error, + }, + }); + } + } }; /** diff --git a/editor.planx.uk/src/ui/ErrorWrapper.tsx b/editor.planx.uk/src/ui/ErrorWrapper.tsx index d4307d01e7..7600899b2c 100644 --- a/editor.planx.uk/src/ui/ErrorWrapper.tsx +++ b/editor.planx.uk/src/ui/ErrorWrapper.tsx @@ -2,11 +2,12 @@ import Box, { BoxProps } from "@mui/material/Box"; import { styled } from "@mui/material/styles"; import Typography from "@mui/material/Typography"; import { ERROR_MESSAGE } from "@planx/components/shared/constants"; -import React, { ReactElement } from "react"; +import { useAnalyticsTracking } from "pages/FlowEditor/lib/analyticsProvider"; +import React, { ReactElement, useEffect } from "react"; import { FONT_WEIGHT_SEMI_BOLD } from "theme"; export interface Props { - error: string | string[] | undefined; + error?: string; children?: ReactElement; id?: string; } @@ -34,16 +35,21 @@ const ErrorText = styled(Typography)(({ theme }) => ({ fontWeight: FONT_WEIGHT_SEMI_BOLD, })); -export default function ErrorWrapper(props: Props): FCReturn { - const id = props.id ? `${ERROR_MESSAGE}-${props.id}` : undefined; +export default function ErrorWrapper({ id, error, children }: Props): FCReturn { + const inputId = id ? `${ERROR_MESSAGE}-${id}` : undefined; + const { trackInputErrors } = useAnalyticsTracking(); + + useEffect(() => { + error && trackInputErrors(error); + }, [error, trackInputErrors]); return ( // role="status" immediately announces the error to screenreaders without interrupting focus - - - {props?.error} + + + {error && error} - {props.children || null} + {children || null} ); }