2023-07-21 22:33:11 +00:00
import { useState , useEffect , useMemo } from 'react'
import { useApolloClient , useQuery } from '@apollo/client'
2021-08-17 18:15:24 +00:00
import Comment , { CommentSkeleton } from './comment'
2022-07-21 22:55:05 +00:00
import Item from './item'
import ItemJob from './item-job'
2023-07-21 22:33:11 +00:00
import { HAS _NOTIFICATIONS , NOTIFICATIONS } from '../fragments/notifications'
2021-09-30 15:46:58 +00:00
import MoreFooter from './more-footer'
2022-01-19 21:02:38 +00:00
import Invite from './invite'
2022-01-20 19:03:48 +00:00
import { ignoreClick } from '../lib/clicks'
2022-03-31 16:49:35 +00:00
import { timeSince } from '../lib/time'
2022-03-17 20:13:19 +00:00
import Link from 'next/link'
2022-03-23 18:54:39 +00:00
import Check from '../svgs/check-double-line.svg'
2022-03-23 21:45:36 +00:00
import HandCoin from '../svgs/hand-coin-fill.svg'
2022-05-17 22:09:15 +00:00
import { COMMENT _DEPTH _LIMIT } from '../lib/constants'
2023-02-01 14:44:35 +00:00
import CowboyHatIcon from '../svgs/cowboy.svg'
import BaldIcon from '../svgs/bald.svg'
2023-05-06 21:51:17 +00:00
import { RootProvider } from './root'
2023-06-12 18:03:44 +00:00
import { Alert } from 'react-bootstrap'
import styles from './notifications.module.css'
Service worker rework, Web Target Share API & Web Push API (#324)
* npm uninstall next-pwa
next-pwa was last updated in August 2022.
There is also an issue which mentions that next-pwa is abandoned (?): https://github.com/shadowwalker/next-pwa/issues/482
But the main reason for me uninstalling it is that it adds a lot of preconfigured stuff which is not necessary for us.
It even lead to a bug since pages were cached without our knowledge.
So I will go with a different PWA approach. This different approach should do the following:
- make it more transparent what the service worker is doing
- gives us more control to configure the service worker and thus making it easier
* Use workbox-webpack-plugin
Every other plugin (`next-offline`, `next-workbox-webpack-plugin`, `next-with-workbox`, ...) added unnecessary configuration which felt contrary to how PWAs should be built.
(PWAs should progressivly enhance the website in small steps, see https://web.dev/learn/pwa/getting-started/#focus-on-a-feature)
These default configurations even lead to worse UX since they made invalid assumptions about stacker.news:
We _do not_ want to cache our start url and we _do not_ want to cache anything unless explicitly told to.
Almost every page on SN should be fresh for the best UX.
To achieve this, by default, the service worker falls back to the network (as if the service worker wasn't there).
Therefore, this should be the simplest configuration with a valid precache and cache busting support.
In the future, we can try to use prefetching to improve performance of navigation requests.
* Add support for Web Share Target API
See https://developer.chrome.com/articles/web-share-target/
* Use Web Push API for push notifications
I followed this (very good!) guide: https://web.dev/notifications/
* Refactor code related to Web Push
* Send push notification to users on events
* Merge notifications
* Send notification to author of every parent recursively
* Remove unused userId param in savePushSubscription
As it should be, the user id is retrieved from the authenticated user in the backend.
* Resubscribe user if push subscription changed
* Update old subscription if oldEndpoint was given
* Allow users to unsubscribe
* Use LTREE operator instead of recursive query
* Always show checkbox for push notifications
* Justify checkbox to end
* Update title of first push notification
* Fix warning from uncontrolled to controlled
* Add comment about Notification.requestPermission
* Fix timestamp
* Catch error on push subscription toggle
* Wrap function bodies in try/catch
* Use Promise.allSettled
* Filter subscriptions by user notification settings
* Fix user notification filter
* Use skipWaiting
---------
Co-authored-by: ekzyis <ek@stacker.news>
2023-07-04 19:36:07 +00:00
import { useServiceWorker } from './serviceworker'
import { Checkbox , Form } from './form'
2023-07-21 22:33:11 +00:00
import { useRouter } from 'next/router'
2021-08-17 18:15:24 +00:00
2021-09-06 22:36:08 +00:00
function Notification ( { n } ) {
2023-06-01 00:51:30 +00:00
switch ( n . _ _typename ) {
case 'Earn' : return < EarnNotification n = { n } / >
case 'Invitification' : return < Invitification n = { n } / >
case 'InvoicePaid' : return < InvoicePaid n = { n } / >
case 'Referral' : return < Referral n = { n } / >
case 'Streak' : return < Streak n = { n } / >
case 'Votification' : return < Votification n = { n } / >
case 'Mention' : return < Mention n = { n } / >
case 'JobChanged' : return < JobChanged n = { n } / >
case 'Reply' : return < Reply n = { n } / >
}
2023-06-01 18:22:39 +00:00
console . error ( '__typename not supported:' , n . _ _typename )
2023-06-01 00:51:30 +00:00
return null
}
2022-01-20 19:03:48 +00:00
2023-07-21 22:33:11 +00:00
function NotificationLayout ( { children , href , as } ) {
const router = useRouter ( )
2023-06-01 00:51:30 +00:00
return (
2023-06-01 18:22:39 +00:00
< div
2023-07-21 22:33:11 +00:00
className = 'clickToContext'
onClick = { ( e ) => ! ignoreClick ( e ) && router . push ( href , as ) }
2023-06-01 18:22:39 +00:00
>
2023-06-01 00:51:30 +00:00
{ children }
2023-02-01 14:44:35 +00:00
< / d i v >
)
}
2023-07-21 22:33:11 +00:00
const defaultOnClick = n => {
2023-06-01 00:51:30 +00:00
if ( ! n . item . title ) {
2023-07-06 15:00:38 +00:00
const path = n . item . path . split ( '.' )
if ( path . length > COMMENT _DEPTH _LIMIT + 1 ) {
const rootId = path . slice ( - ( COMMENT _DEPTH _LIMIT + 1 ) ) [ 0 ]
2023-07-21 22:33:11 +00:00
return {
href : {
pathname : '/items/[id]' ,
query : { id : rootId , commentId : n . item . id }
} ,
as : ` /items/ ${ rootId } `
}
2023-06-01 00:51:30 +00:00
} else {
2023-07-21 22:33:11 +00:00
return {
href : {
pathname : '/items/[id]' ,
query : { id : n . item . root . id , commentId : n . item . id }
} ,
as : ` /items/ ${ n . item . root . id } `
}
2023-06-01 00:51:30 +00:00
}
} else {
2023-07-21 22:33:11 +00:00
return {
href : {
pathname : '/items/[id]' ,
query : { id : n . item . id }
} ,
as : ` /items/ ${ n . item . id } `
}
2023-06-01 00:51:30 +00:00
}
}
2023-02-01 14:44:35 +00:00
function Streak ( { n } ) {
function blurb ( n ) {
const index = Number ( n . id ) % 6
const FOUND _BLURBS = [
'The harsh frontier is no place for the unprepared. This hat will protect you from the sun, dust, and other elements Mother Nature throws your way.' ,
'A cowboy is nothing without a cowboy hat. Take good care of it, and it will protect you from the sun, dust, and other elements on your journey.' ,
"This is not just a hat, it's a matter of survival. Take care of this essential tool, and it will shield you from the scorching sun and the elements." ,
"A cowboy hat isn't just a fashion statement. It's your last defense against the unforgiving elements of the Wild West. Hang onto it tight." ,
"A good cowboy hat is worth its weight in gold, shielding you from the sun, wind, and dust of the western frontier. Don't lose it." ,
'Your cowboy hat is the key to your survival in the wild west. Treat it with respect and it will protect you from the elements.'
]
const LOST _BLURBS = [
'your cowboy hat was taken by the wind storm that blew in from the west. No worries, a true cowboy always finds another hat.' ,
"you left your trusty cowboy hat in the saloon before leaving town. You'll need a replacement for the long journey west." ,
'you lost your cowboy hat in a wild shoot-out on the outskirts of town. Tough luck, tIme to start searching for another one.' ,
'you ran out of food and had to trade your hat for supplies. Better start looking for another hat.' ,
"your hat was stolen by a mischievous prairie dog. You won't catch the dog, but you can always find another hat." ,
'you lost your hat while crossing the river on your journey west. Maybe you can find a replacement hat in the next town.'
]
if ( n . days ) {
return ` After ${ n . days } days, ` + LOST _BLURBS [ index ]
}
return FOUND _BLURBS [ index ]
}
return (
< div className = 'd-flex font-weight-bold ml-2 py-1' >
< div style = { { fontSize : '2rem' } } > { n . days ? < BaldIcon className = 'fill-grey' height = { 40 } width = { 40 } / > : < CowboyHatIcon className = 'fill-grey' height = { 40 } width = { 40 } / > } < / d i v >
< div className = 'ml-1 p-1' >
you { n . days ? 'lost your' : 'found a' } cowboy hat
< div > < small style = { { lineHeight : '140%' , display : 'inline-block' } } > { blurb ( n ) } < / s m a l l > < / d i v >
< / d i v >
2021-08-20 00:13:32 +00:00
< / d i v >
)
}
2023-06-01 18:22:39 +00:00
function EarnNotification ( { n } ) {
2023-06-01 00:51:30 +00:00
return (
2023-07-21 22:33:11 +00:00
< div className = 'd-flex ml-2 py-1' >
< HandCoin className = 'align-self-center fill-boost mx-1' width = { 24 } height = { 24 } style = { { flex : '0 0 24px' , transform : 'rotateY(180deg)' } } / >
< div className = 'ml-2' >
< div className = 'font-weight-bold text-boost' >
you stacked { n . earnedSats } sats in rewards < small className = 'text-muted ml-1' > { timeSince ( new Date ( n . sortTime ) ) } < / s m a l l >
< / d i v >
{ n . sources &&
< div style = { { fontSize : '80%' , color : 'var(--theme-grey)' } } >
{ n . sources . posts > 0 && < span > { n . sources . posts } sats for top posts < / s p a n > }
{ n . sources . comments > 0 && < span > { n . sources . posts > 0 && ' \\ ' } { n . sources . comments } sats for top comments < / s p a n > }
{ n . sources . tipPosts > 0 && < span > { ( n . sources . comments > 0 || n . sources . posts > 0 ) && ' \\ ' } { n . sources . tipPosts } sats for zapping top posts early < / s p a n > }
{ n . sources . tipComments > 0 && < span > { ( n . sources . comments > 0 || n . sources . posts > 0 || n . sources . tipPosts > 0 ) && ' \\ ' } { n . sources . tipComments } sats for zapping top comments early < / s p a n > }
< / d i v > }
< div className = 'pb-1' style = { { lineHeight : '140%' } } >
SN distributes the sats it earns back to its best stackers daily . These sats come from < Link href = '/~jobs' > jobs < / L i n k > , b o o s t s , p o s t i n g f e e s , a n d d o n a t i o n s . Y o u c a n s e e t h e d a i l y r e w a r d s p o o l a n d m a k e a d o n a t i o n < L i n k h r e f = ' / r e w a r d s ' > h e r e < / L i n k > .
2023-06-01 00:51:30 +00:00
< / d i v >
< / d i v >
2023-07-21 22:33:11 +00:00
< / d i v >
2023-06-01 18:22:39 +00:00
)
2023-06-01 00:51:30 +00:00
}
2023-06-01 18:22:39 +00:00
function Invitification ( { n } ) {
2023-06-01 00:51:30 +00:00
return (
2023-07-21 22:33:11 +00:00
< NotificationLayout href = '/invites' >
2023-06-01 00:51:30 +00:00
< small className = 'font-weight-bold text-secondary ml-2' >
2023-07-09 17:37:12 +00:00
your invite has been redeemed by { n . invite . invitees . length } stackers
2023-06-01 00:51:30 +00:00
< / s m a l l >
< div className = 'ml-4 mr-2 mt-1' >
< Invite
invite = { n . invite } active = {
! n . invite . revoked &&
! ( n . invite . limit && n . invite . invitees . length >= n . invite . limit )
}
/ >
< / d i v >
< / N o t i f i c a t i o n L a y o u t >
)
}
2023-06-01 18:22:39 +00:00
function InvoicePaid ( { n } ) {
2023-06-01 00:51:30 +00:00
return (
2023-07-21 22:33:11 +00:00
< NotificationLayout href = { ` /invoices/ ${ n . invoice . id } ` } >
2023-06-01 00:51:30 +00:00
< div className = 'font-weight-bold text-info ml-2 py-1' >
< Check className = 'fill-info mr-1' / > { n . earnedSats } sats were deposited in your account
< small className = 'text-muted ml-1' > { timeSince ( new Date ( n . sortTime ) ) } < / s m a l l >
< / d i v >
< / N o t i f i c a t i o n L a y o u t >
)
}
2023-06-01 18:22:39 +00:00
function Referral ( { n } ) {
2023-06-01 00:51:30 +00:00
return (
< NotificationLayout >
< small className = 'font-weight-bold text-secondary ml-2' >
2023-07-21 22:33:11 +00:00
someone joined via one of your < Link href = '/referrals/month' className = 'text-reset' > referral links < / L i n k >
2023-06-01 00:51:30 +00:00
< small className = 'text-muted ml-1' > { timeSince ( new Date ( n . sortTime ) ) } < / s m a l l >
< / s m a l l >
< / N o t i f i c a t i o n L a y o u t >
)
}
2023-06-01 18:22:39 +00:00
function Votification ( { n } ) {
2023-06-01 00:51:30 +00:00
return (
2023-07-21 22:33:11 +00:00
< NotificationLayout { ... defaultOnClick ( n ) } >
2023-06-01 00:51:30 +00:00
< small className = 'font-weight-bold text-success ml-2' >
your { n . item . title ? 'post' : 'reply' } { n . item . fwdUser ? 'forwarded' : 'stacked' } { n . earnedSats } sats { n . item . fwdUser && ` to @ ${ n . item . fwdUser . name } ` }
< / s m a l l >
< div >
{ n . item . title
? < Item item = { n . item } / >
: (
< div className = 'pb-2' >
< RootProvider root = { n . item . root } >
< Comment item = { n . item } noReply includeParent clickToContext / >
< / R o o t P r o v i d e r >
< / d i v >
2023-06-01 18:22:39 +00:00
) }
2023-06-01 00:51:30 +00:00
< / d i v >
< / N o t i f i c a t i o n L a y o u t >
)
}
2023-06-01 18:22:39 +00:00
function Mention ( { n } ) {
2023-06-01 00:51:30 +00:00
return (
2023-07-21 22:33:11 +00:00
< NotificationLayout { ... defaultOnClick ( n ) } >
2023-06-01 00:51:30 +00:00
< small className = 'font-weight-bold text-info ml-2' >
you were mentioned in
< / s m a l l >
< div >
2023-06-01 18:22:39 +00:00
{ n . item . title
? < Item item = { n . item } / >
: (
< div className = 'pb-2' >
< RootProvider root = { n . item . root } >
< Comment item = { n . item } noReply includeParent rootText = { n . _ _typename === 'Reply' ? 'replying on:' : undefined } clickToContext / >
< / R o o t P r o v i d e r >
< / d i v > ) }
2023-06-01 00:51:30 +00:00
< / d i v >
< / N o t i f i c a t i o n L a y o u t >
)
}
2023-06-01 18:22:39 +00:00
function JobChanged ( { n } ) {
2023-06-01 00:51:30 +00:00
return (
2023-07-21 22:33:11 +00:00
< NotificationLayout { ... defaultOnClick ( n ) } >
2023-06-01 00:51:30 +00:00
< small className = { ` font-weight-bold text- ${ n . item . status === 'ACTIVE' ? 'success' : 'boost' } ml-1 ` } >
{ n . item . status === 'ACTIVE'
? 'your job is active again'
: ( n . item . status === 'NOSATS'
? 'your job promotion ran out of sats'
: 'your job has been stopped' ) }
< / s m a l l >
< ItemJob item = { n . item } / >
< / N o t i f i c a t i o n L a y o u t >
)
}
2023-06-01 18:22:39 +00:00
function Reply ( { n } ) {
2023-06-01 00:51:30 +00:00
return (
2023-07-21 22:33:11 +00:00
< NotificationLayout { ... defaultOnClick ( n ) } rootText = 'replying on:' >
2023-06-01 18:22:39 +00:00
< div className = 'py-2' >
2023-06-01 00:51:30 +00:00
{ n . item . title
? < Item item = { n . item } / >
: (
< div className = 'pb-2' >
< RootProvider root = { n . item . root } >
< Comment item = { n . item } noReply includeParent clickToContext rootText = 'replying on:' / >
< / R o o t P r o v i d e r >
< / d i v >
2023-06-01 18:22:39 +00:00
) }
2023-06-01 00:51:30 +00:00
< / d i v >
< / N o t i f i c a t i o n L a y o u t >
)
}
2023-06-12 18:03:44 +00:00
function NotificationAlert ( ) {
const [ showAlert , setShowAlert ] = useState ( false )
Service worker rework, Web Target Share API & Web Push API (#324)
* npm uninstall next-pwa
next-pwa was last updated in August 2022.
There is also an issue which mentions that next-pwa is abandoned (?): https://github.com/shadowwalker/next-pwa/issues/482
But the main reason for me uninstalling it is that it adds a lot of preconfigured stuff which is not necessary for us.
It even lead to a bug since pages were cached without our knowledge.
So I will go with a different PWA approach. This different approach should do the following:
- make it more transparent what the service worker is doing
- gives us more control to configure the service worker and thus making it easier
* Use workbox-webpack-plugin
Every other plugin (`next-offline`, `next-workbox-webpack-plugin`, `next-with-workbox`, ...) added unnecessary configuration which felt contrary to how PWAs should be built.
(PWAs should progressivly enhance the website in small steps, see https://web.dev/learn/pwa/getting-started/#focus-on-a-feature)
These default configurations even lead to worse UX since they made invalid assumptions about stacker.news:
We _do not_ want to cache our start url and we _do not_ want to cache anything unless explicitly told to.
Almost every page on SN should be fresh for the best UX.
To achieve this, by default, the service worker falls back to the network (as if the service worker wasn't there).
Therefore, this should be the simplest configuration with a valid precache and cache busting support.
In the future, we can try to use prefetching to improve performance of navigation requests.
* Add support for Web Share Target API
See https://developer.chrome.com/articles/web-share-target/
* Use Web Push API for push notifications
I followed this (very good!) guide: https://web.dev/notifications/
* Refactor code related to Web Push
* Send push notification to users on events
* Merge notifications
* Send notification to author of every parent recursively
* Remove unused userId param in savePushSubscription
As it should be, the user id is retrieved from the authenticated user in the backend.
* Resubscribe user if push subscription changed
* Update old subscription if oldEndpoint was given
* Allow users to unsubscribe
* Use LTREE operator instead of recursive query
* Always show checkbox for push notifications
* Justify checkbox to end
* Update title of first push notification
* Fix warning from uncontrolled to controlled
* Add comment about Notification.requestPermission
* Fix timestamp
* Catch error on push subscription toggle
* Wrap function bodies in try/catch
* Use Promise.allSettled
* Filter subscriptions by user notification settings
* Fix user notification filter
* Use skipWaiting
---------
Co-authored-by: ekzyis <ek@stacker.news>
2023-07-04 19:36:07 +00:00
const [ hasSubscription , setHasSubscription ] = useState ( false )
const [ error , setError ] = useState ( null )
2023-07-04 23:44:03 +00:00
const [ supported , setSupported ] = useState ( false )
Service worker rework, Web Target Share API & Web Push API (#324)
* npm uninstall next-pwa
next-pwa was last updated in August 2022.
There is also an issue which mentions that next-pwa is abandoned (?): https://github.com/shadowwalker/next-pwa/issues/482
But the main reason for me uninstalling it is that it adds a lot of preconfigured stuff which is not necessary for us.
It even lead to a bug since pages were cached without our knowledge.
So I will go with a different PWA approach. This different approach should do the following:
- make it more transparent what the service worker is doing
- gives us more control to configure the service worker and thus making it easier
* Use workbox-webpack-plugin
Every other plugin (`next-offline`, `next-workbox-webpack-plugin`, `next-with-workbox`, ...) added unnecessary configuration which felt contrary to how PWAs should be built.
(PWAs should progressivly enhance the website in small steps, see https://web.dev/learn/pwa/getting-started/#focus-on-a-feature)
These default configurations even lead to worse UX since they made invalid assumptions about stacker.news:
We _do not_ want to cache our start url and we _do not_ want to cache anything unless explicitly told to.
Almost every page on SN should be fresh for the best UX.
To achieve this, by default, the service worker falls back to the network (as if the service worker wasn't there).
Therefore, this should be the simplest configuration with a valid precache and cache busting support.
In the future, we can try to use prefetching to improve performance of navigation requests.
* Add support for Web Share Target API
See https://developer.chrome.com/articles/web-share-target/
* Use Web Push API for push notifications
I followed this (very good!) guide: https://web.dev/notifications/
* Refactor code related to Web Push
* Send push notification to users on events
* Merge notifications
* Send notification to author of every parent recursively
* Remove unused userId param in savePushSubscription
As it should be, the user id is retrieved from the authenticated user in the backend.
* Resubscribe user if push subscription changed
* Update old subscription if oldEndpoint was given
* Allow users to unsubscribe
* Use LTREE operator instead of recursive query
* Always show checkbox for push notifications
* Justify checkbox to end
* Update title of first push notification
* Fix warning from uncontrolled to controlled
* Add comment about Notification.requestPermission
* Fix timestamp
* Catch error on push subscription toggle
* Wrap function bodies in try/catch
* Use Promise.allSettled
* Filter subscriptions by user notification settings
* Fix user notification filter
* Use skipWaiting
---------
Co-authored-by: ekzyis <ek@stacker.news>
2023-07-04 19:36:07 +00:00
const sw = useServiceWorker ( )
2023-06-12 18:03:44 +00:00
useEffect ( ( ) => {
Service worker rework, Web Target Share API & Web Push API (#324)
* npm uninstall next-pwa
next-pwa was last updated in August 2022.
There is also an issue which mentions that next-pwa is abandoned (?): https://github.com/shadowwalker/next-pwa/issues/482
But the main reason for me uninstalling it is that it adds a lot of preconfigured stuff which is not necessary for us.
It even lead to a bug since pages were cached without our knowledge.
So I will go with a different PWA approach. This different approach should do the following:
- make it more transparent what the service worker is doing
- gives us more control to configure the service worker and thus making it easier
* Use workbox-webpack-plugin
Every other plugin (`next-offline`, `next-workbox-webpack-plugin`, `next-with-workbox`, ...) added unnecessary configuration which felt contrary to how PWAs should be built.
(PWAs should progressivly enhance the website in small steps, see https://web.dev/learn/pwa/getting-started/#focus-on-a-feature)
These default configurations even lead to worse UX since they made invalid assumptions about stacker.news:
We _do not_ want to cache our start url and we _do not_ want to cache anything unless explicitly told to.
Almost every page on SN should be fresh for the best UX.
To achieve this, by default, the service worker falls back to the network (as if the service worker wasn't there).
Therefore, this should be the simplest configuration with a valid precache and cache busting support.
In the future, we can try to use prefetching to improve performance of navigation requests.
* Add support for Web Share Target API
See https://developer.chrome.com/articles/web-share-target/
* Use Web Push API for push notifications
I followed this (very good!) guide: https://web.dev/notifications/
* Refactor code related to Web Push
* Send push notification to users on events
* Merge notifications
* Send notification to author of every parent recursively
* Remove unused userId param in savePushSubscription
As it should be, the user id is retrieved from the authenticated user in the backend.
* Resubscribe user if push subscription changed
* Update old subscription if oldEndpoint was given
* Allow users to unsubscribe
* Use LTREE operator instead of recursive query
* Always show checkbox for push notifications
* Justify checkbox to end
* Update title of first push notification
* Fix warning from uncontrolled to controlled
* Add comment about Notification.requestPermission
* Fix timestamp
* Catch error on push subscription toggle
* Wrap function bodies in try/catch
* Use Promise.allSettled
* Filter subscriptions by user notification settings
* Fix user notification filter
* Use skipWaiting
---------
Co-authored-by: ekzyis <ek@stacker.news>
2023-07-04 19:36:07 +00:00
const isSupported = sw . support . serviceWorker && sw . support . pushManager && sw . support . notification
2023-07-04 23:44:03 +00:00
if ( isSupported ) {
const isDefaultPermission = sw . permission . notification === 'default'
setShowAlert ( isDefaultPermission && ! localStorage . getItem ( 'hideNotifyPrompt' ) )
sw . registration ? . pushManager . getSubscription ( ) . then ( subscription => setHasSubscription ( ! ! subscription ) )
setSupported ( true )
}
Service worker rework, Web Target Share API & Web Push API (#324)
* npm uninstall next-pwa
next-pwa was last updated in August 2022.
There is also an issue which mentions that next-pwa is abandoned (?): https://github.com/shadowwalker/next-pwa/issues/482
But the main reason for me uninstalling it is that it adds a lot of preconfigured stuff which is not necessary for us.
It even lead to a bug since pages were cached without our knowledge.
So I will go with a different PWA approach. This different approach should do the following:
- make it more transparent what the service worker is doing
- gives us more control to configure the service worker and thus making it easier
* Use workbox-webpack-plugin
Every other plugin (`next-offline`, `next-workbox-webpack-plugin`, `next-with-workbox`, ...) added unnecessary configuration which felt contrary to how PWAs should be built.
(PWAs should progressivly enhance the website in small steps, see https://web.dev/learn/pwa/getting-started/#focus-on-a-feature)
These default configurations even lead to worse UX since they made invalid assumptions about stacker.news:
We _do not_ want to cache our start url and we _do not_ want to cache anything unless explicitly told to.
Almost every page on SN should be fresh for the best UX.
To achieve this, by default, the service worker falls back to the network (as if the service worker wasn't there).
Therefore, this should be the simplest configuration with a valid precache and cache busting support.
In the future, we can try to use prefetching to improve performance of navigation requests.
* Add support for Web Share Target API
See https://developer.chrome.com/articles/web-share-target/
* Use Web Push API for push notifications
I followed this (very good!) guide: https://web.dev/notifications/
* Refactor code related to Web Push
* Send push notification to users on events
* Merge notifications
* Send notification to author of every parent recursively
* Remove unused userId param in savePushSubscription
As it should be, the user id is retrieved from the authenticated user in the backend.
* Resubscribe user if push subscription changed
* Update old subscription if oldEndpoint was given
* Allow users to unsubscribe
* Use LTREE operator instead of recursive query
* Always show checkbox for push notifications
* Justify checkbox to end
* Update title of first push notification
* Fix warning from uncontrolled to controlled
* Add comment about Notification.requestPermission
* Fix timestamp
* Catch error on push subscription toggle
* Wrap function bodies in try/catch
* Use Promise.allSettled
* Filter subscriptions by user notification settings
* Fix user notification filter
* Use skipWaiting
---------
Co-authored-by: ekzyis <ek@stacker.news>
2023-07-04 19:36:07 +00:00
} , [ sw ] )
2023-06-12 18:03:44 +00:00
const close = ( ) => {
localStorage . setItem ( 'hideNotifyPrompt' , 'yep' )
setShowAlert ( false )
}
return (
Service worker rework, Web Target Share API & Web Push API (#324)
* npm uninstall next-pwa
next-pwa was last updated in August 2022.
There is also an issue which mentions that next-pwa is abandoned (?): https://github.com/shadowwalker/next-pwa/issues/482
But the main reason for me uninstalling it is that it adds a lot of preconfigured stuff which is not necessary for us.
It even lead to a bug since pages were cached without our knowledge.
So I will go with a different PWA approach. This different approach should do the following:
- make it more transparent what the service worker is doing
- gives us more control to configure the service worker and thus making it easier
* Use workbox-webpack-plugin
Every other plugin (`next-offline`, `next-workbox-webpack-plugin`, `next-with-workbox`, ...) added unnecessary configuration which felt contrary to how PWAs should be built.
(PWAs should progressivly enhance the website in small steps, see https://web.dev/learn/pwa/getting-started/#focus-on-a-feature)
These default configurations even lead to worse UX since they made invalid assumptions about stacker.news:
We _do not_ want to cache our start url and we _do not_ want to cache anything unless explicitly told to.
Almost every page on SN should be fresh for the best UX.
To achieve this, by default, the service worker falls back to the network (as if the service worker wasn't there).
Therefore, this should be the simplest configuration with a valid precache and cache busting support.
In the future, we can try to use prefetching to improve performance of navigation requests.
* Add support for Web Share Target API
See https://developer.chrome.com/articles/web-share-target/
* Use Web Push API for push notifications
I followed this (very good!) guide: https://web.dev/notifications/
* Refactor code related to Web Push
* Send push notification to users on events
* Merge notifications
* Send notification to author of every parent recursively
* Remove unused userId param in savePushSubscription
As it should be, the user id is retrieved from the authenticated user in the backend.
* Resubscribe user if push subscription changed
* Update old subscription if oldEndpoint was given
* Allow users to unsubscribe
* Use LTREE operator instead of recursive query
* Always show checkbox for push notifications
* Justify checkbox to end
* Update title of first push notification
* Fix warning from uncontrolled to controlled
* Add comment about Notification.requestPermission
* Fix timestamp
* Catch error on push subscription toggle
* Wrap function bodies in try/catch
* Use Promise.allSettled
* Filter subscriptions by user notification settings
* Fix user notification filter
* Use skipWaiting
---------
Co-authored-by: ekzyis <ek@stacker.news>
2023-07-04 19:36:07 +00:00
error
2023-06-12 18:03:44 +00:00
? (
Service worker rework, Web Target Share API & Web Push API (#324)
* npm uninstall next-pwa
next-pwa was last updated in August 2022.
There is also an issue which mentions that next-pwa is abandoned (?): https://github.com/shadowwalker/next-pwa/issues/482
But the main reason for me uninstalling it is that it adds a lot of preconfigured stuff which is not necessary for us.
It even lead to a bug since pages were cached without our knowledge.
So I will go with a different PWA approach. This different approach should do the following:
- make it more transparent what the service worker is doing
- gives us more control to configure the service worker and thus making it easier
* Use workbox-webpack-plugin
Every other plugin (`next-offline`, `next-workbox-webpack-plugin`, `next-with-workbox`, ...) added unnecessary configuration which felt contrary to how PWAs should be built.
(PWAs should progressivly enhance the website in small steps, see https://web.dev/learn/pwa/getting-started/#focus-on-a-feature)
These default configurations even lead to worse UX since they made invalid assumptions about stacker.news:
We _do not_ want to cache our start url and we _do not_ want to cache anything unless explicitly told to.
Almost every page on SN should be fresh for the best UX.
To achieve this, by default, the service worker falls back to the network (as if the service worker wasn't there).
Therefore, this should be the simplest configuration with a valid precache and cache busting support.
In the future, we can try to use prefetching to improve performance of navigation requests.
* Add support for Web Share Target API
See https://developer.chrome.com/articles/web-share-target/
* Use Web Push API for push notifications
I followed this (very good!) guide: https://web.dev/notifications/
* Refactor code related to Web Push
* Send push notification to users on events
* Merge notifications
* Send notification to author of every parent recursively
* Remove unused userId param in savePushSubscription
As it should be, the user id is retrieved from the authenticated user in the backend.
* Resubscribe user if push subscription changed
* Update old subscription if oldEndpoint was given
* Allow users to unsubscribe
* Use LTREE operator instead of recursive query
* Always show checkbox for push notifications
* Justify checkbox to end
* Update title of first push notification
* Fix warning from uncontrolled to controlled
* Add comment about Notification.requestPermission
* Fix timestamp
* Catch error on push subscription toggle
* Wrap function bodies in try/catch
* Use Promise.allSettled
* Filter subscriptions by user notification settings
* Fix user notification filter
* Use skipWaiting
---------
Co-authored-by: ekzyis <ek@stacker.news>
2023-07-04 19:36:07 +00:00
< Alert variant = 'danger' dismissible onClose = { ( ) => setError ( null ) } >
< span > { error . toString ( ) } < / s p a n >
2023-06-12 18:03:44 +00:00
< / A l e r t >
)
Service worker rework, Web Target Share API & Web Push API (#324)
* npm uninstall next-pwa
next-pwa was last updated in August 2022.
There is also an issue which mentions that next-pwa is abandoned (?): https://github.com/shadowwalker/next-pwa/issues/482
But the main reason for me uninstalling it is that it adds a lot of preconfigured stuff which is not necessary for us.
It even lead to a bug since pages were cached without our knowledge.
So I will go with a different PWA approach. This different approach should do the following:
- make it more transparent what the service worker is doing
- gives us more control to configure the service worker and thus making it easier
* Use workbox-webpack-plugin
Every other plugin (`next-offline`, `next-workbox-webpack-plugin`, `next-with-workbox`, ...) added unnecessary configuration which felt contrary to how PWAs should be built.
(PWAs should progressivly enhance the website in small steps, see https://web.dev/learn/pwa/getting-started/#focus-on-a-feature)
These default configurations even lead to worse UX since they made invalid assumptions about stacker.news:
We _do not_ want to cache our start url and we _do not_ want to cache anything unless explicitly told to.
Almost every page on SN should be fresh for the best UX.
To achieve this, by default, the service worker falls back to the network (as if the service worker wasn't there).
Therefore, this should be the simplest configuration with a valid precache and cache busting support.
In the future, we can try to use prefetching to improve performance of navigation requests.
* Add support for Web Share Target API
See https://developer.chrome.com/articles/web-share-target/
* Use Web Push API for push notifications
I followed this (very good!) guide: https://web.dev/notifications/
* Refactor code related to Web Push
* Send push notification to users on events
* Merge notifications
* Send notification to author of every parent recursively
* Remove unused userId param in savePushSubscription
As it should be, the user id is retrieved from the authenticated user in the backend.
* Resubscribe user if push subscription changed
* Update old subscription if oldEndpoint was given
* Allow users to unsubscribe
* Use LTREE operator instead of recursive query
* Always show checkbox for push notifications
* Justify checkbox to end
* Update title of first push notification
* Fix warning from uncontrolled to controlled
* Add comment about Notification.requestPermission
* Fix timestamp
* Catch error on push subscription toggle
* Wrap function bodies in try/catch
* Use Promise.allSettled
* Filter subscriptions by user notification settings
* Fix user notification filter
* Use skipWaiting
---------
Co-authored-by: ekzyis <ek@stacker.news>
2023-07-04 19:36:07 +00:00
: showAlert
? (
< Alert variant = 'info' dismissible onClose = { close } >
< span className = 'align-middle' > Enable push notifications ? < / s p a n >
< button
className = { ` ${ styles . alertBtn } mx-1 ` }
onClick = { async ( ) => {
await sw . requestNotificationPermission ( )
. then ( close )
. catch ( setError )
} }
> Yes
< / b u t t o n >
< button className = { ` ${ styles . alertBtn } ` } onClick = { close } > No < / b u t t o n >
< / A l e r t >
)
: (
2023-07-21 22:33:11 +00:00
< Form className = { ` d-flex justify-content-end ${ supported ? 'visible' : 'invisible' } ` } initial = { { pushNotify : hasSubscription } } >
Service worker rework, Web Target Share API & Web Push API (#324)
* npm uninstall next-pwa
next-pwa was last updated in August 2022.
There is also an issue which mentions that next-pwa is abandoned (?): https://github.com/shadowwalker/next-pwa/issues/482
But the main reason for me uninstalling it is that it adds a lot of preconfigured stuff which is not necessary for us.
It even lead to a bug since pages were cached without our knowledge.
So I will go with a different PWA approach. This different approach should do the following:
- make it more transparent what the service worker is doing
- gives us more control to configure the service worker and thus making it easier
* Use workbox-webpack-plugin
Every other plugin (`next-offline`, `next-workbox-webpack-plugin`, `next-with-workbox`, ...) added unnecessary configuration which felt contrary to how PWAs should be built.
(PWAs should progressivly enhance the website in small steps, see https://web.dev/learn/pwa/getting-started/#focus-on-a-feature)
These default configurations even lead to worse UX since they made invalid assumptions about stacker.news:
We _do not_ want to cache our start url and we _do not_ want to cache anything unless explicitly told to.
Almost every page on SN should be fresh for the best UX.
To achieve this, by default, the service worker falls back to the network (as if the service worker wasn't there).
Therefore, this should be the simplest configuration with a valid precache and cache busting support.
In the future, we can try to use prefetching to improve performance of navigation requests.
* Add support for Web Share Target API
See https://developer.chrome.com/articles/web-share-target/
* Use Web Push API for push notifications
I followed this (very good!) guide: https://web.dev/notifications/
* Refactor code related to Web Push
* Send push notification to users on events
* Merge notifications
* Send notification to author of every parent recursively
* Remove unused userId param in savePushSubscription
As it should be, the user id is retrieved from the authenticated user in the backend.
* Resubscribe user if push subscription changed
* Update old subscription if oldEndpoint was given
* Allow users to unsubscribe
* Use LTREE operator instead of recursive query
* Always show checkbox for push notifications
* Justify checkbox to end
* Update title of first push notification
* Fix warning from uncontrolled to controlled
* Add comment about Notification.requestPermission
* Fix timestamp
* Catch error on push subscription toggle
* Wrap function bodies in try/catch
* Use Promise.allSettled
* Filter subscriptions by user notification settings
* Fix user notification filter
* Use skipWaiting
---------
Co-authored-by: ekzyis <ek@stacker.news>
2023-07-04 19:36:07 +00:00
< Checkbox
2023-07-04 22:19:59 +00:00
name = 'pushNotify' label = { < span className = 'text-muted' > push notifications < / s p a n > }
groupClassName = { ` ${ styles . subFormGroup } mb-1 mr-sm-3 mr-0 ` }
inline checked = { hasSubscription } handleChange = { async ( ) => {
Service worker rework, Web Target Share API & Web Push API (#324)
* npm uninstall next-pwa
next-pwa was last updated in August 2022.
There is also an issue which mentions that next-pwa is abandoned (?): https://github.com/shadowwalker/next-pwa/issues/482
But the main reason for me uninstalling it is that it adds a lot of preconfigured stuff which is not necessary for us.
It even lead to a bug since pages were cached without our knowledge.
So I will go with a different PWA approach. This different approach should do the following:
- make it more transparent what the service worker is doing
- gives us more control to configure the service worker and thus making it easier
* Use workbox-webpack-plugin
Every other plugin (`next-offline`, `next-workbox-webpack-plugin`, `next-with-workbox`, ...) added unnecessary configuration which felt contrary to how PWAs should be built.
(PWAs should progressivly enhance the website in small steps, see https://web.dev/learn/pwa/getting-started/#focus-on-a-feature)
These default configurations even lead to worse UX since they made invalid assumptions about stacker.news:
We _do not_ want to cache our start url and we _do not_ want to cache anything unless explicitly told to.
Almost every page on SN should be fresh for the best UX.
To achieve this, by default, the service worker falls back to the network (as if the service worker wasn't there).
Therefore, this should be the simplest configuration with a valid precache and cache busting support.
In the future, we can try to use prefetching to improve performance of navigation requests.
* Add support for Web Share Target API
See https://developer.chrome.com/articles/web-share-target/
* Use Web Push API for push notifications
I followed this (very good!) guide: https://web.dev/notifications/
* Refactor code related to Web Push
* Send push notification to users on events
* Merge notifications
* Send notification to author of every parent recursively
* Remove unused userId param in savePushSubscription
As it should be, the user id is retrieved from the authenticated user in the backend.
* Resubscribe user if push subscription changed
* Update old subscription if oldEndpoint was given
* Allow users to unsubscribe
* Use LTREE operator instead of recursive query
* Always show checkbox for push notifications
* Justify checkbox to end
* Update title of first push notification
* Fix warning from uncontrolled to controlled
* Add comment about Notification.requestPermission
* Fix timestamp
* Catch error on push subscription toggle
* Wrap function bodies in try/catch
* Use Promise.allSettled
* Filter subscriptions by user notification settings
* Fix user notification filter
* Use skipWaiting
---------
Co-authored-by: ekzyis <ek@stacker.news>
2023-07-04 19:36:07 +00:00
await sw . togglePushSubscription ( ) . catch ( setError )
} }
/ >
< / F o r m >
)
2023-06-12 18:03:44 +00:00
)
}
2023-07-21 22:33:11 +00:00
export default function Notifications ( { ssrData } ) {
const { data , fetchMore } = useQuery ( NOTIFICATIONS )
const client = useApolloClient ( )
2021-08-20 00:13:32 +00:00
2023-07-21 22:33:11 +00:00
useEffect ( ( ) => {
client . writeQuery ( {
query : HAS _NOTIFICATIONS ,
data : {
hasNewNotes : false
}
} )
} , [ client ] )
const { notifications : { notifications , earn , lastChecked , cursor } } = useMemo ( ( ) => {
if ( ! data && ! ssrData ) return { notifications : { } }
return data || ssrData
} , [ data , ssrData ] )
2021-08-20 00:13:32 +00:00
2023-07-21 22:33:11 +00:00
const [ fresh , old ] = useMemo ( ( ) => {
if ( ! notifications ) return [ [ ] , [ ] ]
return notifications . reduce ( ( result , n ) => {
2021-08-20 00:13:32 +00:00
result [ new Date ( n . sortTime ) . getTime ( ) > lastChecked ? 0 : 1 ] . push ( n )
return result
} ,
[ [ ] , [ ] ] )
2023-07-21 22:33:11 +00:00
} , [ notifications , lastChecked ] )
if ( ! data && ! ssrData ) return < CommentsFlatSkeleton / >
2021-08-17 23:59:22 +00:00
2021-08-17 18:15:24 +00:00
return (
< >
2023-06-12 18:03:44 +00:00
< NotificationAlert / >
2021-11-04 18:22:03 +00:00
< div className = 'fresh' >
2022-04-24 16:13:07 +00:00
{ earn && < Notification n = { earn } key = 'earn' / > }
2021-08-20 00:13:32 +00:00
{ fresh . map ( ( n , i ) => (
2023-07-21 22:33:11 +00:00
< Notification n = { n } key = { n . _ _typename + n . id + n . sortTime } / >
2021-08-20 00:13:32 +00:00
) ) }
< / d i v >
{ old . map ( ( n , i ) => (
2023-07-21 22:33:11 +00:00
< Notification n = { n } key = { n . _ _typename + n . id + n . sortTime } / >
2021-08-17 18:15:24 +00:00
) ) }
2023-07-21 22:33:11 +00:00
< MoreFooter cursor = { cursor } count = { notifications ? . length } fetchMore = { fetchMore } Skeleton = { CommentsFlatSkeleton } noMoreText = 'NO MORE' / >
2021-08-17 18:15:24 +00:00
< / >
)
}
function CommentsFlatSkeleton ( ) {
const comments = new Array ( 21 ) . fill ( null )
return (
2023-07-21 22:33:11 +00:00
< div >
{ comments . map ( ( _ , i ) => (
< CommentSkeleton key = { i } skeletonChildren = { 0 } / >
) ) }
2021-08-17 18:15:24 +00:00
< / d i v >
)
}