ntfy/web/src/components/ActionBar.js

172 lines
6.4 KiB
JavaScript
Raw Normal View History

2022-02-25 17:46:22 +00:00
import AppBar from "@mui/material/AppBar";
import Navigation from "./Navigation";
import Toolbar from "@mui/material/Toolbar";
import IconButton from "@mui/material/IconButton";
import MenuIcon from "@mui/icons-material/Menu";
import Typography from "@mui/material/Typography";
import * as React from "react";
2022-03-06 03:33:34 +00:00
import {useEffect, useRef, useState} from "react";
2022-02-28 21:56:38 +00:00
import Box from "@mui/material/Box";
import {topicShortUrl} from "../app/utils";
2022-03-06 03:33:34 +00:00
import {useLocation, useNavigate} from "react-router-dom";
import ClickAwayListener from '@mui/material/ClickAwayListener';
import Grow from '@mui/material/Grow';
import Paper from '@mui/material/Paper';
import Popper from '@mui/material/Popper';
import MenuItem from '@mui/material/MenuItem';
import MenuList from '@mui/material/MenuList';
import MoreVertIcon from "@mui/icons-material/MoreVert";
2022-03-08 21:56:41 +00:00
import NotificationsIcon from '@mui/icons-material/Notifications';
import NotificationsOffIcon from '@mui/icons-material/NotificationsOff';
2022-03-06 03:33:34 +00:00
import api from "../app/Api";
import routes from "./routes";
2022-03-06 03:33:34 +00:00
import subscriptionManager from "../app/SubscriptionManager";
2022-03-09 20:58:21 +00:00
import logo from "../img/ntfy.svg"
2022-02-25 17:46:22 +00:00
const ActionBar = (props) => {
2022-03-04 21:10:04 +00:00
const location = useLocation();
let title = "ntfy";
if (props.selected) {
title = topicShortUrl(props.selected.baseUrl, props.selected.topic);
2022-03-04 21:10:04 +00:00
} else if (location.pathname === "/settings") {
title = "Settings";
}
2022-02-25 17:46:22 +00:00
return (
2022-02-26 19:22:21 +00:00
<AppBar position="fixed" sx={{
width: '100%',
2022-02-26 19:36:23 +00:00
zIndex: { sm: 1250 }, // > Navigation (1200), but < Dialog (1300)
2022-02-26 19:22:21 +00:00
ml: { sm: `${Navigation.width}px` }
}}>
2022-02-25 17:46:22 +00:00
<Toolbar sx={{pr: '24px'}}>
<IconButton
color="inherit"
edge="start"
onClick={props.onMobileDrawerToggle}
sx={{ mr: 2, display: { sm: 'none' } }}
>
<MenuIcon />
</IconButton>
2022-03-09 20:58:21 +00:00
<Box component="img" src={logo} sx={{
2022-02-28 21:56:38 +00:00
display: { xs: 'none', sm: 'block' },
marginRight: '10px',
height: '28px'
}}/>
2022-02-26 19:22:21 +00:00
<Typography variant="h6" noWrap component="div" sx={{ flexGrow: 1 }}>
{title}
</Typography>
2022-03-08 21:56:41 +00:00
{props.selected &&
<SettingsIcons
subscription={props.selected}
onUnsubscribe={props.onUnsubscribe}
/>}
2022-02-25 17:46:22 +00:00
</Toolbar>
</AppBar>
);
};
2022-03-06 03:33:34 +00:00
// Originally from https://mui.com/components/menus/#MenuListComposition.js
2022-03-08 21:56:41 +00:00
const SettingsIcons = (props) => {
2022-03-06 03:33:34 +00:00
const navigate = useNavigate();
const [open, setOpen] = useState(false);
const anchorRef = useRef(null);
2022-03-08 21:56:41 +00:00
const subscription = props.subscription;
2022-03-06 03:33:34 +00:00
2022-03-08 21:56:41 +00:00
const handleToggleOpen = () => {
2022-03-06 03:33:34 +00:00
setOpen((prevOpen) => !prevOpen);
};
2022-03-08 21:56:41 +00:00
const handleToggleMute = async () => {
const mutedUntil = (subscription.mutedUntil) ? 0 : 1; // Make this a timestamp in the future
await subscriptionManager.setMutedUntil(subscription.id, mutedUntil);
}
2022-03-06 03:33:34 +00:00
const handleClose = (event) => {
if (anchorRef.current && anchorRef.current.contains(event.target)) {
return;
}
setOpen(false);
};
const handleClearAll = async (event) => {
handleClose(event);
console.log(`[ActionBar] Deleting all notifications from ${props.subscription.id}`);
await subscriptionManager.deleteNotifications(props.subscription.id);
};
const handleUnsubscribe = async (event) => {
console.log(`[ActionBar] Unsubscribing from ${props.subscription.id}`);
handleClose(event);
await subscriptionManager.remove(props.subscription.id);
const newSelected = await subscriptionManager.first(); // May be undefined
if (newSelected) {
navigate(routes.forSubscription(newSelected));
2022-03-06 21:35:31 +00:00
} else {
navigate(routes.root);
2022-03-06 03:33:34 +00:00
}
};
const handleSendTestMessage = () => {
const baseUrl = props.subscription.baseUrl;
const topic = props.subscription.topic;
api.publish(baseUrl, topic,
`This is a test notification sent by the ntfy Web UI at ${new Date().toString()}.`); // FIXME result ignored
setOpen(false);
}
const handleListKeyDown = (event) => {
if (event.key === 'Tab') {
event.preventDefault();
setOpen(false);
} else if (event.key === 'Escape') {
setOpen(false);
}
}
// return focus to the button when we transitioned from !open -> open
const prevOpen = useRef(open);
useEffect(() => {
if (prevOpen.current === true && open === false) {
anchorRef.current.focus();
}
prevOpen.current = open;
}, [open]);
return (
<>
2022-03-08 21:56:41 +00:00
<IconButton color="inherit" size="large" edge="end" onClick={handleToggleMute} sx={{marginRight: 0}}>
{subscription.mutedUntil ? <NotificationsOffIcon/> : <NotificationsIcon/>}
</IconButton>
<IconButton color="inherit" size="large" edge="end" ref={anchorRef} onClick={handleToggleOpen}>
2022-03-06 03:33:34 +00:00
<MoreVertIcon/>
</IconButton>
<Popper
open={open}
anchorEl={anchorRef.current}
role={undefined}
placement="bottom-start"
transition
disablePortal
>
{({TransitionProps, placement}) => (
<Grow
{...TransitionProps}
2022-03-08 21:56:41 +00:00
style={{transformOrigin: placement === 'bottom-start' ? 'left top' : 'left bottom'}}
2022-03-06 03:33:34 +00:00
>
<Paper>
<ClickAwayListener onClickAway={handleClose}>
2022-03-09 20:58:21 +00:00
<MenuList autoFocusItem={open} onKeyDown={handleListKeyDown}>
2022-03-06 03:33:34 +00:00
<MenuItem onClick={handleSendTestMessage}>Send test notification</MenuItem>
<MenuItem onClick={handleClearAll}>Clear all notifications</MenuItem>
<MenuItem onClick={handleUnsubscribe}>Unsubscribe</MenuItem>
</MenuList>
</ClickAwayListener>
</Paper>
</Grow>
)}
</Popper>
</>
);
};
2022-02-25 17:46:22 +00:00
export default ActionBar;