From 2c8eaf326ae68414f67a6d606b568d917d152eed Mon Sep 17 00:00:00 2001 From: Jo Humphrey <31373245+jamdelion@users.noreply.github.com> Date: Tue, 12 Nov 2024 15:58:40 +0000 Subject: [PATCH] feat: can toggle feedback required and show validation errors (#3939) --- .../Feedback/Editor/Editor.test.tsx | 9 +- .../Feedback/Public/Feedback.stories.tsx | 4 + .../Feedback/Public/Public.test.tsx | 45 -------- .../components/Feedback/Public/Public.tsx | 102 ++++++++++-------- .../tests/Public.accessibility.test.tsx | 56 ++++++++++ .../Public/tests/Public.submit.test.tsx | 97 +++++++++++++++++ .../src/@planx/components/Feedback/model.ts | 14 +++ 7 files changed, 235 insertions(+), 92 deletions(-) delete mode 100644 editor.planx.uk/src/@planx/components/Feedback/Public/Public.test.tsx create mode 100644 editor.planx.uk/src/@planx/components/Feedback/Public/tests/Public.accessibility.test.tsx create mode 100644 editor.planx.uk/src/@planx/components/Feedback/Public/tests/Public.submit.test.tsx diff --git a/editor.planx.uk/src/@planx/components/Feedback/Editor/Editor.test.tsx b/editor.planx.uk/src/@planx/components/Feedback/Editor/Editor.test.tsx index e005ed1193..dfad5c904e 100644 --- a/editor.planx.uk/src/@planx/components/Feedback/Editor/Editor.test.tsx +++ b/editor.planx.uk/src/@planx/components/Feedback/Editor/Editor.test.tsx @@ -7,12 +7,19 @@ import { setup } from "testUtils"; import { FeedbackEditor } from "./Editor"; describe("When the Feedback editor modal is rendered", () => { - it("does not throw an error", () => { + beforeEach(() => { setup( , ); + }); + it("does not throw an error", () => { expect(screen.getByText("Feedback")).toBeInTheDocument(); }); + it("displays the default title if no edits are made", () => { + expect( + screen.getByPlaceholderText("Tell us what you think"), + ).toBeInTheDocument(); + }); }); diff --git a/editor.planx.uk/src/@planx/components/Feedback/Public/Feedback.stories.tsx b/editor.planx.uk/src/@planx/components/Feedback/Public/Feedback.stories.tsx index e8f0b3d8bb..2bc537e27f 100644 --- a/editor.planx.uk/src/@planx/components/Feedback/Public/Feedback.stories.tsx +++ b/editor.planx.uk/src/@planx/components/Feedback/Public/Feedback.stories.tsx @@ -15,3 +15,7 @@ export default meta; export const Basic = { args: defaultContent, } satisfies Story; + +export const WithFeedbackRequired = { + args: { ...defaultContent, feedbackRequired: true }, +} satisfies Story; diff --git a/editor.planx.uk/src/@planx/components/Feedback/Public/Public.test.tsx b/editor.planx.uk/src/@planx/components/Feedback/Public/Public.test.tsx deleted file mode 100644 index 4f50a17f86..0000000000 --- a/editor.planx.uk/src/@planx/components/Feedback/Public/Public.test.tsx +++ /dev/null @@ -1,45 +0,0 @@ -import { screen } from "@testing-library/react"; -import { - getInternalFeedbackMetadata, - insertFeedbackMutation, -} from "lib/feedback"; -import React from "react"; -import { setup } from "testUtils"; -import { vi } from "vitest"; -import { axe } from "vitest-axe"; - -import FeedbackComponent from "./Public"; - -const handleSubmit = vi.fn(); -vi.mock("lib/feedback", () => ({ - getInternalFeedbackMetadata: vi.fn(), - insertFeedbackMutation: vi.fn(), -})); - -describe("when the Feedback component is rendered", async () => { - it("should not have any accessibility violations", async () => { - const { container } = setup( - , - ); - const results = await axe(container); - expect(results).toHaveNoViolations(); - }); - it("should call handleSubmit when the continue button is pressed", async () => { - const { user } = setup( - , - ); - - await user.click(screen.getByTestId("feedback-button-terrible")); - await user.click(screen.getByTestId("continue-button")); - - expect(getInternalFeedbackMetadata).toBeCalled(); - expect(insertFeedbackMutation).toBeCalled(); - }); -}); diff --git a/editor.planx.uk/src/@planx/components/Feedback/Public/Public.tsx b/editor.planx.uk/src/@planx/components/Feedback/Public/Public.tsx index f5d64d39d7..975455b3c7 100644 --- a/editor.planx.uk/src/@planx/components/Feedback/Public/Public.tsx +++ b/editor.planx.uk/src/@planx/components/Feedback/Public/Public.tsx @@ -17,17 +17,20 @@ import NeutralFace from "ui/images/feedback_filled-03.svg"; import GoodFace from "ui/images/feedback_filled-04.svg"; import ExcellentFace from "ui/images/feedback_filled-05.svg"; import InputLabel from "ui/public/InputLabel"; +import ErrorWrapper from "ui/shared/ErrorWrapper"; import Input from "ui/shared/Input/Input"; import ReactMarkdownOrHtml from "ui/shared/ReactMarkdownOrHtml/ReactMarkdownOrHtml"; import { getPreviouslySubmittedData, makeData } from "../../shared/utils"; import { FaceBox } from "../components/FaceBox"; -import { Feedback, FormProps } from "../model"; +import { createFeedbackSchema, Feedback, FormProps } from "../model"; import { StyledToggleButtonGroup } from "../styled"; export const PASSPORT_FEEDBACK_KEY = "_feedback"; const FeedbackComponent = (props: PublicProps): FCReturn => { + const feedbackDataSchema = createFeedbackSchema(props.feedbackRequired); + const handleSubmitFeedback = async (values: FormProps) => { const metadata = await getInternalFeedbackMetadata(); const data = { @@ -52,6 +55,9 @@ const FeedbackComponent = (props: PublicProps): FCReturn => { userComment: "", }, onSubmit: handleSubmitFeedback, + validateOnBlur: false, + validateOnChange: false, + validationSchema: feedbackDataSchema, }); const handleFeedbackChange = ( @@ -88,52 +94,54 @@ const FeedbackComponent = (props: PublicProps): FCReturn => { } /> )} - - + - - - - - - - + + + + + + + + + {props.freeformQuestion && ( ): FCReturn => { bordered onChange={formik.handleChange} aria-label="user comment" + data-testid="user-comment" + errorMessage={formik.errors.userComment} /> {props.disclaimer && } diff --git a/editor.planx.uk/src/@planx/components/Feedback/Public/tests/Public.accessibility.test.tsx b/editor.planx.uk/src/@planx/components/Feedback/Public/tests/Public.accessibility.test.tsx new file mode 100644 index 0000000000..1501cc1f5e --- /dev/null +++ b/editor.planx.uk/src/@planx/components/Feedback/Public/tests/Public.accessibility.test.tsx @@ -0,0 +1,56 @@ +import { screen } from "@testing-library/react"; +import { + getInternalFeedbackMetadata, + insertFeedbackMutation, +} from "lib/feedback"; +import React from "react"; +import { setup } from "testUtils"; +import { vi } from "vitest"; +import { axe } from "vitest-axe"; + +import FeedbackComponent from "../Public"; + +vi.mock("lib/feedback", () => ({ + getInternalFeedbackMetadata: vi.fn(), + insertFeedbackMutation: vi.fn(), +})); + +describe("when the Feedback component is rendered", async () => { + it("should not have any accessibility violations", async () => { + const { container } = setup(); + const results = await axe(container); + expect(results).toHaveNoViolations(); + }); + it("should be navigable by keyboard", async () => { + const { user } = setup(); + + const ratingButtons = screen.getAllByRole("button"); + + // user tabs through all rating buttons and selects the last one + await user.tab(); + expect(ratingButtons[0]).toHaveFocus(); + await user.tab(); + expect(ratingButtons[1]).toHaveFocus(); + await user.tab(); + expect(ratingButtons[2]).toHaveFocus(); + await user.tab(); + expect(ratingButtons[3]).toHaveFocus(); + await user.tab(); + expect(ratingButtons[4]).toHaveFocus(); + await user.keyboard("[Space]"); // select 'Excellent' button + + await user.tab(); + expect(screen.getByRole("textbox")).toHaveFocus(); + + await user.tab(); + expect(screen.getByTestId("continue-button")).toHaveFocus(); + await user.keyboard("[Space]"); // submits + + expect(getInternalFeedbackMetadata).toBeCalled(); + expect(insertFeedbackMutation).toBeCalledWith({ + feedbackScore: 5, + feedbackType: "component", + userComment: "", + }); + }); +}); diff --git a/editor.planx.uk/src/@planx/components/Feedback/Public/tests/Public.submit.test.tsx b/editor.planx.uk/src/@planx/components/Feedback/Public/tests/Public.submit.test.tsx new file mode 100644 index 0000000000..abbf2b084f --- /dev/null +++ b/editor.planx.uk/src/@planx/components/Feedback/Public/tests/Public.submit.test.tsx @@ -0,0 +1,97 @@ +import { screen } from "@testing-library/react"; +import { + getInternalFeedbackMetadata, + insertFeedbackMutation, +} from "lib/feedback"; +import React from "react"; +import { setup } from "testUtils"; +import { vi } from "vitest"; + +import FeedbackComponent from "../Public"; + +const handleSubmit = vi.fn(); +vi.mock("lib/feedback", () => ({ + getInternalFeedbackMetadata: vi.fn(), + insertFeedbackMutation: vi.fn(), +})); + +describe.each([ + { + dataType: "a rating", + expectedData: { + feedbackScore: 1, + feedbackType: "component", + userComment: "", + }, + }, + { + dataType: "a comment", + expectedData: { + feedbackScore: "", + feedbackType: "component", + userComment: "I had a great time", + }, + }, + { + dataType: "nothing", + expectedData: { + feedbackScore: "", + feedbackType: "component", + userComment: "", + }, + }, +])( + "when a user submits $dataType on a feedback component where feedback is not required", + ({ dataType, expectedData }) => { + beforeEach(async () => { + const { user } = setup( + , + ); + + switch (dataType) { + case "a rating": + await user.click(screen.getByTestId("feedback-button-terrible")); + + break; + case "a comment": + await user.type( + screen.getByTestId("user-comment"), + "I had a great time", + ); + break; + case "nothing": + default: + break; + } + + await user.click(screen.getByTestId("continue-button")); + }); + + it("should call the handleSubmit function with the correct data", async () => { + expect(getInternalFeedbackMetadata).toBeCalled(); + expect(insertFeedbackMutation).toBeCalledWith(expectedData); + }); + }, +); + +describe("when feedback is required but the user does not submit any data", async () => { + beforeEach(async () => { + const { user } = setup( + , + ); + await user.click(screen.getByTestId("continue-button")); + }); + + it("displays an appropriate error message for each missing field", async () => { + const errorMessages = [ + "Please rate your experience", + "Enter your feedback", + ]; + errorMessages.map((error) => { + expect(screen.getByText(error)).toBeVisible(); + }); + }); +}); diff --git a/editor.planx.uk/src/@planx/components/Feedback/model.ts b/editor.planx.uk/src/@planx/components/Feedback/model.ts index ee1b1cd94f..daeaa81124 100644 --- a/editor.planx.uk/src/@planx/components/Feedback/model.ts +++ b/editor.planx.uk/src/@planx/components/Feedback/model.ts @@ -1,5 +1,8 @@ +import { number, object, string } from "yup"; + import { BaseNodeData, parseBaseNodeData } from "../shared"; import { defaultContent } from "./components/defaultContent"; + export interface Feedback extends BaseNodeData { title?: string; description?: string; @@ -24,3 +27,14 @@ export const parseFeedback = ( feedbackRequired: data?.feedbackRequired || defaultContent.feedbackRequired, ...parseBaseNodeData(data), }); + +export const createFeedbackSchema = (feedbackRequired: boolean) => { + return object().shape({ + userComment: feedbackRequired + ? string().required("Enter your feedback") + : string(), + feedbackScore: feedbackRequired + ? number().integer().required("Please rate your experience") + : number().integer(), + }); +};