diff --git a/packages/admin-ui/package.json b/packages/admin-ui/package.json index b8fc70d15..5d671771a 100644 --- a/packages/admin-ui/package.json +++ b/packages/admin-ui/package.json @@ -22,9 +22,9 @@ }, "dependencies": { "@dappnode/common": "^0.1.0", - "@dappnode/eventbus": "^0.1.0", "@dappnode/dappmanager": "^0.1.0", - "@dappnode/types": "^0.1.25", + "@dappnode/eventbus": "^0.1.0", + "@dappnode/types": "^0.1.26", "@reduxjs/toolkit": "^1.3.5", "@types/clipboard": "^2.0.7", "@types/node": "^18.11.18", diff --git a/packages/admin-ui/src/App.tsx b/packages/admin-ui/src/App.tsx index 31e08d116..6aa57aaad 100644 --- a/packages/admin-ui/src/App.tsx +++ b/packages/admin-ui/src/App.tsx @@ -15,56 +15,57 @@ import { Login } from "./start-pages/Login"; import { Register } from "./start-pages/Register"; import { NoConnection } from "start-pages/NoConnection"; // Types -import { Theme, UsageMode } from "types"; +import { AppContextIface, Theme, UiModuleStatus } from "types"; -export const UsageContext = React.createContext({ - usage: "advanced", - toggleUsage: () => {} -}); - -export const ThemeContext = React.createContext({ +export const AppContext = React.createContext({ theme: "light", - toggleTheme: () => {} + stakersModuleStatus: "enabled", + rollupsModuleStatus: "disabled", + toggleTheme: () => {}, + toggleStakersModuleStatus: () => {}, + toggleRollupsModuleStatus: () => {} }); +const useLocalStorage = ( + key: string, + initialValue: T +): [T, React.Dispatch>] => { + const [storedValue, setStoredValue] = useState(() => { + try { + const item = window.localStorage.getItem(key); + // Assert that either the item or initialValue is of type T + return (item as T) || initialValue; + } catch (error) { + return initialValue; + } + }); + + useEffect(() => { + window.localStorage.setItem(key, storedValue); + }, [key, storedValue]); + + return [storedValue, setStoredValue]; +}; + function MainApp({ username }: { username: string }) { // App is the parent container of any other component. // If this re-renders, the whole app will. So DON'T RERENDER APP! // Check ONCE what is the status of the VPN and redirect to the login page. - const [screenWidth, setScreenWidth] = useState(window.screen.width); - - //const storedUsage = localStorage.getItem("usage"); - const storedTheme = localStorage.getItem("theme"); - //const initialUsage = storedUsage === "advanced" ? "advanced" : "basic"; - const initialUsage = "advanced"; - const initialTheme = - storedTheme === "light" || storedTheme === "dark" ? storedTheme : "light"; - - const [theme, setTheme] = useState(initialTheme); - const [usage, setUsage] = useState(initialUsage); - - const toggleTheme = () => { - setTheme(curr => (curr === "light" ? "dark" : "light")); - }; - - const toggleUsage = () => { - setUsage(curr => (curr === "basic" ? "advanced" : "basic")); - }; - - useEffect(() => { - localStorage.setItem("theme", theme); - }, [theme]); - - useEffect(() => { - localStorage.setItem("usage", usage); - }, [usage]); + const [screenWidth, setScreenWidth] = useState(window.innerWidth); + const [theme, setTheme] = useLocalStorage("theme", "light"); + const [stakersModuleStatus, setStakersModuleStatus] = useLocalStorage< + UiModuleStatus + >("stakersModuleStatus", "enabled"); + const [rollupsModuleStatus, setRollupsModuleStatus] = useLocalStorage< + UiModuleStatus + >("rollupsModuleStatus", "disabled"); useEffect(() => { const handleResize = () => setScreenWidth(window.innerWidth); window.addEventListener("resize", handleResize); return () => window.removeEventListener("resize", handleResize); - }, [screenWidth]); + }, []); // Scroll to top on pathname change const screenLocation = useLocation(); @@ -72,45 +73,55 @@ function MainApp({ username }: { username: string }) { window.scrollTo(0, 0); }, [screenLocation.pathname]); + const appContext: AppContextIface = { + theme, + stakersModuleStatus, + rollupsModuleStatus, + toggleTheme: () => + setTheme((curr: Theme) => (curr === "light" ? "dark" : "light")), + toggleStakersModuleStatus: () => + setStakersModuleStatus((curr: UiModuleStatus) => + curr === "enabled" ? "disabled" : "enabled" + ), + toggleRollupsModuleStatus: () => + setRollupsModuleStatus((curr: UiModuleStatus) => + curr === "enabled" ? "disabled" : "enabled" + ) + }; + return ( - - -
- - -
- - - - - {Object.values(pages).map(({ RootComponent, rootPath }) => ( - - - - } - /> - ))} - {/* Redirection for routes with hashes */} - {/* 404 routes redirect to dashboard or default page */} - } /> - -
- - {/* Place here non-page components */} - - + +
+ + +
+ + + + + {/** Provide the app context only to the dashboard (where the modules switch is handled) */} + {Object.values(pages).map(({ RootComponent, rootPath }) => ( + + + + } + /> + ))} + {/* Redirection for routes with hashes */} + {/* 404 routes redirect to dashboard or default page */} + } /> +
- - + + {/* Place here non-page components */} + + +
+
); } diff --git a/packages/admin-ui/src/__mock-backend__/index.ts b/packages/admin-ui/src/__mock-backend__/index.ts index ed55e15dc..c9435cd4a 100644 --- a/packages/admin-ui/src/__mock-backend__/index.ts +++ b/packages/admin-ui/src/__mock-backend__/index.ts @@ -223,11 +223,11 @@ export const otherCalls: Omit = { ethProvider: "http://geth.dappnode:8545", fullnodeDomainTarget: "geth.dnp.dappnode.eth", newFeatureIds: [ - "repository", - "repository-fallback", - "system-auto-updates", - "enable-ethical-metrics", - "change-host-password" + //"repository", + //"repository-fallback", + //"system-auto-updates", + //"enable-ethical-metrics", + //"change-host-password" ] }), natRenewalEnable: async () => {}, @@ -252,7 +252,123 @@ export const otherCalls: Omit = { mail: "@example.com", isEnabled: true }), - disableEthicalMetrics: async () => {} + disableEthicalMetrics: async () => {}, + optimismConfigGet: async () => ({ + executionClients: [ + { + status: "ok", + dnpName: "op-geth.dnp.dappnode.eth", + avatarUrl: "", + isInstalled: true, + isUpdated: true, + isRunning: true, + data: { + dnpName: "package", + reqVersion: "0.1.0", + semVersion: "0.1.0", + imageFile: { + hash: "QM..", + source: "ipfs", + size: 123 + }, + warnings: {}, + signedSafe: true, + metadata: { + name: "geth.dnp.dappnode.eth", + description: "Go implementation of ethereum. Execution client", + shortDescription: "Go implementation of ethereum", + version: "0.1.0" + } + }, + isSelected: true, + enableHistorical: true + }, + { + status: "ok", + dnpName: "op-erigon.dnp.dappnode.eth", + avatarUrl: "", + isInstalled: true, + isUpdated: true, + isRunning: true, + data: { + dnpName: "package", + reqVersion: "0.1.0", + semVersion: "0.1.0", + imageFile: { + hash: "QM..", + source: "ipfs", + size: 123 + }, + warnings: {}, + signedSafe: true, + metadata: { + name: "geth.dnp.dappnode.eth", + description: "Go implementation of ethereum. Execution client", + shortDescription: "Go implementation of ethereum", + version: "0.1.0" + } + }, + isSelected: false, + enableHistorical: false + } + ], + rollup: { + status: "ok", + dnpName: "op-node.dnp.dappnode.eth", + avatarUrl: "", + isInstalled: false, + isUpdated: false, + isRunning: true, + data: { + dnpName: "package", + reqVersion: "0.1.0", + semVersion: "0.1.0", + imageFile: { + hash: "QM..", + source: "ipfs", + size: 123 + }, + warnings: {}, + signedSafe: true, + metadata: { + name: "geth.dnp.dappnode.eth", + description: "Go implementation of ethereum. Execution client", + shortDescription: "Go implementation of ethereum", + version: "0.1.0" + } + }, + isSelected: false, + mainnetRpcUrl: "" + }, + archive: { + status: "ok", + dnpName: "op-l2geth.dnp.dappnode.eth", + avatarUrl: "", + isInstalled: false, + isUpdated: false, + isRunning: true, + data: { + dnpName: "package", + reqVersion: "0.1.0", + semVersion: "0.1.0", + imageFile: { + hash: "QM..", + source: "ipfs", + size: 123 + }, + warnings: {}, + signedSafe: true, + metadata: { + name: "geth.dnp.dappnode.eth", + description: "Go implementation of ethereum. Execution client", + shortDescription: "Go implementation of ethereum", + version: "0.1.0" + } + }, + isSelected: true + } + }), + optimismConfigSet: async () => {} }; export const calls: Routes = { diff --git a/packages/admin-ui/src/components/sidebar/SideBar.tsx b/packages/admin-ui/src/components/sidebar/SideBar.tsx index 903b6f6be..3b4f2a769 100644 --- a/packages/admin-ui/src/components/sidebar/SideBar.tsx +++ b/packages/admin-ui/src/components/sidebar/SideBar.tsx @@ -1,23 +1,31 @@ import React from "react"; import { NavLink } from "react-router-dom"; -import { advancedItems, basicItems, fundedBy } from "./navbarItems"; +import { sidenavItems, fundedBy } from "./navbarItems"; import logoWide from "img/dappnode-logo-wide-min.png"; import logoWideDark from "img/dappnode-logo-wide-min-dark.png"; import logomin from "img/dappnode-logo-only.png"; -import { ThemeContext, UsageContext } from "../../App"; +import { AppContext } from "../../App"; import "./sidebar.scss"; -if (!Array.isArray(advancedItems)) - throw Error("advancedItems must be an array"); -if (!Array.isArray(basicItems)) throw Error("basicItems must be an array"); +if (!Array.isArray(sidenavItems)) throw Error("sidenavItems must be an array"); if (!Array.isArray(fundedBy)) throw Error("fundedBy must be an array"); export default function SideBar({ screenWidth }: { screenWidth: number }) { - const { theme } = React.useContext(ThemeContext); - const { usage } = React.useContext(UsageContext); + const { theme, rollupsModuleStatus, stakersModuleStatus } = React.useContext( + AppContext + ); + + const stakersItem = sidenavItems.find(item => item.name === "Stakers"); + if (stakersItem) { + if (stakersModuleStatus === "enabled") stakersItem.show = true; + else stakersItem.show = false; + } + const rollupsItem = sidenavItems.find(item => item.name === "Rollups"); + if (rollupsItem) { + if (rollupsModuleStatus === "enabled") rollupsItem.show = true; + else rollupsItem.show = false; + } - const sidenavItems = - usage === "advanced" ? [...basicItems, ...advancedItems] : basicItems; return (