diff --git a/Client/src/Components/Avatar/index.jsx b/Client/src/Components/Avatar/index.jsx
index 4f4351508..6c4bf80c7 100644
--- a/Client/src/Components/Avatar/index.jsx
+++ b/Client/src/Components/Avatar/index.jsx
@@ -18,7 +18,7 @@ import { useEffect, useState } from "react";
const Avatar = ({ src, small, sx }) => {
const { user } = useSelector((state) => state.auth);
- const style = small ? { width: 25, height: 25 } : { width: 64, height: 64 };
+ const style = small ? { width: 32, height: 32 } : { width: 64, height: 64 };
const border = small ? 1 : 3;
const [image, setImage] = useState();
@@ -35,7 +35,7 @@ const Avatar = ({ src, small, sx }) => {
src ? src : user?.avatarImage ? image : "/static/images/avatar/2.jpg"
}
sx={{
- fontSize: small ? "13px" : "22px",
+ fontSize: small ? "16px" : "22px",
fontWeight: 400,
display: "inline-flex",
"&::before": {
@@ -52,7 +52,8 @@ const Avatar = ({ src, small, sx }) => {
...sx,
}}
>
- {user.firstName?.charAt(0)}{user.lastName?.charAt(0)}
+ {user.firstName?.charAt(0)}
+ {!small && user.lastName?.charAt(0)}
);
};
diff --git a/Client/src/Components/NavBar/index.css b/Client/src/Components/NavBar/index.css
deleted file mode 100644
index fcedc53a2..000000000
--- a/Client/src/Components/NavBar/index.css
+++ /dev/null
@@ -1,34 +0,0 @@
-/* NavBar Component Styles*/
-.MuiToolbar-root {
- min-height: var(--env-var-nav-bar-height) !important;
-}
-
-.icon-button-toggle-title {
- font-size: var(--env-var-font-size-medium);
- color: var(--env-var-color-5);
-}
-
-.icon-button-toggle-pic {
- width: 10px;
- height: 5px;
-}
-
-#icon-button {
- -webkit-transition: none;
- transition: none;
- outline: none;
- box-shadow: none;
- background-color: transparent;
-}
-
-#menu-appbar svg {
- width: 16px;
- height: 16px;
- color: var(--env-var-color-25);
-}
-
-#bw-uptime-logo-dashboard {
- width: fit-content;
- height: 16px;
- margin-left: 15px;
-}
diff --git a/Client/src/Components/NavBar/index.jsx b/Client/src/Components/NavBar/index.jsx
deleted file mode 100644
index 95123d975..000000000
--- a/Client/src/Components/NavBar/index.jsx
+++ /dev/null
@@ -1,203 +0,0 @@
-import "./index.css";
-import { cloneElement, useState } from "react";
-import AppBar from "@mui/material/AppBar";
-import Box from "@mui/material/Box";
-import Toolbar from "@mui/material/Toolbar";
-import IconButton from "@mui/material/IconButton";
-import Typography from "@mui/material/Typography";
-import Menu from "@mui/material/Menu";
-import Avatar from "../Avatar";
-import Tooltip from "@mui/material/Tooltip";
-import MenuItem from "@mui/material/MenuItem";
-import { useTheme } from "@mui/material/styles";
-import KeyboardArrowDownIcon from "@mui/icons-material/KeyboardArrowDown";
-import { clearAuthState } from "../../Features/Auth/authSlice";
-import { clearUptimeMonitorState } from "../../Features/UptimeMonitors/uptimeMonitorsSlice";
-
-import { useDispatch, useSelector } from "react-redux";
-import { useNavigate } from "react-router-dom";
-import LockSvg from "../../assets/icons/lock.svg?react";
-import UserSvg from "../../assets/icons/user.svg?react";
-import TeamSvg from "../../assets/icons/user-two.svg?react";
-import LogoutSvg from "../../assets/icons/logout.svg?react";
-import { Stack, useScrollTrigger } from "@mui/material";
-import axiosIntance from "../../Utils/axiosConfig";
-import axios from "axios";
-
-const icons = {
- Profile: ,
- Team: ,
- Password: ,
- Logout: ,
-};
-
-function AddBorderOnScroll(props) {
- const { children, window } = props;
- const trigger = useScrollTrigger({
- target: window ? window() : undefined,
- disableHysteresis: true,
- threshold: 0,
- });
-
- return (
-
- {children}
-
- );
-}
-
-/**
- * NavBar component
- *
- * A responsive navigation bar component with a user menu.
- *
- * @component
- * @example
- * return (
- *
- * )
- */
-function NavBar() {
- const theme = useTheme();
- const [anchorElUser, setAnchorElUser] = useState(null);
- const dispatch = useDispatch();
- const navigate = useNavigate();
- const authState = useSelector((state) => state.auth);
-
- // Initialize settings and update based on user role
- let settings = ["Profile", "Password", "Team", "Logout"];
- if (authState.user?.role && !authState.user.role.includes("admin")) {
- settings = ["Profile", "Password", "Logout"];
- }
-
- /**
- * Handles opening the user menu.
- *
- * @param {React.MouseEvent} event - The event triggered by clicking the user menu button.
- */
- const handleOpenUserMenu = (event) => {
- setAnchorElUser(event.currentTarget);
- };
-
- /**
- * Handles logging out the user
- *
- */
- const logout = async () => {
- // Clear auth state
- dispatch(clearAuthState());
- dispatch(clearUptimeMonitorState());
- // Make request to BE to remove JWT from user
- await axiosIntance.post(
- "/auth/logout",
- { email: authState.user.email },
- {
- headers: {
- Authorization: `Bearer ${authState.authToken}`,
- "Content-Type": "application/json",
- },
- }
- );
- navigate("/login");
- };
-
- /**
- * Handles closing the user menu.
- */
- const handleCloseUserMenu = (setting) => {
- setAnchorElUser(null);
- switch (setting) {
- case "Profile":
- navigate("/account/profile");
- break;
- case "Team":
- navigate("/account/team");
- break;
- case "Password":
- navigate("/account/password");
- break;
- case "Logout":
- logout();
- break;
- default:
- break;
- }
- };
-
- return (
-
-
-
-
-
-
-
- {authState.user?.firstName} {authState.user?.lastName}
-
-
-
-
-
-
-
-
- );
-}
-
-export default NavBar;
diff --git a/Client/src/Components/Sidebar/index.css b/Client/src/Components/Sidebar/index.css
index 6fa9d2134..4840043c3 100644
--- a/Client/src/Components/Sidebar/index.css
+++ b/Client/src/Components/Sidebar/index.css
@@ -18,11 +18,14 @@ aside span.MuiTypography-root {
color: var(--env-var-color-5);
line-height: 1;
}
-aside .MuiStack-root:nth-last-child(2) {
+aside .MuiStack-root:nth-last-child(3) {
margin-top: auto;
+}
+aside .MuiStack-root:last-child,
+aside .MuiStack-root:nth-last-child(3) {
position: relative;
}
-aside .MuiStack-root:nth-last-child(2):before {
+aside .MuiStack-root:last-child:before {
content: "";
position: absolute;
top: -10px;
@@ -31,5 +34,35 @@ aside .MuiStack-root:nth-last-child(2):before {
border-top: solid 1px var(--env-var-color-6);
}
aside .MuiStack-root:last-child {
- margin-bottom: 10px;
+ margin-top: 15px;
+}
+
+.sidebar-menu {
+ margin-top: -20px;
+}
+.sidebar-menu .MuiPaper-root {
+ box-shadow: var(--env-var-shadow-1);
+ border: solid 1px var(--env-var-color-6);
+ border-radius: var(--env-var-radius-1);
+ gap: 1px;
+}
+.sidebar-menu .MuiList-root {
+ min-width: 100px;
+ width: 150px;
+ padding-bottom: 0;
+}
+.sidebar-menu li.MuiButtonBase-root:last-child {
+ border-top: solid 1px var(--env-var-color-6);
+ padding: 12px 16px;
+}
+.sidebar-menu li.MuiButtonBase-root {
+ min-height: fit-content;
+}
+.sidebar-menu .MuiList-root svg {
+ width: 16px;
+ height: 16px;
+}
+.sidebar-menu span {
+ font-size: var(--env-var-font-size-medium);
+ color: var(--env-var-color-2);
}
diff --git a/Client/src/Components/Sidebar/index.jsx b/Client/src/Components/Sidebar/index.jsx
index 3b2a32394..804704e07 100644
--- a/Client/src/Components/Sidebar/index.jsx
+++ b/Client/src/Components/Sidebar/index.jsx
@@ -1,7 +1,15 @@
-import { Stack, Typography } from "@mui/material";
+import { useState } from "react";
+import { Box, Menu, MenuItem, Stack, Tooltip, Typography } from "@mui/material";
import { useLocation, useNavigate } from "react-router";
import { useTheme } from "@emotion/react";
-
+import { useDispatch, useSelector } from "react-redux";
+import { clearAuthState } from "../../Features/Auth/authSlice";
+import { clearUptimeMonitorState } from "../../Features/UptimeMonitors/uptimeMonitorsSlice";
+import Avatar from "../Avatar";
+import LockSvg from "../../assets/icons/lock.svg?react";
+import UserSvg from "../../assets/icons/user.svg?react";
+import TeamSvg from "../../assets/icons/user-two.svg?react";
+import LogoutSvg from "../../assets/icons/logout.svg?react";
import BWULogo from "../../assets/Images/bwl-logo.svg?react";
import Support from "../../assets/icons/support.svg?react";
import StatusPages from "../../assets/icons/status-pages.svg?react";
@@ -11,6 +19,7 @@ import Incidents from "../../assets/icons/incidents.svg?react";
import Integrations from "../../assets/icons/integrations.svg?react";
import PageSpeed from "../../assets/icons/page-speed.svg?react";
import Settings from "../../assets/icons/settings.svg?react";
+import Arrow from "../../assets/icons/down-arrow.svg?react";
import "./index.css";
@@ -32,10 +41,80 @@ const menu = [
{ name: "Settings", path: "settings", icon: },
];
+const icons = {
+ Profile: ,
+ Team: ,
+ Password: ,
+ Logout: ,
+};
+
function Sidebar() {
const theme = useTheme();
const navigate = useNavigate();
const location = useLocation();
+ const dispatch = useDispatch();
+ const [anchorElUser, setAnchorElUser] = useState(null);
+ const authState = useSelector((state) => state.auth);
+
+ // Initialize settings and update based on user role
+ let settings = ["Profile", "Password", "Team", "Logout"];
+ if (authState.user?.role && !authState.user.role.includes("admin")) {
+ settings = ["Profile", "Password", "Logout"];
+ }
+
+ /**
+ * Handles opening the user menu.
+ *
+ * @param {React.MouseEvent} event - The event triggered by clicking the user menu button.
+ */
+ const handleOpenUserMenu = (event) => {
+ setAnchorElUser(event.currentTarget);
+ };
+
+ /**
+ * Handles logging out the user
+ *
+ */
+ const logout = async () => {
+ // Clear auth state
+ dispatch(clearAuthState());
+ dispatch(clearUptimeMonitorState());
+ // Make request to BE to remove JWT from user
+ await axiosIntance.post(
+ "/auth/logout",
+ { email: authState.user.email },
+ {
+ headers: {
+ Authorization: `Bearer ${authState.authToken}`,
+ "Content-Type": "application/json",
+ },
+ }
+ );
+ navigate("/login");
+ };
+
+ /**
+ * Handles closing the user menu.
+ */
+ const handleCloseUserMenu = (setting) => {
+ setAnchorElUser(null);
+ switch (setting) {
+ case "Profile":
+ navigate("/account/profile");
+ break;
+ case "Team":
+ navigate("/account/team");
+ break;
+ case "Password":
+ navigate("/account/password");
+ break;
+ case "Logout":
+ logout();
+ break;
+ default:
+ break;
+ }
+ };
return (
@@ -48,6 +127,8 @@ function Sidebar() {
key={item.path}
direction="row"
alignItems="center"
+ py={theme.gap.small}
+ px={theme.gap.medium}
gap={theme.gap.small}
borderRadius={`${theme.shape.borderRadius}px`}
onClick={() =>
@@ -59,12 +140,66 @@ function Sidebar() {
)
: navigate(`/${item.path}`)
}
- sx={{ p: `${theme.gap.small} ${theme.gap.medium}` }}
>
{item.icon}
{item.name}
))}
+
+
+
+
+ {authState.user?.firstName} {authState.user?.lastName}
+
+
+
+
+
);
}
diff --git a/Client/src/Pages/Account/index.jsx b/Client/src/Pages/Account/index.jsx
index db08ec3a2..82f37a969 100644
--- a/Client/src/Pages/Account/index.jsx
+++ b/Client/src/Pages/Account/index.jsx
@@ -29,7 +29,7 @@ const Account = ({ open = "profile" }) => {
if (!user.role.includes("admin")) tabList = ["Profile", "Password"];
return (
-
+
+
+