core/pages/_app.tsx

219 lines
6.7 KiB
TypeScript

import { DefaultSeo } from 'next-seo'
import App, { AppContext, AppProps } from 'next/app'
import dynamic from 'next/dynamic'
import Head from 'next/head'
import { Router, useRouter } from 'next/router'
import NProgress from 'nprogress'
import { useEffect, useState } from 'react'
import ReactGA from 'react-ga'
import { GlobalHotKeys } from 'react-hotkeys'
import Package from '../package.json'
import { Theme } from '@types'
import { DESCRIPTION, shortcutKeyMap, THEME_COLOR, TITLE } from '@utils/Constants'
import Logger from '@utils/Logger'
import { handlePWA, parseCookie, parseDockerhubTag, systemTheme } from '@utils/Tools'
const Footer = dynamic(() => import('@components/Footer'))
const Navbar = dynamic(() => import('@components/Navbar'))
const Modal = dynamic(() => import('@components/Modal'))
import PlatformDisplay from '@components/PlatformDisplay'
import '@fortawesome/fontawesome-free/css/all.css'
import 'rc-tooltip/assets/bootstrap_white.css'
import '../app.css'
import '../github-markdown.css'
// Progress Bar
NProgress.configure({ showSpinner: false })
Router.events.on('routeChangeStart', NProgress.start)
Router.events.on('routeChangeComplete', NProgress.done)
Router.events.on('routeChangeError', NProgress.done)
// Google Analytics
ReactGA.initialize('UA-165454387-1')
const KoreanbotsApp = ({ Component, pageProps, err, cookie }: KoreanbotsProps): JSX.Element => {
const [shortcutModal, setShortcutModal] = useState(false)
const [theme, setTheme] = useState<Theme>('system')
const [standalone, setStandalone] = useState(false)
const router = useRouter()
useEffect(() => {
console.log(
'%c' + 'KOREANBOTS',
'color: #3366FF; -webkit-text-stroke: 2px black; font-size: 72px; font-weight: bold;'
)
Logger.debug(
`[BUILD INFO] Tag: ${parseDockerhubTag(process.env.NEXT_PUBLIC_TAG)}, Version: v${
Package.version
}, Hash: ${process.env.NEXT_PUBLIC_SOURCE_COMMIT}`
)
console.log(
'%c' + '이곳에 코드를 붙여넣으면 공격자에게 엑세스 토큰을 넘겨줄 수 있습니다!!',
'color: #ff0000; font-size: 20px; font-weight: bold;'
)
if (!localStorage.theme) {
Logger.debug(`[THEME] ${systemTheme().toUpperCase()} THEME DETECTED`)
setTheme(systemTheme())
} else setTheme(localStorage.theme)
setStandalone(handlePWA())
const script = document.querySelector('script[src*=googlesyndication]')
if (script)
script.addEventListener('error', () => {
ReactGA.ga(
'send',
'event',
'adblock',
'adblock_' +
(navigator.userAgent.match(
/Android|BlackBerry|iPhone|iPad|iPod|Opera Mini|IEMobile|WPDesktop/i
)
? 'mobile'
: 'pc')
)
})
}, [])
return (
<div className={theme}>
<DefaultSeo
titleTemplate='%s - 한국 디스코드 리스트'
defaultTitle={TITLE}
description={DESCRIPTION}
openGraph={{
type: 'website',
title: TITLE,
url: 'https://koreanbots.dev',
site_name: TITLE,
description: DESCRIPTION,
images: [
{
url: '/logo.png',
width: 300,
height: 300,
alt: 'Logo',
},
],
}}
twitter={{
site: '@koreanbots',
handle: '@koreanbots',
cardType: 'summary',
}}
/>
<Head>
{/* META */}
<meta charSet='utf-8' />
<meta httpEquiv='X-UA-Compatible' content='IE=edge' />
<meta name='keywords' content='Korea, Korean, Discord, Bot, 디스코드봇, 한디리' />
<meta
name='viewport'
content='width=device-width,initial-scale=1,minimum-scale=1,maximum-scale=1,user-scalable=no'
/>
{/* Android */}
<meta name='theme-color' content={THEME_COLOR} />
<meta name='mobile-web-app-capable' content='yes' />
{/* iOS */}
<meta name='apple-mobile-web-app-title' content='Application Title' />
<meta name='apple-mobile-web-app-capable' content='yes' />
<meta name='apple-mobile-web-app-status-bar-style' content='default' />
{/* Windows */}
<meta name='msapplication-navbutton-color' content={THEME_COLOR} />
<meta name='msapplication-TileColor' content={THEME_COLOR} />
<meta name='msapplication-TileImage' content='/static/ms-icon-144x144.png' />
<meta name='msapplication-config' content='browserconfig.xml' />
{/* Pinned Sites */}
<meta name='application-name' content={TITLE} />
<meta name='msapplication-tooltip' content={DESCRIPTION} />
<meta name='msapplication-starturl' content='/' />
{/* Tap highlighting */}
<meta name='msapplication-tap-highlight' content='no' />
{/* UC Mobile Browser */}
<meta name='full-screen' content='yes' />
<meta name='browsermode' content='application' />
<meta name='nightmode' content='disable' />
<meta name='layoutmode' content='fitscreen' />
<meta name='imagemode' content='force' />
<meta name='screen-orientation' content='portrait' />
</Head>
<Navbar token={cookie.token} />
<div className='iu-is-the-best min-h-screen bg-white text-black dark:bg-discord-dark dark:text-gray-100'>
<Component {...pageProps} err={err} theme={theme} setTheme={setTheme} pwa={standalone} />
</div>
{!router.pathname.startsWith('/developers') && <Footer theme={theme} setTheme={setTheme} />}
<Modal
full
isOpen={shortcutModal}
onClose={() => setShortcutModal(false)}
dark={theme === 'dark'}
header='단축키 안내'
>
<div className='h-80 px-3'>
<h3 className='text-md font-semibold'></h3>
<ul>
<li className='pt-2'>
<h4 className='text-xs text-gray-500 dark:text-gray-400'> </h4>
<kbd>
<PlatformDisplay osx='CMD'>Ctrl</PlatformDisplay>
</kbd>{' '}
<kbd>/</kbd>
</li>
<li className='pt-2'>
<h4 className='text-xs text-gray-500 dark:text-gray-400'> </h4>
<kbd>
<PlatformDisplay osx='CMD'>Ctrl</PlatformDisplay>
</kbd>
<kbd>Shift</kbd> <kbd>D</kbd>
</li>
</ul>
</div>
</Modal>
<GlobalHotKeys
keyMap={shortcutKeyMap}
handlers={{
SHORTCUT_HELP: (event) => {
event.preventDefault()
setShortcutModal((value) => !value)
return
},
CHANGE_THEME: (event) => {
event.preventDefault()
const overwrite = (localStorage.theme || systemTheme()) === 'dark' ? 'light' : 'dark'
setTheme(overwrite)
localStorage.setItem('theme', overwrite)
return false
},
}}
/>
</div>
)
}
KoreanbotsApp.getInitialProps = async (appCtx: AppContext) => {
const appProps = await App.getInitialProps(appCtx)
const parsed = parseCookie(appCtx.ctx.req)
return {
...appProps,
cookie: parsed,
}
}
export default KoreanbotsApp
interface KoreanbotsProps extends AppProps {
err: unknown
cookie: {
token?: string
}
}