Skip to content

Commit

Permalink
Merge pull request #530 from akaene/feature/483-unsaved-changes-modal
Browse files Browse the repository at this point in the history
483 - Unsaved changes modal
  • Loading branch information
blcham authored Jul 21, 2024
2 parents c5de424 + c00298d commit bec8842
Show file tree
Hide file tree
Showing 8 changed files with 125 additions and 22 deletions.
3 changes: 2 additions & 1 deletion public/locales/cs/translation.json
Original file line number Diff line number Diff line change
Expand Up @@ -101,7 +101,8 @@
"requiredFailureRate": "Požadovaná intenzita poruch",
"calculatedFailureRate": "Vypočtená intenzita poruch",
"operationalFailureRate": "Provozní intenzita poruch",
"manuallyDefinedFailureRate": "Manuálně definovaná intenzita poruch"
"manuallyDefinedFailureRate": "Manuálně definovaná intenzita poruch",
"unsavedChanges": "Máte neuložené změny. Chcete před pokračováním uložit změny?"
},
"appBar": {
"selectSystemPlaceholder": "Vyberte systém"
Expand Down
3 changes: 2 additions & 1 deletion public/locales/en/translation.json
Original file line number Diff line number Diff line change
Expand Up @@ -101,7 +101,8 @@
"requiredFailureRate": "Required failure rate",
"calculatedFailureRate": "Calculated failure rate",
"operationalFailureRate": "Operational failure rate",
"manuallyDefinedFailureRate": "Manually defined failure rate"
"manuallyDefinedFailureRate": "Manually defined failure rate",
"unsavedChanges": "You have unsaved changes. Do you want to save your changes before proceeding?"
},
"appBar": {
"selectSystemPlaceholder": "Select system"
Expand Down
7 changes: 6 additions & 1 deletion src/components/appBar/AppBar.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -53,6 +53,7 @@ const AppBar = ({ title, showBackButton = false, topPanelHeight }: Props) => {
const { appBarTitle, systemsList } = useAppBar();
const location = useLocation();
const isGlobalSystemSwitchDisabled = shouldSystemSwitchBeDisabled(location.pathname);
const { isModified, setShowUnsavedChangesDialog } = useAppBar();

const [anchorEl, setAnchorEl] = React.useState<null | HTMLElement>(null);
const [selectedSystem, setSelectedSystem] = useSelectedSystemSummaries();
Expand Down Expand Up @@ -84,7 +85,11 @@ const AppBar = ({ title, showBackButton = false, topPanelHeight }: Props) => {
};

const goBack = () => {
history(-1);
if (isModified) {
setShowUnsavedChangesDialog(true);
} else {
history(-1);
}
};

const menuId = "user-account-menu";
Expand Down
8 changes: 7 additions & 1 deletion src/components/editor/faultTree/canvas/EditorCanvas.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -26,6 +26,7 @@ import { useFaultTrees } from "@hooks/useFaultTrees";
import { calculateCutSets } from "@services/faultTreeService";
import { SnackbarType, useSnackbar } from "@hooks/useSnackbar";
import { useNavigate } from "react-router-dom";
import { useAppBar } from "@contexts/AppBarContext";

enum MOVE_NODE {
DRAGGING = 0,
Expand Down Expand Up @@ -92,6 +93,7 @@ const EditorCanvas = ({
const [updatedMinOperationalHours, setUpdatedMinOperationalHours] = useState(initialMinOperationalHours);
const [inputColor, setInputColor] = useState("");
const [showSnackbar] = useSnackbar();
const { isModified, setShowUnsavedChangesDialog } = useAppBar();

let dragStartPosition = null;

Expand Down Expand Up @@ -300,6 +302,10 @@ const EditorCanvas = ({
};

const handleSetNewDefaultOperationalHours = () => {
if (isModified) {
setShowUnsavedChangesDialog(true);
return;
}
const newOperationalDataFilter = Object.assign({}, faultTree.operationalDataFilter);
newOperationalDataFilter.minOperationalHours =
updatedMinOperationalHours || faultTree.operationalDataFilter.minOperationalHours;
Expand Down Expand Up @@ -347,7 +353,7 @@ const EditorCanvas = ({
</CurrentFaultTreeTableProvider>
{!showTable && (
<FaultEventMenu
shapeToolData={sidebarSelectedEvent}
selectedShapeToolData={sidebarSelectedEvent}
onEventUpdated={onEventUpdated}
refreshTree={refreshTree}
rootIri={rootEvent?.iri}
Expand Down
32 changes: 25 additions & 7 deletions src/components/editor/faultTree/menu/SidebarMenuHeader.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -3,27 +3,45 @@ import DiagramOptions, { Props as DiagramOptionsProps } from "../../menu/Diagram
import FaultTreeFailureModesTable from "./failureModesTable/FaultTreeFailureModesTable";
import { useCurrentFaultTreeTable } from "@hooks/useCurrentFaultTreeTable";
import { Box } from "@mui/material";
import { useAppBar } from "@contexts/AppBarContext";

const SidebarMenuHeader = ({
type SidebarMenuHeaderProps = DiagramOptionsProps;

const useActionCall = (isModified: boolean, setShowUnsavedChangesDialog: (show: boolean) => void) => {
return React.useCallback(
(action: () => void) => {
if (isModified) {
setShowUnsavedChangesDialog(true);
} else {
action();
}
},
[isModified, setShowUnsavedChangesDialog],
);
};

const SidebarMenuHeader: React.FC<SidebarMenuHeaderProps> = ({
onConvertToTable,
onExportDiagram,
onRestoreLayout,
onCutSetAnalysis,
rendering,
}: DiagramOptionsProps) => {
}) => {
const table = useCurrentFaultTreeTable();
const { isModified, setShowUnsavedChangesDialog } = useAppBar();

const actionCall = useActionCall(isModified, setShowUnsavedChangesDialog);

return (
<Box padding={2}>
<DiagramOptions
onExportDiagram={onExportDiagram}
onConvertToTable={onConvertToTable}
onRestoreLayout={onRestoreLayout}
onCutSetAnalysis={onCutSetAnalysis}
onExportDiagram={() => actionCall(onExportDiagram)}
onConvertToTable={() => actionCall(onConvertToTable)}
onRestoreLayout={() => actionCall(onRestoreLayout)}
onCutSetAnalysis={() => actionCall(onCutSetAnalysis)}
tableConversionAllowed={!table}
rendering={rendering}
/>

<FaultTreeFailureModesTable table={table} />
</Box>
);
Expand Down
42 changes: 32 additions & 10 deletions src/components/editor/faultTree/menu/faultEvent/FaultEventMenu.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -26,9 +26,11 @@ import { asArray } from "@utils/utils";
import { ReusableFaultEventsProvider } from "@hooks/useReusableFaultEvents";
import { useSelectedSystemSummaries } from "@hooks/useSelectedSystemSummaries";
import { useForm } from "react-hook-form";
import UnsavedChangesDialog from "./UnsavedChangesDialog";
import { useAppBar } from "@contexts/AppBarContext";

interface Props {
shapeToolData?: FaultEvent;
selectedShapeToolData?: FaultEvent;
onEventUpdated: (faultEvent: FaultEvent) => void;
refreshTree: () => void;
rootIri?: string;
Expand Down Expand Up @@ -57,16 +59,25 @@ const getFailureRateIris = (supertypes) => {
);
};

const FaultEventMenu = ({ shapeToolData, onEventUpdated, refreshTree, rootIri }: Props) => {
const FaultEventMenu = ({ selectedShapeToolData, onEventUpdated, refreshTree, rootIri }: Props) => {
const { t } = useTranslation();
const formMethods = useForm();
const { formState, getValues } = formMethods;
const { isDirty } = formState;
const { classes } = useStyles();
const { isModified, setIsModified, showUnsavedChangesDialog, setShowUnsavedChangesDialog } = useAppBar();
const theme = useTheme();
const [failureModeDialogOpen, setFailureModeDialogOpen] = useState(false);
const [resetMenu, setResetMenu] = useState<boolean>(false);
const [showSaveAndRejectButton, setShowSaveAndRejectButton] = useState<boolean>(false);
const [shapeToolData, setShapeToolData] = useState<FaultEvent | undefined>(undefined);

useEffect(() => {
if (isModified) {
setShowUnsavedChangesDialog(true);
} else {
setShapeToolData(selectedShapeToolData);
}
}, [selectedShapeToolData]);

const [failureModeOverviewDialogOpen, setFailureModeOverviewDialogOpen] = useState(false);
const [failureModeOverview, setFailureModeOverview] = useState<FailureMode | null>(null);
Expand All @@ -93,7 +104,7 @@ const FaultEventMenu = ({ shapeToolData, onEventUpdated, refreshTree, rootIri }:
const [snsPredictedIri, setSnsPredictedIri] = useState<string | undefined>(undefined);

const handleOnSave = async () => {
setShowSaveAndRejectButton(false);
setIsModified(false);
const values = getValues();
const { description, gateType } = values;
const updateEvent = async (data) => await onEventUpdated({ ...shapeToolData, ...data, gateType, description });
Expand Down Expand Up @@ -125,7 +136,7 @@ const FaultEventMenu = ({ shapeToolData, onEventUpdated, refreshTree, rootIri }:
const handleOnDiscard = () => {
formMethods.reset();
setResetMenu(!resetMenu);
setShowSaveAndRejectButton(false);
setIsModified(false);
};

const handleManuallyDefinedFailureRateChange = (event, type: NodeTypeWithManualFailureRate) => {
Expand All @@ -138,17 +149,17 @@ const FaultEventMenu = ({ shapeToolData, onEventUpdated, refreshTree, rootIri }:
if (type === NodeTypeWithManualFailureRate.External) {
setExternalManuallyDefinedFailureRate(inputValue);
}
setShowSaveAndRejectButton(true);
setIsModified(true);
}
};

const handleSnsBasicSelectedFailureRateChange = (event: React.ChangeEvent<HTMLInputElement>) => {
if (event.target.value === preselectedRadioButton) {
setSelectedRadioButton(event.target.value as RadioButtonType);
setShowSaveAndRejectButton(false);
setIsModified(false);
} else {
setSelectedRadioButton(event.target.value as RadioButtonType);
setShowSaveAndRejectButton(true);
setIsModified(true);
}
};

Expand All @@ -157,6 +168,12 @@ const FaultEventMenu = ({ shapeToolData, onEventUpdated, refreshTree, rootIri }:
setFailureModeOverviewDialogOpen(true);
};

const handleUnsavedChanges = (action) => {
action();
setShowUnsavedChangesDialog(false);
setShapeToolData(selectedShapeToolData);
};

useEffect(() => {
const setInitialStates = () => {
setSnsPredictedFailureRate(undefined);
Expand Down Expand Up @@ -263,7 +280,7 @@ const FaultEventMenu = ({ shapeToolData, onEventUpdated, refreshTree, rootIri }:
}, [shapeToolData, resetMenu]);

useEffect(() => {
setShowSaveAndRejectButton(isDirty);
setIsModified(isDirty);
}, [isDirty]);

const basedFailureRate = shapeToolData?.supertypes?.hasFailureRate?.estimate?.value;
Expand Down Expand Up @@ -443,7 +460,7 @@ const FaultEventMenu = ({ shapeToolData, onEventUpdated, refreshTree, rootIri }:
</Box>
)}

{showSaveAndRejectButton && (
{isModified && (
<Box display="flex" flexDirection="row">
<Button onClick={handleOnSave}>{t("common.save")}</Button>
<Button onClick={handleOnDiscard}>{t("common.discard")}</Button>
Expand Down Expand Up @@ -510,6 +527,11 @@ const FaultEventMenu = ({ shapeToolData, onEventUpdated, refreshTree, rootIri }:
/>
</EventFailureModeProvider>
)}
<UnsavedChangesDialog
isModalOpen={showUnsavedChangesDialog}
onDiscard={() => handleUnsavedChanges(handleOnDiscard)}
onSave={() => handleUnsavedChanges(handleOnSave)}
/>
</Box>
);
};
Expand Down
Original file line number Diff line number Diff line change
@@ -0,0 +1,28 @@
import React from "react";
import { Button, Dialog, DialogTitle, DialogActions } from "@mui/material";
import { useTranslation } from "react-i18next";

interface UnsavedChangesDialogProps {
isModalOpen: boolean;
onSave: () => void;
onDiscard: () => void;
}

const UnsavedChangesDialog: React.FC<UnsavedChangesDialogProps> = ({ isModalOpen, onSave, onDiscard }) => {
const { t } = useTranslation();
return (
<Dialog open={isModalOpen}>
<DialogTitle style={{ fontSize: 16 }}>{t("faultEventMenu.unsavedChanges")}</DialogTitle>
<DialogActions>
<Button onClick={onDiscard} color="primary">
{t("common.discard")}
</Button>
<Button onClick={onSave} color="primary">
{t("common.save")}
</Button>
</DialogActions>
</Dialog>
);
};

export default UnsavedChangesDialog;
24 changes: 23 additions & 1 deletion src/contexts/AppBarContext.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -14,6 +14,10 @@ interface AppBarTitleContextType {
systemsList: System[];
setSystemsList: React.Dispatch<React.SetStateAction<System[]>>;
addSystemToList: (system: System) => void;
isModified: boolean;
setIsModified: React.Dispatch<React.SetStateAction<boolean>>;
showUnsavedChangesDialog: boolean;
setShowUnsavedChangesDialog: React.Dispatch<React.SetStateAction<boolean>>;
}

interface AppBarTitleProviderProps {
Expand All @@ -33,13 +37,19 @@ const AppBarMainContext = createContext<AppBarTitleContextType>({
systemsList: [],
setSystemsList: () => {},
addSystemToList: (system: System) => {},
isModified: false,
setIsModified: () => {},
showUnsavedChangesDialog: false,
setShowUnsavedChangesDialog: () => {},
});

export const useAppBar = () => useContext(AppBarMainContext);

export const AppBarProvider = ({ children }: AppBarTitleProviderProps) => {
const [appBarTitle, setAppBarTitle] = useState<string>("");
const [systemsList, setSystemsList] = useState<System[]>([]);
const [isModified, setIsModified] = useState<boolean>(false);
const [showUnsavedChangesDialog, setShowUnsavedChangesDialog] = useState<boolean>(false);
const location = useLocation();
const { t } = useTranslation();
const [showSnackbar] = useSnackbar();
Expand Down Expand Up @@ -83,7 +93,19 @@ export const AppBarProvider = ({ children }: AppBarTitleProviderProps) => {
};

return (
<AppBarMainContext.Provider value={{ appBarTitle, setAppBarTitle, systemsList, setSystemsList, addSystemToList }}>
<AppBarMainContext.Provider
value={{
appBarTitle,
setAppBarTitle,
systemsList,
setSystemsList,
addSystemToList,
isModified,
setIsModified,
showUnsavedChangesDialog,
setShowUnsavedChangesDialog,
}}
>
{children}
</AppBarMainContext.Provider>
);
Expand Down

0 comments on commit bec8842

Please sign in to comment.