Skip to content

Commit

Permalink
new files
Browse files Browse the repository at this point in the history
  • Loading branch information
Bob Zhao committed Oct 11, 2024
1 parent 9a4df85 commit 0158544
Show file tree
Hide file tree
Showing 3 changed files with 350 additions and 0 deletions.
110 changes: 110 additions & 0 deletions containers/tefca-viewer/src/app/query/components/SelectQuery.tsx
Original file line number Diff line number Diff line change
@@ -0,0 +1,110 @@
"use client";
import React, { useEffect, useState } from "react";
import { FHIR_SERVERS, USE_CASES, ValueSetItem } from "../../constants";
import CustomizeQuery from "./CustomizeQuery";
import SelectSavedQuery from "./selectQuery/SelectSavedQuery";

import { QueryResponse } from "@/app/query-service";
import { Patient } from "fhir/r4";
import {
fetchQueryResponse,
fetchUseCaseValueSets,
} from "./selectQuery/queryHooks";

interface SelectQueryProps {
onSubmit: () => void; // Callback when the user submits the query
goBack: () => void;
selectedQuery: USE_CASES;
setSelectedQuery: React.Dispatch<React.SetStateAction<USE_CASES>>;
patientForQuery: Patient | undefined;
resultsQueryResponse: QueryResponse;
setResultsQueryResponse: React.Dispatch<React.SetStateAction<QueryResponse>>;
fhirServer: FHIR_SERVERS;
setFhirServer: React.Dispatch<React.SetStateAction<FHIR_SERVERS>>;
}

/**
* @param root0 - SelectQueryProps
* @param root0.onSubmit - Callback for submit action
* @param root0.goBack - Callback to return to previous page
* @param root0.selectedQuery - query we chose for further customization
* @param root0.setSelectedQuery - callback function to update the selected query
* @param root0.patientForQuery - patient to apply a particular query for
* @param root0.resultsQueryResponse - Response of selected query
* @param root0.setResultsQueryResponse - Callback function to update selected
* query
* @param root0.fhirServer - the FHIR server that we're running the query against
* @param root0.setFhirServer - callback function to update the FHIR server
* @returns - The selectQuery component.
*/
const SelectQuery: React.FC<SelectQueryProps> = ({
onSubmit,
goBack,
selectedQuery,
setSelectedQuery,
patientForQuery,
resultsQueryResponse,
setResultsQueryResponse,
fhirServer,
setFhirServer,
}) => {
const [showCustomizeQuery, setShowCustomizedQuery] = useState(false);
const [queryValueSets, setQueryValueSets] = useState<ValueSetItem[]>(
[] as ValueSetItem[],
);
useEffect(() => {
// Gate whether we actually update state after fetching so we
// avoid name-change race conditions
let isSubscribed = true;

fetchUseCaseValueSets(selectedQuery, setQueryValueSets, isSubscribed).catch(
console.error,
);

// Destructor hook to prevent future state updates
return () => {
isSubscribed = false;
};
}, [selectedQuery, setQueryValueSets]);

useEffect(() => {
let isSubscribed = true;

fetchQueryResponse(
patientForQuery,
selectedQuery,
isSubscribed,
queryValueSets,
setResultsQueryResponse,
fhirServer,
).catch(console.error);

// Destructor hook to prevent future state updates
return () => {
isSubscribed = false;
};
}, [patientForQuery, selectedQuery, queryValueSets, setResultsQueryResponse]);

return showCustomizeQuery ? (
<CustomizeQuery
useCaseQueryResponse={resultsQueryResponse}
queryType={selectedQuery}
queryValuesets={queryValueSets}
setQueryValuesets={setQueryValueSets}
goBack={() => setShowCustomizedQuery(false)}
></CustomizeQuery>
) : (
<SelectSavedQuery
goBack={goBack}
selectedQuery={selectedQuery}
setSelectedQuery={setSelectedQuery}
setShowCustomizedQuery={setShowCustomizedQuery}
handleSubmit={onSubmit}
fhirServer={fhirServer}
setFhirServer={setFhirServer}
></SelectSavedQuery>
);
};

export default SelectQuery;
export const RETURN_TO_STEP_ONE_LABEL = "Return to Select patient";
Original file line number Diff line number Diff line change
@@ -0,0 +1,139 @@
import {
FHIR_SERVERS,
FhirServers,
USE_CASES,
demoQueryOptions,
} from "@/app/constants";
import { Select, Button } from "@trussworks/react-uswds";
import { RETURN_TO_STEP_ONE_LABEL } from "../PatientSearchResults";
import Backlink from "../backLink/Backlink";
import styles from "./selectQuery.module.css";
import { useState } from "react";

type SelectSavedQueryProps = {
goBack: () => void;
setSelectedQuery: (selectedQuery: USE_CASES) => void;
selectedQuery: string;
setShowCustomizedQuery: (showCustomize: boolean) => void;
handleSubmit: () => void;
fhirServer: FHIR_SERVERS;
setFhirServer: React.Dispatch<React.SetStateAction<FHIR_SERVERS>>;
};

/**
* Component to poulate pre-existing queries
* @param root0 - params
* @param root0.goBack - return function for the previous page
* @param root0.selectedQuery - specified query for future customization /
* application
* @param root0.setSelectedQuery - callback function for specified query
* @param root0.setShowCustomizedQuery - toggle to switch to customization
* view
* @param root0.handleSubmit - submit handler
* @param root0.fhirServer - fhir server to apply a query against
* @param root0.setFhirServer - function to update the fhir server
* @returns SelectedSavedQuery component
*/
const SelectSavedQuery: React.FC<SelectSavedQueryProps> = ({
goBack,
selectedQuery,
setSelectedQuery,
setShowCustomizedQuery,
handleSubmit,
fhirServer,
setFhirServer,
}) => {
const [showAdvanced, setShowAdvanced] = useState(false);

return (
<form className="content-container-smaller-width">
{/* Back button */}
<div className="text-bold">
<Backlink onClick={goBack} label={RETURN_TO_STEP_ONE_LABEL} />
</div>
<h1 className={`${styles.selectQueryHeaderText}`}>
Step 3: Select a query
</h1>
<div
className={`font-sans-md text-light ${styles.selectQueryExplanationText}`}
>
We will request all data related to your selected patient and query. By
only showing relevant data for your query, we decrease the burden on our
systems and protect patient privacy. If you would like to customize the
query response, click on the "customize query" button.
</div>
<h3 className="margin-bottom-3">Query</h3>
<div className={styles.queryRow}>
{/* Select a query drop down */}
<Select
id="querySelect"
name="query"
value={selectedQuery}
onChange={(e) => setSelectedQuery(e.target.value as USE_CASES)}
className={`${styles.queryDropDown}`}
required
>
{demoQueryOptions.map((option) => (
<option key={option.value} value={option.value}>
{option.label}
</option>
))}
</Select>
<Button
type="button"
className={`usa-button--outline bg-white ${styles.customizeButton}`}
onClick={() => setShowCustomizedQuery(true)}
>
Customize query
</Button>
</div>

{showAdvanced && (
<div>
<h3 className="margin-bottom-3">Health Care Organization (HCO)</h3>
<Select
id="fhir_server"
name="fhir_server"
value={fhirServer}
onChange={(e) => setFhirServer(e.target.value as FHIR_SERVERS)}
required
className={`${styles.queryDropDown}`}
>
Select HCO
{FhirServers.map((fhirServer: string) => (
<option key={fhirServer} value={fhirServer}>
{fhirServer}
</option>
))}
</Select>
</div>
)}

{!showAdvanced && (
<div>
<Button
className={`usa-button--unstyled margin-left-auto ${styles.searchCallToActionButton}`}
type="button"
onClick={() => setShowAdvanced(true)}
>
Advanced...
</Button>
</div>
)}

{/* Submit Button */}
<div className="padding-top-6">
<Button
type="button"
disabled={!selectedQuery}
className={selectedQuery ? "usa-button" : "usa-button disabled"}
onClick={() => handleSubmit()}
>
Submit
</Button>
</div>
</form>
);
};

export default SelectSavedQuery;
Original file line number Diff line number Diff line change
@@ -0,0 +1,101 @@
import {
FHIR_SERVERS,
USE_CASES,
UseCaseToQueryName,
ValueSetItem,
} from "@/app/constants";
import {
getSavedQueryByName,
mapQueryRowsToValueSetItems,
} from "@/app/database-service";
import { UseCaseQuery, UseCaseQueryResponse } from "@/app/query-service";
import { Patient } from "fhir/r4";

type SetStateCallback<T> = React.Dispatch<React.SetStateAction<T>>;

/**
* Query to grab valuesets based on use case
* @param selectedQuery - Query use case that's been selected
* @param valueSetStateCallback - state update function to set the valuesets
* @param isSubscribed - state destructor hook to prevent race conditions
*/
export async function fetchUseCaseValueSets(
selectedQuery: USE_CASES,
valueSetStateCallback: SetStateCallback<ValueSetItem[]>,
isSubscribed: boolean,
) {
if (selectedQuery) {
const queryName = UseCaseToQueryName[selectedQuery as USE_CASES];
const queryResults = await getSavedQueryByName(queryName);
const vsItems = await mapQueryRowsToValueSetItems(queryResults);

// Only update if the fetch hasn't altered state yet
if (isSubscribed) {
valueSetStateCallback(vsItems);
}
}
}

/**
* Query to apply for future view
* @param patientForQuery - patient to do query against
* @param selectedQuery - query use case
* @param isSubscribed - state destructor hook to prevent race conditions
* @param queryValueSets - valuesets to filter query from default usecase
* @param queryResponseStateCallback - callback function to update state of the
* query response
* @param fhirServer - fhir server to do the querying against
*/
export async function fetchQueryResponse(
patientForQuery: Patient | undefined,
selectedQuery: USE_CASES,
isSubscribed: boolean,
queryValueSets: ValueSetItem[],
queryResponseStateCallback: SetStateCallback<UseCaseQueryResponse>,
fhirServer: FHIR_SERVERS,
) {
if (patientForQuery && selectedQuery && isSubscribed) {
const patientFirstName =
getNthElementIfDefined(patientForQuery.name, -1)?.given?.join(" ") ??
"Hyper";
const patientLastName =
getNthElementIfDefined(patientForQuery.name, -1)?.family ?? "Unlucky";

const patientMRN =
getNthElementIfDefined(patientForQuery.identifier)?.value ??
HYPER_UNLUCKY_MRN;

const newRequest = {
first_name: patientFirstName as string,
last_name: patientLastName as string,
dob: patientForQuery.birthDate as string,
mrn: patientMRN,
fhir_server: fhirServer,
use_case: selectedQuery,
};

const queryResponse = await UseCaseQuery(
newRequest,
queryValueSets.filter((item) => item.include),
{
Patient: [patientForQuery],
},
);

queryResponseStateCallback(queryResponse);
}
}

function getNthElementIfDefined<T>(
arr: T[] | undefined,
n: number = 0,
): T | undefined {
if (arr && arr.length > n) {
const positionToCheck = n === -1 ? arr.length - 1 : n;
return arr[positionToCheck];
} else {
return undefined;
}
}

const HYPER_UNLUCKY_MRN = "8692756";

0 comments on commit 0158544

Please sign in to comment.