feature/IO-3255-simplified-parts-management - centralize alerts
This commit is contained in:
@@ -6,24 +6,15 @@ import { connect } from "react-redux";
|
|||||||
import { Link } from "react-router-dom";
|
import { Link } from "react-router-dom";
|
||||||
import { createStructuredSelector } from "reselect";
|
import { createStructuredSelector } from "reselect";
|
||||||
import WssStatusDisplayComponent from "../../components/wss-status-display/wss-status-display.component.jsx";
|
import WssStatusDisplayComponent from "../../components/wss-status-display/wss-status-display.component.jsx";
|
||||||
import { addAlerts } from "../../redux/application/application.actions.js";
|
import { selectIsPartsEntry } from "../../redux/application/application.selectors.js";
|
||||||
import { selectAlerts, selectIsPartsEntry } from "../../redux/application/application.selectors.js";
|
|
||||||
import { selectBodyshop, selectInstanceConflict } from "../../redux/user/user.selectors";
|
|
||||||
import InstanceRenderManager from "../../utils/instanceRenderMgr.js";
|
import InstanceRenderManager from "../../utils/instanceRenderMgr.js";
|
||||||
|
|
||||||
const { Footer } = Layout;
|
const { Footer } = Layout;
|
||||||
|
|
||||||
const mapStateToProps = createStructuredSelector({
|
const mapStateToProps = createStructuredSelector({
|
||||||
conflict: selectInstanceConflict,
|
|
||||||
bodyshop: selectBodyshop,
|
|
||||||
alerts: selectAlerts,
|
|
||||||
isPartsEntry: selectIsPartsEntry
|
isPartsEntry: selectIsPartsEntry
|
||||||
});
|
});
|
||||||
|
|
||||||
const mapDispatchToProps = (dispatch) => ({
|
|
||||||
setAlerts: (alerts) => dispatch(addAlerts(alerts))
|
|
||||||
});
|
|
||||||
|
|
||||||
export function GlobalFooter({ isPartsEntry }) {
|
export function GlobalFooter({ isPartsEntry }) {
|
||||||
const { t } = useTranslation();
|
const { t } = useTranslation();
|
||||||
|
|
||||||
@@ -101,4 +92,4 @@ export function GlobalFooter({ isPartsEntry }) {
|
|||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
export default connect(mapStateToProps, mapDispatchToProps)(GlobalFooter);
|
export default connect(mapStateToProps)(GlobalFooter);
|
||||||
|
|||||||
96
client/src/hooks/useAlertsNotifications.jsx
Normal file
96
client/src/hooks/useAlertsNotifications.jsx
Normal file
@@ -0,0 +1,96 @@
|
|||||||
|
import { useEffect, useState } from "react";
|
||||||
|
import { useDispatch, useSelector } from "react-redux";
|
||||||
|
import * as Sentry from "@sentry/react";
|
||||||
|
import { useNotification } from "../contexts/Notifications/notificationContext.jsx";
|
||||||
|
import { addAlerts } from "../redux/application/application.actions.js";
|
||||||
|
import { selectAlerts } from "../redux/application/application.selectors.js";
|
||||||
|
import InstanceRenderManager from "../utils/instanceRenderMgr.js";
|
||||||
|
|
||||||
|
const DEFAULT_ALERT_FILE_URL = InstanceRenderManager({
|
||||||
|
imex: "https://images.imex.online/alerts/alerts-imex.json",
|
||||||
|
rome: "https://images.imex.online/alerts/alerts-rome.json"
|
||||||
|
});
|
||||||
|
|
||||||
|
/**
|
||||||
|
* useAlertsNotifications
|
||||||
|
* - Fetches alerts JSON and stores it in Redux
|
||||||
|
* - Shows notifications for new alerts (deduped via localStorage)
|
||||||
|
* - Persists dismissed alert IDs under a stable key (default: "displayedAlerts")
|
||||||
|
*/
|
||||||
|
export default function useAlertsNotifications({
|
||||||
|
storageKey = "displayedAlerts",
|
||||||
|
alertFileUrl = DEFAULT_ALERT_FILE_URL
|
||||||
|
} = {}) {
|
||||||
|
const dispatch = useDispatch();
|
||||||
|
const alerts = useSelector(selectAlerts);
|
||||||
|
const notification = useNotification();
|
||||||
|
|
||||||
|
// Track which alerts have been shown/dismissed
|
||||||
|
const [displayedAlertIds, setDisplayedAlertIds] = useState([]);
|
||||||
|
|
||||||
|
// Load displayed alert IDs from localStorage on mount
|
||||||
|
useEffect(() => {
|
||||||
|
try {
|
||||||
|
const raw = localStorage.getItem(storageKey) || "[]";
|
||||||
|
const parsed = JSON.parse(raw);
|
||||||
|
setDisplayedAlertIds(Array.isArray(parsed) ? parsed : []);
|
||||||
|
} catch {
|
||||||
|
setDisplayedAlertIds([]);
|
||||||
|
}
|
||||||
|
}, [storageKey]);
|
||||||
|
|
||||||
|
// Fetch alerts from the remote JSON and dispatch to Redux
|
||||||
|
useEffect(() => {
|
||||||
|
let aborted = false;
|
||||||
|
|
||||||
|
async function fetchAlerts() {
|
||||||
|
try {
|
||||||
|
const response = await fetch(alertFileUrl, { cache: "no-store" });
|
||||||
|
if (!response.ok) throw new Error(`HTTP ${response.status}`);
|
||||||
|
const fetchedAlerts = await response.json();
|
||||||
|
if (!aborted) dispatch(addAlerts(fetchedAlerts));
|
||||||
|
} catch (error) {
|
||||||
|
Sentry.captureException(error, {
|
||||||
|
tags: { scope: "useAlertsNotifications", phase: "fetch-alerts" }
|
||||||
|
});
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
fetchAlerts();
|
||||||
|
return () => {
|
||||||
|
aborted = true;
|
||||||
|
};
|
||||||
|
}, [alertFileUrl, dispatch]);
|
||||||
|
|
||||||
|
// Show notifications for any alerts that haven't been dismissed yet
|
||||||
|
useEffect(() => {
|
||||||
|
if (!alerts || Object.keys(alerts).length === 0) return;
|
||||||
|
|
||||||
|
const alertArray = Object.values(alerts).filter(Boolean);
|
||||||
|
const newAlerts = alertArray.filter((a) => !displayedAlertIds.includes(a.id));
|
||||||
|
|
||||||
|
newAlerts.forEach((alert) => {
|
||||||
|
notification.open({
|
||||||
|
key: `notification-alerts-${alert.id}`,
|
||||||
|
message: alert.message,
|
||||||
|
description: alert.description,
|
||||||
|
type: alert.type || "info",
|
||||||
|
duration: 0,
|
||||||
|
closable: true,
|
||||||
|
onClose: () => {
|
||||||
|
setDisplayedAlertIds((prev) => {
|
||||||
|
const next = prev.includes(alert.id) ? prev : [...prev, alert.id];
|
||||||
|
try {
|
||||||
|
localStorage.setItem(storageKey, JSON.stringify(next));
|
||||||
|
} catch (e) {
|
||||||
|
Sentry.captureException(e, {
|
||||||
|
tags: { scope: "useAlertsNotifications", phase: "persist-dismissed" }
|
||||||
|
});
|
||||||
|
}
|
||||||
|
return next;
|
||||||
|
});
|
||||||
|
}
|
||||||
|
});
|
||||||
|
});
|
||||||
|
}, [alerts, displayedAlertIds, notification, storageKey]);
|
||||||
|
}
|
||||||
@@ -20,11 +20,9 @@ import LoadingSpinner from "../../components/loading-spinner/loading-spinner.com
|
|||||||
import PartnerPingComponent from "../../components/partner-ping/partner-ping.component";
|
import PartnerPingComponent from "../../components/partner-ping/partner-ping.component";
|
||||||
import ShopSubStatusComponent from "../../components/shop-sub-status/shop-sub-status.component";
|
import ShopSubStatusComponent from "../../components/shop-sub-status/shop-sub-status.component";
|
||||||
import UpdateAlert from "../../components/update-alert/update-alert.component";
|
import UpdateAlert from "../../components/update-alert/update-alert.component";
|
||||||
import { useNotification } from "../../contexts/Notifications/notificationContext.jsx";
|
|
||||||
import { addAlerts } from "../../redux/application/application.actions.js";
|
|
||||||
import { selectAlerts } from "../../redux/application/application.selectors.js";
|
|
||||||
import { selectBodyshop, selectInstanceConflict } from "../../redux/user/user.selectors";
|
import { selectBodyshop, selectInstanceConflict } from "../../redux/user/user.selectors";
|
||||||
import InstanceRenderManager from "../../utils/instanceRenderMgr.js";
|
import InstanceRenderManager from "../../utils/instanceRenderMgr.js";
|
||||||
|
import useAlertsNotifications from "../../hooks/useAlertsNotifications.jsx";
|
||||||
|
|
||||||
const PrintCenterModalContainer = lazy(
|
const PrintCenterModalContainer = lazy(
|
||||||
() => import("../../components/print-center-modal/print-center-modal.container")
|
() => import("../../components/print-center-modal/print-center-modal.container")
|
||||||
@@ -108,78 +106,16 @@ const { Content } = Layout;
|
|||||||
|
|
||||||
const mapStateToProps = createStructuredSelector({
|
const mapStateToProps = createStructuredSelector({
|
||||||
conflict: selectInstanceConflict,
|
conflict: selectInstanceConflict,
|
||||||
bodyshop: selectBodyshop,
|
bodyshop: selectBodyshop
|
||||||
alerts: selectAlerts
|
|
||||||
});
|
});
|
||||||
|
|
||||||
const ALERT_FILE_URL = InstanceRenderManager({
|
export function Manage({ conflict, bodyshop }) {
|
||||||
imex: "https://images.imex.online/alerts/alerts-imex.json",
|
|
||||||
rome: "https://images.imex.online/alerts/alerts-rome.json"
|
|
||||||
});
|
|
||||||
|
|
||||||
const mapDispatchToProps = (dispatch) => ({
|
|
||||||
setAlerts: (alerts) => dispatch(addAlerts(alerts))
|
|
||||||
});
|
|
||||||
|
|
||||||
export function Manage({ conflict, bodyshop, alerts, setAlerts }) {
|
|
||||||
const { t } = useTranslation();
|
const { t } = useTranslation();
|
||||||
const [chatVisible] = useState(false);
|
const [chatVisible] = useState(false);
|
||||||
const notification = useNotification();
|
|
||||||
|
|
||||||
// State to track displayed alerts
|
// Centralized alerts handling (fetch + dedupe + notifications)
|
||||||
const [displayedAlertIds, setDisplayedAlertIds] = useState([]);
|
useAlertsNotifications();
|
||||||
|
|
||||||
// Fetch displayed alerts from localStorage on mount
|
|
||||||
useEffect(() => {
|
|
||||||
const displayedAlerts = JSON.parse(localStorage.getItem("displayedAlerts") || "[]");
|
|
||||||
setDisplayedAlertIds(displayedAlerts);
|
|
||||||
}, []);
|
|
||||||
|
|
||||||
// Fetch alerts from the JSON file and dispatch to Redux store
|
|
||||||
useEffect(() => {
|
|
||||||
const fetchAlerts = async () => {
|
|
||||||
try {
|
|
||||||
const response = await fetch(ALERT_FILE_URL);
|
|
||||||
const fetchedAlerts = await response.json();
|
|
||||||
setAlerts(fetchedAlerts);
|
|
||||||
} catch (error) {
|
|
||||||
console.warn("Error fetching alerts:", error.message);
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
fetchAlerts().catch((err) => `Error fetching Bodyshop Alerts: ${err?.message || ""}`);
|
|
||||||
}, [setAlerts]);
|
|
||||||
|
|
||||||
// Use useEffect to watch for new alerts
|
|
||||||
useEffect(() => {
|
|
||||||
if (alerts && Object.keys(alerts).length > 0) {
|
|
||||||
// Convert the alerts object into an array
|
|
||||||
const alertArray = Object.values(alerts);
|
|
||||||
|
|
||||||
// Filter out alerts that have already been dismissed
|
|
||||||
const newAlerts = alertArray.filter((alert) => !displayedAlertIds.includes(alert.id));
|
|
||||||
|
|
||||||
newAlerts.forEach((alert) => {
|
|
||||||
// Display the notification
|
|
||||||
notification.open({
|
|
||||||
key: "notification-alerts-" + alert.id,
|
|
||||||
message: alert.message,
|
|
||||||
description: alert.description,
|
|
||||||
type: alert.type || "info",
|
|
||||||
duration: 0,
|
|
||||||
closable: true,
|
|
||||||
onClose: () => {
|
|
||||||
// When the notification is closed, update displayed alerts state and localStorage
|
|
||||||
setDisplayedAlertIds((prevIds) => {
|
|
||||||
const updatedIds = [...prevIds, alert.id];
|
|
||||||
localStorage.setItem("displayedAlerts", JSON.stringify(updatedIds));
|
|
||||||
return updatedIds;
|
|
||||||
});
|
|
||||||
}
|
|
||||||
});
|
|
||||||
});
|
|
||||||
}
|
|
||||||
}, [alerts, displayedAlertIds, notification]);
|
|
||||||
useEffect(() => {
|
useEffect(() => {
|
||||||
window.Canny("initChangelog", {
|
window.Canny("initChangelog", {
|
||||||
appID: "680bd2c7ee501290377f6686",
|
appID: "680bd2c7ee501290377f6686",
|
||||||
@@ -657,4 +593,4 @@ export function Manage({ conflict, bodyshop, alerts, setAlerts }) {
|
|||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
export default connect(mapStateToProps, mapDispatchToProps)(Manage);
|
export default connect(mapStateToProps)(Manage);
|
||||||
|
|||||||
@@ -1,6 +1,6 @@
|
|||||||
import * as Sentry from "@sentry/react";
|
import * as Sentry from "@sentry/react";
|
||||||
import { FloatButton, Layout, Spin } from "antd";
|
import { FloatButton, Layout, Spin } from "antd";
|
||||||
import { lazy, Suspense, useEffect, useState } from "react";
|
import { lazy, Suspense, useEffect } from "react";
|
||||||
import { useTranslation } from "react-i18next";
|
import { useTranslation } from "react-i18next";
|
||||||
import { connect } from "react-redux";
|
import { connect } from "react-redux";
|
||||||
import { Route, Routes } from "react-router-dom";
|
import { Route, Routes } from "react-router-dom";
|
||||||
@@ -12,11 +12,9 @@ import GlobalFooter from "../../components/global-footer/global-footer.component
|
|||||||
import LoadingSpinner from "../../components/loading-spinner/loading-spinner.component.jsx";
|
import LoadingSpinner from "../../components/loading-spinner/loading-spinner.component.jsx";
|
||||||
import ShopSubStatusComponent from "../../components/shop-sub-status/shop-sub-status.component.jsx";
|
import ShopSubStatusComponent from "../../components/shop-sub-status/shop-sub-status.component.jsx";
|
||||||
import UpdateAlert from "../../components/update-alert/update-alert.component.jsx";
|
import UpdateAlert from "../../components/update-alert/update-alert.component.jsx";
|
||||||
import { useNotification } from "../../contexts/Notifications/notificationContext.jsx";
|
|
||||||
import { addAlerts } from "../../redux/application/application.actions.js";
|
|
||||||
import { selectAlerts } from "../../redux/application/application.selectors.js";
|
|
||||||
import { selectBodyshop, selectInstanceConflict } from "../../redux/user/user.selectors.js";
|
import { selectBodyshop, selectInstanceConflict } from "../../redux/user/user.selectors.js";
|
||||||
import InstanceRenderManager from "../../utils/instanceRenderMgr.js";
|
import InstanceRenderManager from "../../utils/instanceRenderMgr.js";
|
||||||
|
import useAlertsNotifications from "../../hooks/useAlertsNotifications.jsx";
|
||||||
|
|
||||||
const SimplifiedPartsJobsPage = lazy(() => import("../simplified-parts-jobs/simplified-parts-jobs.page.jsx"));
|
const SimplifiedPartsJobsPage = lazy(() => import("../simplified-parts-jobs/simplified-parts-jobs.page.jsx"));
|
||||||
const SimplifiedPartsJobsDetailPage = lazy(
|
const SimplifiedPartsJobsDetailPage = lazy(
|
||||||
@@ -35,77 +33,14 @@ const { Content } = Layout;
|
|||||||
|
|
||||||
const mapStateToProps = createStructuredSelector({
|
const mapStateToProps = createStructuredSelector({
|
||||||
conflict: selectInstanceConflict,
|
conflict: selectInstanceConflict,
|
||||||
bodyshop: selectBodyshop,
|
bodyshop: selectBodyshop
|
||||||
alerts: selectAlerts
|
|
||||||
});
|
});
|
||||||
|
|
||||||
const ALERT_FILE_URL = InstanceRenderManager({
|
export function SimplifiedPartsPage({ conflict, bodyshop }) {
|
||||||
imex: "https://images.imex.online/alerts/alerts-imex.json",
|
|
||||||
rome: "https://images.imex.online/alerts/alerts-rome.json"
|
|
||||||
});
|
|
||||||
|
|
||||||
const mapDispatchToProps = (dispatch) => ({
|
|
||||||
setAlerts: (alerts) => dispatch(addAlerts(alerts))
|
|
||||||
});
|
|
||||||
|
|
||||||
export function SimplifiedPartsPage({ conflict, bodyshop, alerts, setAlerts }) {
|
|
||||||
const { t } = useTranslation();
|
const { t } = useTranslation();
|
||||||
const notification = useNotification();
|
|
||||||
|
|
||||||
// State to track displayed alerts
|
// Centralized alerts handling (fetch + dedupe + notifications)
|
||||||
const [displayedAlertIds, setDisplayedAlertIds] = useState([]);
|
useAlertsNotifications();
|
||||||
|
|
||||||
// Fetch displayed alerts from localStorage on mount
|
|
||||||
useEffect(() => {
|
|
||||||
const displayedAlerts = JSON.parse(localStorage.getItem("displayedAlerts") || "[]");
|
|
||||||
setDisplayedAlertIds(displayedAlerts);
|
|
||||||
}, []);
|
|
||||||
|
|
||||||
// Fetch alerts from the JSON file and dispatch to Redux store
|
|
||||||
useEffect(() => {
|
|
||||||
const fetchAlerts = async () => {
|
|
||||||
try {
|
|
||||||
const response = await fetch(ALERT_FILE_URL);
|
|
||||||
const fetchedAlerts = await response.json();
|
|
||||||
setAlerts(fetchedAlerts);
|
|
||||||
} catch (error) {
|
|
||||||
console.warn("Error fetching alerts:", error.message);
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
fetchAlerts().catch((err) => `Error fetching Bodyshop Alerts: ${err?.message || ""}`);
|
|
||||||
}, [setAlerts]);
|
|
||||||
|
|
||||||
// Use useEffect to watch for new alerts
|
|
||||||
useEffect(() => {
|
|
||||||
if (alerts && Object.keys(alerts).length > 0) {
|
|
||||||
// Convert the alerts object into an array
|
|
||||||
const alertArray = Object.values(alerts);
|
|
||||||
|
|
||||||
// Filter out alerts that have already been dismissed
|
|
||||||
const newAlerts = alertArray.filter((alert) => !displayedAlertIds.includes(alert.id));
|
|
||||||
|
|
||||||
newAlerts.forEach((alert) => {
|
|
||||||
// Display the notification
|
|
||||||
notification.open({
|
|
||||||
key: "notification-alerts-" + alert.id,
|
|
||||||
message: alert.message,
|
|
||||||
description: alert.description,
|
|
||||||
type: alert.type || "info",
|
|
||||||
duration: 0,
|
|
||||||
closable: true,
|
|
||||||
onClose: () => {
|
|
||||||
// When the notification is closed, update displayed alerts state and localStorage
|
|
||||||
setDisplayedAlertIds((prevIds) => {
|
|
||||||
const updatedIds = [...prevIds, alert.id];
|
|
||||||
localStorage.setItem("displayedAlerts", JSON.stringify(updatedIds));
|
|
||||||
return updatedIds;
|
|
||||||
});
|
|
||||||
}
|
|
||||||
});
|
|
||||||
});
|
|
||||||
}
|
|
||||||
}, [alerts, displayedAlertIds, notification]);
|
|
||||||
|
|
||||||
useEffect(() => {
|
useEffect(() => {
|
||||||
document.title = InstanceRenderManager({
|
document.title = InstanceRenderManager({
|
||||||
@@ -193,7 +128,6 @@ export function SimplifiedPartsPage({ conflict, bodyshop, alerts, setAlerts }) {
|
|||||||
return (
|
return (
|
||||||
<Layout style={{ minHeight: "100vh" }} className="layout-container">
|
<Layout style={{ minHeight: "100vh" }} className="layout-container">
|
||||||
<UpdateAlert />
|
<UpdateAlert />
|
||||||
{/* <HeaderContainer /> */}
|
|
||||||
<Content className="content-container">
|
<Content className="content-container">
|
||||||
<Sentry.ErrorBoundary fallback={<ErrorBoundary />} showDialog>
|
<Sentry.ErrorBoundary fallback={<ErrorBoundary />} showDialog>
|
||||||
{PageContent}
|
{PageContent}
|
||||||
@@ -205,4 +139,4 @@ export function SimplifiedPartsPage({ conflict, bodyshop, alerts, setAlerts }) {
|
|||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
export default connect(mapStateToProps, mapDispatchToProps)(SimplifiedPartsPage);
|
export default connect(mapStateToProps)(SimplifiedPartsPage);
|
||||||
|
|||||||
Reference in New Issue
Block a user