mirror of
https://github.com/muerwre/vault-frontend.git
synced 2025-04-24 20:36:40 +07:00
added notification settings
This commit is contained in:
parent
23701a5261
commit
ad85f71a4f
10 changed files with 193 additions and 21 deletions
17
src/api/notifications/settings.ts
Normal file
17
src/api/notifications/settings.ts
Normal file
|
@ -0,0 +1,17 @@
|
||||||
|
import { API } from '~/constants/api';
|
||||||
|
import { api, cleanResult } from '~/utils/api';
|
||||||
|
|
||||||
|
import {
|
||||||
|
ApiGetNotificationSettingsResponse,
|
||||||
|
ApiGetNotificationsResponse,
|
||||||
|
} from './types';
|
||||||
|
|
||||||
|
export const apiGetNotificationSettings = () =>
|
||||||
|
api
|
||||||
|
.get<ApiGetNotificationSettingsResponse>(API.NOTIFICATIONS.SETTINGS)
|
||||||
|
.then(cleanResult);
|
||||||
|
|
||||||
|
export const apiGetNotifications = () =>
|
||||||
|
api
|
||||||
|
.get<ApiGetNotificationsResponse>(API.NOTIFICATIONS.LIST)
|
||||||
|
.then(cleanResult);
|
13
src/api/notifications/types.ts
Normal file
13
src/api/notifications/types.ts
Normal file
|
@ -0,0 +1,13 @@
|
||||||
|
import { NotificationItem } from '~/types/notifications';
|
||||||
|
|
||||||
|
export interface ApiGetNotificationSettingsResponse {
|
||||||
|
enabled: boolean;
|
||||||
|
flow: boolean;
|
||||||
|
comments: boolean;
|
||||||
|
last_seen?: string | null;
|
||||||
|
last_date?: string | null;
|
||||||
|
}
|
||||||
|
|
||||||
|
export interface ApiGetNotificationsResponse {
|
||||||
|
items?: NotificationItem[];
|
||||||
|
}
|
|
@ -62,4 +62,8 @@ export const API = {
|
||||||
STATS: '/nodes/lab/stats',
|
STATS: '/nodes/lab/stats',
|
||||||
UPDATES: '/nodes/lab/updates',
|
UPDATES: '/nodes/lab/updates',
|
||||||
},
|
},
|
||||||
|
NOTIFICATIONS: {
|
||||||
|
LIST: '/notifications/',
|
||||||
|
SETTINGS: '/notifications/settings',
|
||||||
|
},
|
||||||
};
|
};
|
||||||
|
|
28
src/hooks/notifications/useNotificationSettings.ts
Normal file
28
src/hooks/notifications/useNotificationSettings.ts
Normal file
|
@ -0,0 +1,28 @@
|
||||||
|
import { isAfter, isValid, parse, parseISO } from 'date-fns';
|
||||||
|
|
||||||
|
import { useAuth } from '../auth/useAuth';
|
||||||
|
|
||||||
|
import { useNotificationSettingsRequest } from './useNotificationSettingsRequest';
|
||||||
|
|
||||||
|
export const useNotificationSettings = () => {
|
||||||
|
const { isUser } = useAuth();
|
||||||
|
|
||||||
|
const {
|
||||||
|
error: settingsError,
|
||||||
|
enabled: settingsEnabled,
|
||||||
|
lastSeen,
|
||||||
|
lastDate,
|
||||||
|
isLoading: isLoadingSettings,
|
||||||
|
} = useNotificationSettingsRequest();
|
||||||
|
|
||||||
|
const enabled = !isLoadingSettings && !settingsError && settingsEnabled;
|
||||||
|
|
||||||
|
const hasNew =
|
||||||
|
enabled && !!lastDate && (!lastSeen || isAfter(lastDate, lastSeen));
|
||||||
|
|
||||||
|
return {
|
||||||
|
enabled,
|
||||||
|
hasNew,
|
||||||
|
available: isUser,
|
||||||
|
};
|
||||||
|
};
|
36
src/hooks/notifications/useNotificationSettingsRequest.ts
Normal file
36
src/hooks/notifications/useNotificationSettingsRequest.ts
Normal file
|
@ -0,0 +1,36 @@
|
||||||
|
import { isValid, parseISO } from 'date-fns';
|
||||||
|
import useSWR from 'swr';
|
||||||
|
|
||||||
|
import { apiGetNotificationSettings } from '~/api/notifications/settings';
|
||||||
|
import { API } from '~/constants/api';
|
||||||
|
|
||||||
|
import { useAuth } from '../auth/useAuth';
|
||||||
|
|
||||||
|
const refreshInterval = 60e3; // 1min
|
||||||
|
|
||||||
|
export const useNotificationSettingsRequest = () => {
|
||||||
|
const { isUser } = useAuth();
|
||||||
|
const {
|
||||||
|
data,
|
||||||
|
isValidating: isLoading,
|
||||||
|
error,
|
||||||
|
} = useSWR(
|
||||||
|
isUser ? API.NOTIFICATIONS.SETTINGS : null,
|
||||||
|
async () => apiGetNotificationSettings(),
|
||||||
|
{ refreshInterval },
|
||||||
|
);
|
||||||
|
|
||||||
|
return {
|
||||||
|
isLoading,
|
||||||
|
error,
|
||||||
|
lastSeen:
|
||||||
|
data?.last_seen && isValid(parseISO(data.last_seen))
|
||||||
|
? parseISO(data?.last_seen)
|
||||||
|
: undefined,
|
||||||
|
lastDate:
|
||||||
|
data?.last_date && isValid(parseISO(data.last_date))
|
||||||
|
? parseISO(data?.last_date)
|
||||||
|
: undefined,
|
||||||
|
enabled: !!data?.enabled && (data.flow || data.comments),
|
||||||
|
};
|
||||||
|
};
|
20
src/hooks/notifications/useNotificationsList.ts
Normal file
20
src/hooks/notifications/useNotificationsList.ts
Normal file
|
@ -0,0 +1,20 @@
|
||||||
|
import useSWR from 'swr';
|
||||||
|
|
||||||
|
import { apiGetNotifications } from '~/api/notifications/settings';
|
||||||
|
import { API } from '~/constants/api';
|
||||||
|
|
||||||
|
import { useAuth } from '../auth/useAuth';
|
||||||
|
|
||||||
|
export const useNotificationsList = () => {
|
||||||
|
const { isUser } = useAuth();
|
||||||
|
|
||||||
|
const {
|
||||||
|
data,
|
||||||
|
isValidating: isLoading,
|
||||||
|
error,
|
||||||
|
} = useSWR(isUser ? API.NOTIFICATIONS.LIST : null, async () =>
|
||||||
|
apiGetNotifications(),
|
||||||
|
);
|
||||||
|
|
||||||
|
return { isLoading, error, ...data };
|
||||||
|
};
|
|
@ -1,5 +1,3 @@
|
||||||
import React from 'react';
|
|
||||||
|
|
||||||
import App from 'next/app';
|
import App from 'next/app';
|
||||||
import Head from 'next/head';
|
import Head from 'next/head';
|
||||||
|
|
||||||
|
@ -16,14 +14,15 @@ import { UserContextProvider } from '~/utils/context/UserContextProvider';
|
||||||
import { AudioPlayerProvider } from '~/utils/providers/AudioPlayerProvider';
|
import { AudioPlayerProvider } from '~/utils/providers/AudioPlayerProvider';
|
||||||
import { AuthProvider } from '~/utils/providers/AuthProvider';
|
import { AuthProvider } from '~/utils/providers/AuthProvider';
|
||||||
import { MetadataProvider } from '~/utils/providers/MetadataProvider';
|
import { MetadataProvider } from '~/utils/providers/MetadataProvider';
|
||||||
|
import { NotificationProvider } from '~/utils/providers/NotificationProvider';
|
||||||
import { SWRConfigProvider } from '~/utils/providers/SWRConfigProvider';
|
import { SWRConfigProvider } from '~/utils/providers/SWRConfigProvider';
|
||||||
import { SearchProvider } from '~/utils/providers/SearchProvider';
|
import { SearchProvider } from '~/utils/providers/SearchProvider';
|
||||||
import { SidebarProvider } from '~/utils/providers/SidebarProvider';
|
import { SidebarProvider } from '~/utils/providers/SidebarProvider';
|
||||||
import { ThemeProvider } from '~/utils/providers/ThemeProvider';
|
import { ThemeProvider } from '~/utils/providers/ThemeProvider';
|
||||||
import { ToastProvider } from '~/utils/providers/ToastProvider';
|
import { ToastProvider } from '~/utils/providers/ToastProvider';
|
||||||
|
|
||||||
import '~/styles/main.scss';
|
|
||||||
import 'tippy.js/dist/tippy.css';
|
import 'tippy.js/dist/tippy.css';
|
||||||
|
import '~/styles/main.scss';
|
||||||
|
|
||||||
const mobxStore = getMOBXStore();
|
const mobxStore = getMOBXStore();
|
||||||
|
|
||||||
|
@ -45,26 +44,28 @@ export default class MyApp extends App {
|
||||||
<AudioPlayerProvider>
|
<AudioPlayerProvider>
|
||||||
<MetadataProvider>
|
<MetadataProvider>
|
||||||
<AuthProvider>
|
<AuthProvider>
|
||||||
<SidebarProvider>
|
<NotificationProvider>
|
||||||
<Head>
|
<SidebarProvider>
|
||||||
<meta
|
<Head>
|
||||||
name="viewport"
|
<meta
|
||||||
content="width=device-width, initial-scale=1.0, minimum-scale=1.0, user-scalable=0"
|
name="viewport"
|
||||||
/>
|
content="width=device-width, initial-scale=1.0, minimum-scale=1.0, user-scalable=0"
|
||||||
|
/>
|
||||||
|
|
||||||
{!!canonicalURL && (
|
{!!canonicalURL && (
|
||||||
<link rel="canonical" href={canonicalURL} />
|
<link rel="canonical" href={canonicalURL} />
|
||||||
)}
|
)}
|
||||||
</Head>
|
</Head>
|
||||||
|
|
||||||
<MainLayout>
|
<MainLayout>
|
||||||
<ToastProvider />
|
<ToastProvider />
|
||||||
<Modal />
|
<Modal />
|
||||||
<Sprites />
|
<Sprites />
|
||||||
<Component {...pageProps} />
|
<Component {...pageProps} />
|
||||||
</MainLayout>
|
</MainLayout>
|
||||||
<BottomContainer />
|
<BottomContainer />
|
||||||
</SidebarProvider>
|
</SidebarProvider>
|
||||||
|
</NotificationProvider>
|
||||||
</AuthProvider>
|
</AuthProvider>
|
||||||
</MetadataProvider>
|
</MetadataProvider>
|
||||||
</AudioPlayerProvider>
|
</AudioPlayerProvider>
|
||||||
|
|
|
@ -28,3 +28,9 @@ export interface ISocialAccount {
|
||||||
name: string;
|
name: string;
|
||||||
photo: string;
|
photo: string;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
export interface ShallowUser {
|
||||||
|
id: number;
|
||||||
|
username: string;
|
||||||
|
photo: string;
|
||||||
|
}
|
||||||
|
|
16
src/types/notifications/index.ts
Normal file
16
src/types/notifications/index.ts
Normal file
|
@ -0,0 +1,16 @@
|
||||||
|
import { ShallowUser } from '../auth';
|
||||||
|
|
||||||
|
export interface NotificationItem {
|
||||||
|
id: number;
|
||||||
|
url: string;
|
||||||
|
type: NotificationType;
|
||||||
|
text: string;
|
||||||
|
user: ShallowUser;
|
||||||
|
thumbnail: string;
|
||||||
|
created_at: string;
|
||||||
|
}
|
||||||
|
|
||||||
|
export enum NotificationType {
|
||||||
|
Node = 'node',
|
||||||
|
Comment = 'comment',
|
||||||
|
}
|
31
src/utils/providers/NotificationProvider.tsx
Normal file
31
src/utils/providers/NotificationProvider.tsx
Normal file
|
@ -0,0 +1,31 @@
|
||||||
|
import { createContext, FC, useContext } from 'react';
|
||||||
|
|
||||||
|
import { observer } from 'mobx-react-lite';
|
||||||
|
|
||||||
|
import { useNotificationSettings } from '~/hooks/notifications/useNotificationSettings';
|
||||||
|
|
||||||
|
interface NotificationProviderProps {}
|
||||||
|
|
||||||
|
const defaultValue = {
|
||||||
|
available: false,
|
||||||
|
enabled: false,
|
||||||
|
hasNew: false,
|
||||||
|
};
|
||||||
|
|
||||||
|
const NotificationContext = createContext(defaultValue);
|
||||||
|
|
||||||
|
const NotificationProvider: FC<NotificationProviderProps> = observer(
|
||||||
|
({ children }) => {
|
||||||
|
const value = useNotificationSettings();
|
||||||
|
|
||||||
|
return (
|
||||||
|
<NotificationContext.Provider value={value}>
|
||||||
|
{children}
|
||||||
|
</NotificationContext.Provider>
|
||||||
|
);
|
||||||
|
},
|
||||||
|
);
|
||||||
|
|
||||||
|
export const useNotifications = () => useContext(NotificationContext);
|
||||||
|
|
||||||
|
export { NotificationProvider };
|
Loading…
Add table
Add a link
Reference in a new issue