mirror of
https://github.com/idanoo/autobrr
synced 2025-07-23 08:49:13 +00:00

* fix(auth): invalid cookie handling and wrongful basic auth invalidation * fix(auth): fix test to reflect new HTTP status code * fix(auth/web): do not throw on error * fix(http): replace http codes in middleware to prevent basic auth invalidation fix typo in comment * fix test * fix(web): api client handle 403 * refactor(http): auth_test use testify.assert * refactor(http): set session opts after valid login * refactor(http): send more client headers * fix(http): test * refactor(web): move router to tanstack/router * refactor(web): use route loaders and suspense * refactor(web): useSuspense for settings * refactor(web): invalidate cookie in middleware * fix: loclfile * fix: load filter/id * fix(web): login, onboard, types, imports * fix(web): filter load * fix(web): build errors * fix(web): ts-expect-error * fix(tests): filter_test.go * fix(filters): tests * refactor: remove duplicate spinner components refactor: ReleaseTable.tsx loading animation refactor: remove dedicated `pendingComponent` for `settingsRoute` * fix: refactor missed SectionLoader to RingResizeSpinner * fix: substitute divides with borders to account for unloaded elements * fix(api): action status URL param * revert: action status URL param add comment * fix(routing): notfound handling and split files * fix(filters): notfound get params * fix(queries): colon * fix(queries): comments ts-ignore * fix(queries): extract queryKeys * fix(queries): remove err * fix(routes): move zob schema inline * fix(auth): middleware and redirect to login * fix(auth): failing test * fix(logs): invalidate correct key * fix(logs): invalidate correct key * fix(logs): invalidate correct key * fix: JSX element stealing focus from searchbar * reimplement empty release table state text * fix(context): use deep-copy * fix(releases): empty state and filter input warnings * fix(releases): empty states * fix(auth): onboarding * fix(cache): invalidate queries --------- Co-authored-by: ze0s <43699394+zze0s@users.noreply.github.com>
144 lines
6.1 KiB
TypeScript
144 lines
6.1 KiB
TypeScript
/*
|
|
* Copyright (c) 2021 - 2024, Ludvig Lundgren and the autobrr contributors.
|
|
* SPDX-License-Identifier: GPL-2.0-or-later
|
|
*/
|
|
|
|
import { useMutation, useQueryClient, useSuspenseQuery } from "@tanstack/react-query";
|
|
import { PlusIcon } from "@heroicons/react/24/solid";
|
|
import toast from "react-hot-toast";
|
|
|
|
import { APIClient } from "@api/APIClient";
|
|
import { NotificationKeys } from "@api/query_keys";
|
|
import { NotificationsQueryOptions } from "@api/queries";
|
|
import { EmptySimple } from "@components/emptystates";
|
|
import { useToggle } from "@hooks/hooks";
|
|
import { NotificationAddForm, NotificationUpdateForm } from "@forms/settings/NotificationForms";
|
|
import { componentMapType } from "@forms/settings/DownloadClientForms";
|
|
import Toast from "@components/notifications/Toast";
|
|
import {
|
|
DiscordIcon,
|
|
GotifyIcon,
|
|
LunaSeaIcon,
|
|
NotifiarrIcon,
|
|
NtfyIcon,
|
|
PushoverIcon,
|
|
Section,
|
|
TelegramIcon
|
|
} from "./_components";
|
|
import { Checkbox } from "@components/Checkbox";
|
|
|
|
function NotificationSettings() {
|
|
const [addNotificationsIsOpen, toggleAddNotifications] = useToggle(false);
|
|
|
|
const notificationsQuery = useSuspenseQuery(NotificationsQueryOptions())
|
|
|
|
return (
|
|
<Section
|
|
title="Notifications"
|
|
description="Send notifications on events."
|
|
rightSide={
|
|
<button
|
|
type="button"
|
|
onClick={toggleAddNotifications}
|
|
className="relative inline-flex items-center px-4 py-2 border border-transparent shadow-sm text-sm font-medium rounded-md text-white bg-blue-600 dark:bg-blue-600 hover:bg-blue-700 dark:hover:bg-blue-700 focus:outline-none focus:ring-2 focus:ring-offset-2 focus:ring-blue-500"
|
|
>
|
|
<PlusIcon className="h-5 w-5 mr-1" />
|
|
Add new
|
|
</button>
|
|
}
|
|
>
|
|
<NotificationAddForm isOpen={addNotificationsIsOpen} toggle={toggleAddNotifications} />
|
|
|
|
{notificationsQuery.data && notificationsQuery.data.length > 0 ? (
|
|
<ul className="min-w-full">
|
|
<li className="grid grid-cols-12 border-b border-gray-200 dark:border-gray-700">
|
|
<div className="col-span-2 sm:col-span-1 pl-1 sm:pl-4 py-3 text-left text-xs font-medium text-gray-500 dark:text-gray-400 uppercase tracking-wider">Enabled</div>
|
|
<div className="col-span-6 pl-10 sm:pl-12 pr-6 py-3 text-left text-xs font-medium text-gray-500 dark:text-gray-400 uppercase tracking-wider">Name</div>
|
|
<div className="hidden md:flex col-span-2 py-3 text-left text-xs font-medium text-gray-500 dark:text-gray-400 uppercase tracking-wider">Type</div>
|
|
<div className="hidden md:flex col-span-3 px-6 py-3 text-left text-xs font-medium text-gray-500 dark:text-gray-400 uppercase tracking-wider">Events</div>
|
|
</li>
|
|
|
|
{notificationsQuery.data.map((n) => <ListItem key={n.id} notification={n} />)}
|
|
</ul>
|
|
) : (
|
|
<EmptySimple title="No notifications" subtitle="" buttonText="Create new notification" buttonAction={toggleAddNotifications} />
|
|
)}
|
|
</Section>
|
|
);
|
|
}
|
|
|
|
const iconStyle = "flex items-center px-2 py-0.5 rounded bg-gray-200 dark:bg-gray-700 text-gray-800 dark:text-gray-400";
|
|
const iconComponentMap: componentMapType = {
|
|
DISCORD: <span className={iconStyle}><DiscordIcon /> Discord</span>,
|
|
NOTIFIARR: <span className={iconStyle}><NotifiarrIcon /> Notifiarr</span>,
|
|
TELEGRAM: <span className={iconStyle}><TelegramIcon /> Telegram</span>,
|
|
PUSHOVER: <span className={iconStyle}><PushoverIcon /> Pushover</span>,
|
|
GOTIFY: <span className={iconStyle}><GotifyIcon /> Gotify</span>,
|
|
NTFY: <span className={iconStyle}><NtfyIcon /> ntfy</span>,
|
|
SHOUTRRR: <span className={iconStyle}><NtfyIcon /> Shoutrrr</span>,
|
|
LUNASEA: <span className={iconStyle}><LunaSeaIcon /> LunaSea</span>
|
|
};
|
|
|
|
interface ListItemProps {
|
|
notification: ServiceNotification;
|
|
}
|
|
|
|
function ListItem({ notification }: ListItemProps) {
|
|
const [updateFormIsOpen, toggleUpdateForm] = useToggle(false);
|
|
|
|
const queryClient = useQueryClient();
|
|
|
|
const mutation = useMutation({
|
|
mutationFn: (notification: ServiceNotification) => APIClient.notifications.update(notification).then(() => notification),
|
|
onSuccess: (notification: ServiceNotification) => {
|
|
toast.custom(t => <Toast type="success" body={`${notification.name} was ${notification.enabled ? "enabled" : "disabled"} successfully.`} t={t} />);
|
|
queryClient.invalidateQueries({ queryKey: NotificationKeys.lists() });
|
|
}
|
|
});
|
|
|
|
const onToggleMutation = (newState: boolean) => {
|
|
mutation.mutate({
|
|
...notification,
|
|
enabled: newState
|
|
});
|
|
};
|
|
|
|
return (
|
|
<li key={notification.id} className="text-gray-500 dark:text-gray-400">
|
|
<NotificationUpdateForm isOpen={updateFormIsOpen} toggle={toggleUpdateForm} notification={notification} />
|
|
|
|
<div className="grid grid-cols-12 items-center py-2">
|
|
<div className="col-span-2 sm:col-span-1 pl-1 py-0.5 sm:pl-6 flex items-center">
|
|
<Checkbox
|
|
value={notification.enabled}
|
|
setValue={onToggleMutation}
|
|
/>
|
|
</div>
|
|
<div className="col-span-8 md:col-span-6 pl-10 sm:pl-12 pr-2 sm:pr-6 truncate block items-center text-sm font-medium text-gray-900 dark:text-white" title={notification.name}>
|
|
{notification.name}
|
|
</div>
|
|
<div className="hidden md:flex col-span-2 items-center">
|
|
{iconComponentMap[notification.type]}
|
|
</div>
|
|
<div className="hidden md:flex col-span-2 px-6 items-center sm:px-6">
|
|
<span
|
|
className="mr-2 inline-flex items-center px-2.5 py-1 rounded-md text-sm font-medium bg-gray-200 dark:bg-gray-700 text-gray-800 dark:text-gray-400"
|
|
title={notification.events.join(", ")}
|
|
>
|
|
{notification.events.length}
|
|
</span>
|
|
</div>
|
|
<div className="col-span-1 flex first-letter:px-6 whitespace-nowrap text-right text-sm font-medium">
|
|
<span
|
|
className="col-span-1 px-0 sm:px-6 text-blue-600 dark:text-gray-300 hover:text-blue-900 dark:hover:text-blue-500 cursor-pointer"
|
|
onClick={toggleUpdateForm}
|
|
>
|
|
Edit
|
|
</span>
|
|
</div>
|
|
</div>
|
|
</li>
|
|
);
|
|
}
|
|
|
|
export default NotificationSettings;
|