diff --git a/frontend/src/components/EditableText/EditableText.tsx b/frontend/src/components/EditableText/EditableText.tsx index 93fc4971..18d9b37a 100644 --- a/frontend/src/components/EditableText/EditableText.tsx +++ b/frontend/src/components/EditableText/EditableText.tsx @@ -23,6 +23,10 @@ export const EditableText: FC = ({ }, onSave = () => { }, + onEdit = () => { + }, + onCancel = () => { + }, textFormatter = (text) => text }: EditableTextProps) => { const [isEditing, setIsEditing] = useState(false) @@ -45,7 +49,7 @@ export const EditableText: FC = ({ const _onEdit = () => { setIsEditing(true) - // onEdit() + onEdit() } const _onSave = () => { @@ -56,6 +60,7 @@ export const EditableText: FC = ({ const _onCancel = () => { setValue(text) setIsEditing(false) + onCancel() } const _onKeyDown = (e: React.KeyboardEvent) => { if (e.key === 'Enter') { diff --git a/frontend/src/pages/ReviewTemplate.tsx b/frontend/src/pages/ReviewTemplate.tsx index 890293d8..7d1999a9 100644 --- a/frontend/src/pages/ReviewTemplate.tsx +++ b/frontend/src/pages/ReviewTemplate.tsx @@ -1,11 +1,10 @@ -import { ExtractStep } from "../utils/constants"; -import { useNavigate } from "react-router-dom"; -import { useEffect, useState } from "react"; +import {ExtractStep} from "../utils/constants"; +import {useNavigate} from "react-router-dom"; +import React, {useEffect, useState} from "react"; import ExtractDataHeader from "../components/ExtractDataHeader"; -import React from "react"; -import { ExtractStepper } from "../components/ExtractStepper"; -import { Table, Icon, Button } from "@trussworks/react-uswds"; -import { Divider } from "../components/Divider"; +import {ExtractStepper} from "../components/ExtractStepper"; +import {Button, Icon} from "@trussworks/react-uswds"; +import {Divider} from "../components/Divider"; import documentImage from "./SyphForm.png"; //Please enter your file of choice here import "./ReviewTemplate.scss"; import aiIconUrl from "../assets/ai_icon.svg"; @@ -13,238 +12,268 @@ import {SortableTable} from "../components/SortableTable/SortableTable.tsx"; import {EditableText} from "../components/EditableText/EditableText.tsx"; interface Result { - text: string; - confidence: number; - edited?: boolean; + text: string; + confidence: number; + edited?: boolean; } interface SubmissionData { - template_name: string; - file_image: string; - results: { - [key: string]: Result; - }; + template_name: string; + file_image: string; + results: { + [key: string]: Result; + }; } const ReviewTemplate: React.FC = () => { - const navigate = useNavigate(); - const [index, setIndex] = useState(0); - const [submissionData, setSubmissionData] = useState( - null - ); + const navigate = useNavigate(); + const [index, setIndex] = useState(0); + const [submissionData, setSubmissionData] = useState( + null + ); - const [images, setImages] = useState([]); + const [images, setImages] = useState([]); + const [templates, setTemplates] = useState([]); + const [selectedField, setSelectedField] = useState(null); - const fakeData = { - template_name: "Syph Template", - file_image: documentImage, - results: { - Name: { - text: "TESTPATIENT,13", - confidence: 97, - }, - Patient_ID: { - text: "12090546", - confidence: 98, - }, - DrawLocation: { - text: "BH_1Diamondd_LAB", - confidence: 56, - }, - }, - }; - - const handleImageChange = (index: number) => { - setIndex(index); - } - - useEffect(() => { - const data = localStorage.getItem("submission"); - if (data) { - setSubmissionData(JSON.parse(data)); - } else { - // use fakedata if no data - setSubmissionData(fakeData); + const fakeData = { + template_name: "Test", + template_image: "", + file_name: "image name", + file_image: "", + results: { + Name: { + text: "TESTPATIENT,13", + confidence: 97, + }, + Patient_ID: { + text: "12090546", + confidence: 98, + }, + DrawLocation: { + text: "BH_1Diamondd_LAB", + confidence: 56, + }, + }, + }; + + + const handleImageChange = (index: number) => { + setIndex(index); } - }, []); - const [editedValues, setEditedValues] = useState>(new Map()); + useEffect(() => { + const data = localStorage.getItem("submission"); + if (data) { + setSubmissionData(JSON.parse(data)); + } else { + // use fakedata if no data + setSubmissionData(fakeData); + } + }, []); + + const [editedValues, setEditedValues] = useState>(new Map()); + + useEffect(() => { + const data = localStorage.getItem("images"); + if (data) { + setImages(JSON.parse(data)); + } else { + // use fakedata if no data + setImages([documentImage]); + } + }, []); + + const handleBack = () => { + navigate("/extract/upload"); + }; + + const handleSubmit = () => { + navigate("/extract/submit"); + }; + + const handleClose = () => { + navigate("/"); + }; + + useEffect(() => { + const data = localStorage.getItem("templates"); + if (data) { + setTemplates(JSON.parse(data)); + } + }, []); - useEffect(() => { - const data = localStorage.getItem("images"); - if (data) { - setImages(JSON.parse(data)); - } else { - // use fakedata if no data - setImages([documentImage]); + + const template = templates.filter(t => t.name === submissionData?.template_name)?.[0] + const fieldShapes = template?.pages[index]?.fieldNames.reduce((acc, curr, idx) => { + acc[curr.label] = template.pages[index].shapes[idx].points + return acc + }, {}) || {}; + + + const maskShape = fieldShapes[selectedField]?.map(([x, y]) => `${x},${y}`)?.join(" "); + + + //fallback if no valid template is available can edit if needed + if (!submissionData) { + return
No submission Data
; } - }, []); - - const handleBack = () => { - navigate("/extract/upload"); - }; - - const handleSubmit = () => { - navigate("/extract/submit"); - }; - - const handleClose = () => { - navigate("/"); - }; - - - - //fallback if no valid template is available can edit if needed - if (!submissionData) { - return
No submission Data
; - } - const { file_image, results } = submissionData; - const confidenceVal = 75; - - const resultsTable = Object.entries(results).map(([k, v], idx) => { - const overrideValue = editedValues[k] - return { - index: idx, - name: k, - value: overrideValue || v.text, - confidence: overrideValue ? 100 : v.confidence, - isError: overrideValue ? false : v.confidence <= confidenceVal, - isEdited: !!overrideValue + const {file_image, results} = submissionData; + const confidenceVal = 75; + + const resultsTable = Object.entries(results).map(([k, v], idx) => { + const overrideValue = editedValues[k] + return { + index: idx, + name: k, + value: overrideValue || v.text, + confidence: overrideValue ? 100 : v.confidence, + isError: overrideValue ? false : v.confidence <= confidenceVal, + isEdited: !!overrideValue + } + }); + + const calculateOverallConfidence = () => { + if (!submissionData) return 0; + const results = resultsTable; + const totalConfidence = results.reduce((sum, result) => { + return sum + result.confidence; + }, 0); + return (totalConfidence / results.length).toFixed(2); + }; + + const errorCount = resultsTable.filter( + (r) => r.isError + ).length; + const hasErrors = errorCount > 0; + const overallConfidence = calculateOverallConfidence(); + + + const isErrorFormatter = (d) => { + return d ? : ""; + } + + const labelConfidenceFormatter = (d, _idx, row) => { + return row.isEdited ? "Edited" : redTextOnErrorFormatter(`${d.toFixed(0)}%`, row.index, row) + } + + const redTextOnErrorFormatter = (d, _idx, row) => { + return row.isError ? {d} : d; } - }); - - const calculateOverallConfidence = () => { - if (!submissionData) return 0; - const results = resultsTable; - const totalConfidence = results.reduce((sum, result) => { - return sum + result.confidence; - }, 0); - return (totalConfidence / results.length).toFixed(2); - }; - - const errorCount = resultsTable.filter( - (r) => r.isError - ).length; - const hasErrors = errorCount > 0; - const overallConfidence = calculateOverallConfidence(); - - - - const isErrorFormatter = (d) => { - return d ? : ""; - } - - const labelConfidenceFormatter = (d, _idx, row) => { - return row.isEdited ? "Edited" : redTextOnErrorFormatter(`${d.toFixed(0)}%`, row.index, row) - } - - const redTextOnErrorFormatter = (d, _idx, row) => { - return row.isError ? {d} : d; - } - - const editCellFormatter = (d, _idx, row) => { - return { - setEditedValues({...editedValues, [row.name]: value}) - }} textFormatter={(s) => row.isError ? {s} : s}/> - } - - return ( -
- - -
- -
- -
-
-
-
-

Extracted Data

+ + const editCellFormatter = (d, _idx, row) => { + return { + setEditedValues({...editedValues, [row.name]: value}) + setSelectedField(null) + }} textFormatter={(s) => row.isError ? {s} : s} + onEdit={() => setSelectedField(row.name)} + onCancel={() => setSelectedField(null)} + /> + } + + return ( +
+ + +
+
-
- AI Icon - + +
+
+
+
+

Extracted Data

+
+
+ AI Icon + Overall confidence score (CS): {overallConfidence}% -
- - +
+ + Overall Confidence Score is the average of all Individual Confidence Scores. -
-
-

- Review and edit errors before you submit. -

-
- {hasErrors && ( - <> +
+
+

+ Review and edit errors before you submit. +

+
+ {hasErrors && ( + <> Errors: {errorCount} - - - )} -
-
- - + + + )} +
+
-
-
-
-
- {images.map((_, index) => ( - - ))} + + +
+
+
+
+ {images.map((_, index) => ( + + ))} +
+ { + images.map((image, innerIndex) => ( + + + + + + + + )) + } +
+
- { - images.map((image, innerIndex) => ( - {`Document - )) - } -
-
-
- ); + ); }; export default ReviewTemplate; \ No newline at end of file