2021-10-27 09:38:10 -05:00
|
|
|
// @flow
|
2021-11-01 13:51:23 -05:00
|
|
|
import React, { useEffect, useState, useMemo } from 'react';
|
|
|
|
import pushNotifications from '$web/src/push-notifications';
|
|
|
|
import { BrowserNotificationErrorModal } from '$web/component/browserNotificationHints';
|
2021-10-27 09:38:10 -05:00
|
|
|
|
|
|
|
// @todo: Once we are on Redux 7 we should have proper hooks we can use here for store access.
|
|
|
|
import { store } from '$ui/store';
|
|
|
|
import { selectUser } from 'redux/selectors/user';
|
2021-11-24 19:38:21 +08:00
|
|
|
import analytics, { GA_DIMENSIONS } from 'analytics';
|
2021-10-27 09:38:10 -05:00
|
|
|
|
|
|
|
export default () => {
|
|
|
|
const [pushPermission, setPushPermission] = useState(window.Notification?.permission);
|
|
|
|
const [subscribed, setSubscribed] = useState(false);
|
|
|
|
const [pushEnabled, setPushEnabled] = useState(false);
|
2021-11-10 13:15:40 -06:00
|
|
|
const [pushSupported, setPushSupported] = useState(true);
|
2021-11-01 13:51:23 -05:00
|
|
|
const [encounteredError, setEncounteredError] = useState(false);
|
2021-11-12 11:06:07 -06:00
|
|
|
const [pushInitialized, setPushInitialized] = useState(false);
|
2021-10-27 09:38:10 -05:00
|
|
|
|
|
|
|
const [user] = useState(selectUser(store.getState()));
|
|
|
|
|
|
|
|
useEffect(() => {
|
2021-12-29 13:03:05 -06:00
|
|
|
if (!user) return;
|
2021-11-01 13:51:23 -05:00
|
|
|
setPushSupported(pushNotifications.supported);
|
|
|
|
if (pushNotifications.supported) {
|
2021-11-12 11:06:07 -06:00
|
|
|
pushNotifications.subscribed(user.id).then((isSubscribed: boolean) => {
|
|
|
|
setSubscribed(isSubscribed);
|
|
|
|
setPushInitialized(true);
|
|
|
|
});
|
2021-11-01 13:51:23 -05:00
|
|
|
}
|
2021-10-27 09:38:10 -05:00
|
|
|
}, [user]);
|
|
|
|
|
|
|
|
useMemo(() => setPushEnabled(pushPermission === 'granted' && subscribed), [pushPermission, subscribed]);
|
|
|
|
|
|
|
|
const subscribe = async () => {
|
2021-12-29 13:03:05 -06:00
|
|
|
if (!user) return;
|
2021-11-01 13:51:23 -05:00
|
|
|
setEncounteredError(false);
|
|
|
|
try {
|
2021-12-29 13:26:28 -06:00
|
|
|
const subscribed = await pushNotifications.subscribe(user.id);
|
|
|
|
if (subscribed) {
|
2021-11-01 13:51:23 -05:00
|
|
|
setSubscribed(true);
|
|
|
|
setPushPermission(window.Notification?.permission);
|
|
|
|
}
|
|
|
|
} catch {
|
|
|
|
setEncounteredError(true);
|
2021-11-24 19:38:21 +08:00
|
|
|
analytics.reportEvent('browser_notification', { [GA_DIMENSIONS.ACTION]: 'subscribe_failed' });
|
2021-10-27 09:38:10 -05:00
|
|
|
}
|
2021-12-29 13:26:28 -06:00
|
|
|
analytics.reportEvent('browser_notification', { [GA_DIMENSIONS.ACTION]: 'subscribed' });
|
2021-10-27 09:38:10 -05:00
|
|
|
};
|
|
|
|
|
|
|
|
const unsubscribe = async () => {
|
2021-12-29 13:03:05 -06:00
|
|
|
if (!user) return;
|
2021-11-01 13:51:23 -05:00
|
|
|
if (await pushNotifications.unsubscribe(user.id)) {
|
2021-10-27 09:38:10 -05:00
|
|
|
setSubscribed(false);
|
2021-11-24 19:38:21 +08:00
|
|
|
analytics.reportEvent('browser_notification', { [GA_DIMENSIONS.ACTION]: 'unsubscribed' });
|
2021-10-27 09:38:10 -05:00
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
const pushToggle = async () => {
|
|
|
|
return !pushEnabled ? subscribe() : unsubscribe();
|
|
|
|
};
|
|
|
|
|
|
|
|
const pushRequest = async () => {
|
|
|
|
return window.Notification?.permission !== 'granted' ? subscribe() : null;
|
|
|
|
};
|
|
|
|
|
2021-11-01 13:51:23 -05:00
|
|
|
const pushErrorModal = () => {
|
|
|
|
return <>{encounteredError && <BrowserNotificationErrorModal doHideModal={() => setEncounteredError(false)} />}</>;
|
2021-10-27 09:38:10 -05:00
|
|
|
};
|
|
|
|
|
|
|
|
return {
|
2021-11-12 11:06:07 -06:00
|
|
|
pushInitialized,
|
2021-10-27 09:38:10 -05:00
|
|
|
pushSupported,
|
|
|
|
pushEnabled,
|
|
|
|
pushPermission,
|
|
|
|
pushToggle,
|
|
|
|
pushRequest,
|
2021-11-01 13:51:23 -05:00
|
|
|
pushErrorModal,
|
2021-10-27 09:38:10 -05:00
|
|
|
};
|
|
|
|
};
|