151 lines
5.9 KiB
JavaScript
151 lines
5.9 KiB
JavaScript
import '@/styles/globals.scss'
|
|
import { ApolloProvider, gql } from '@apollo/client'
|
|
import { MeProvider } from '@/components/me'
|
|
import PlausibleProvider from 'next-plausible'
|
|
import getApolloClient from '@/lib/apollo.js'
|
|
import { PriceProvider } from '@/components/price'
|
|
import { BlockHeightProvider } from '@/components/block-height'
|
|
import Head from 'next/head'
|
|
import { useRouter } from 'next/dist/client/router'
|
|
import { useCallback, useEffect } from 'react'
|
|
import { ShowModalProvider } from '@/components/modal'
|
|
import ErrorBoundary from '@/components/error-boundary'
|
|
import { LightningProvider } from '@/components/lightning'
|
|
import { ToastProvider } from '@/components/toast'
|
|
import { ServiceWorkerProvider } from '@/components/serviceworker'
|
|
import { SSR } from '@/lib/constants'
|
|
import NProgress from 'nprogress'
|
|
import 'nprogress/nprogress.css'
|
|
import { LoggerProvider } from '@/components/logger'
|
|
import { ChainFeeProvider } from '@/components/chain-fee.js'
|
|
import dynamic from 'next/dynamic'
|
|
import { HasNewNotesProvider } from '@/components/use-has-new-notes'
|
|
import { WebLnProvider } from '@/wallets/webln/client'
|
|
import { AccountProvider } from '@/components/account'
|
|
import { WalletsProvider } from '@/wallets/index'
|
|
|
|
const PWAPrompt = dynamic(() => import('react-ios-pwa-prompt'), { ssr: false })
|
|
|
|
NProgress.configure({
|
|
showSpinner: false
|
|
})
|
|
|
|
function writeQuery (client, apollo, data) {
|
|
if (apollo && data) {
|
|
client.writeQuery({
|
|
query: gql`${apollo.query}`,
|
|
data,
|
|
variables: apollo.variables,
|
|
overwrite: SSR,
|
|
broadcast: false
|
|
})
|
|
}
|
|
}
|
|
|
|
export default function MyApp ({ Component, pageProps: { ...props } }) {
|
|
const client = getApolloClient()
|
|
const router = useRouter()
|
|
|
|
const shouldShowProgressBar = useCallback((newPathname, shallow) => {
|
|
return !shallow || newPathname !== router.pathname
|
|
}, [router.pathname])
|
|
|
|
useEffect(() => {
|
|
const nprogressStart = (newPathname, { shallow }) => {
|
|
shouldShowProgressBar(newPathname, shallow) && NProgress.start()
|
|
}
|
|
const nprogressDone = (newPathname, { shallow }) => {
|
|
NProgress.done()
|
|
}
|
|
|
|
router.events.on('routeChangeStart', nprogressStart)
|
|
router.events.on('routeChangeComplete', nprogressDone)
|
|
router.events.on('routeChangeError', nprogressDone)
|
|
|
|
if (!props?.apollo) return
|
|
// HACK: 'cause there's no way to tell Next to skip SSR
|
|
// So every page load, we modify the route in browser history
|
|
// to point to the same page but without SSR, ie ?nodata=true
|
|
// this nodata var will get passed to the server on back/foward and
|
|
// 1. prevent data from reloading and 2. perserve scroll
|
|
// (2) is not possible while intercepting nav with beforePopState
|
|
router.replace({
|
|
pathname: router.pathname,
|
|
query: { ...router.query, nodata: true }
|
|
}, router.asPath, { ...router.options, shallow: true }).catch((e) => {
|
|
// workaround for https://github.com/vercel/next.js/issues/37362
|
|
if (!e.cancelled) {
|
|
console.log(e)
|
|
throw e
|
|
}
|
|
})
|
|
|
|
return () => {
|
|
router.events.off('routeChangeStart', nprogressStart)
|
|
router.events.off('routeChangeComplete', nprogressDone)
|
|
router.events.off('routeChangeError', nprogressDone)
|
|
}
|
|
}, [router.asPath, props?.apollo, shouldShowProgressBar])
|
|
|
|
useEffect(() => {
|
|
// hack to disable ios pwa prompt for https://github.com/stackernews/stacker.news/issues/953
|
|
// see https://github.com/chrisdancee/react-ios-pwa-prompt/blob/66e91c4f033b740cff42c3220cf13ebdf39e3078/src/index.js#L30
|
|
if (router?.query?.disablePrompt) {
|
|
window.localStorage.setItem('iosPwaPrompt', JSON.stringify({ isiOS: false, visits: 0 }))
|
|
}
|
|
}, [router?.query?.disablePrompt])
|
|
|
|
/*
|
|
If we are on the client, we populate the apollo cache with the
|
|
ssr data
|
|
*/
|
|
const { apollo, ssrData, me, price, blockHeight, chainFee, ...otherProps } = props
|
|
useEffect(() => {
|
|
writeQuery(client, apollo, ssrData)
|
|
}, [client, apollo, ssrData])
|
|
|
|
return (
|
|
<>
|
|
<Head>
|
|
<meta name='viewport' content='initial-scale=1.0, width=device-width, viewport-fit=cover' />
|
|
</Head>
|
|
<ErrorBoundary>
|
|
<PlausibleProvider domain='stacker.news' trackOutboundLinks>
|
|
<ApolloProvider client={client}>
|
|
<MeProvider me={me}>
|
|
<WalletsProvider>
|
|
<HasNewNotesProvider>
|
|
<LoggerProvider>
|
|
<WebLnProvider>
|
|
<ServiceWorkerProvider>
|
|
<AccountProvider>
|
|
<PriceProvider price={price}>
|
|
<LightningProvider>
|
|
<ToastProvider>
|
|
<ShowModalProvider>
|
|
<BlockHeightProvider blockHeight={blockHeight}>
|
|
<ChainFeeProvider chainFee={chainFee}>
|
|
<ErrorBoundary>
|
|
<Component ssrData={ssrData} {...otherProps} />
|
|
{!router?.query?.disablePrompt && <PWAPrompt copyBody='This website has app functionality. Add it to your home screen to use it in fullscreen and receive notifications. In Safari:' promptOnVisit={2} />}
|
|
</ErrorBoundary>
|
|
</ChainFeeProvider>
|
|
</BlockHeightProvider>
|
|
</ShowModalProvider>
|
|
</ToastProvider>
|
|
</LightningProvider>
|
|
</PriceProvider>
|
|
</AccountProvider>
|
|
</ServiceWorkerProvider>
|
|
</WebLnProvider>
|
|
</LoggerProvider>
|
|
</HasNewNotesProvider>
|
|
</WalletsProvider>
|
|
</MeProvider>
|
|
</ApolloProvider>
|
|
</PlausibleProvider>
|
|
</ErrorBoundary>
|
|
</>
|
|
)
|
|
}
|