Du kannst nicht mehr als 25 Themen auswählen Themen müssen mit entweder einem Buchstaben oder einer Ziffer beginnen. Sie können Bindestriche („-“) enthalten und bis zu 35 Zeichen lang sein.

splash.tsx 3.2KB

1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980
  1. 'use client'
  2. import type { FC, PropsWithChildren } from 'react'
  3. import { useEffect } from 'react'
  4. import { useCallback } from 'react'
  5. import { useWebAppStore } from '@/context/web-app-context'
  6. import { useRouter, useSearchParams } from 'next/navigation'
  7. import AppUnavailable from '@/app/components/base/app-unavailable'
  8. import { checkOrSetAccessToken, removeAccessToken, setAccessToken } from '@/app/components/share/utils'
  9. import { useTranslation } from 'react-i18next'
  10. import { fetchAccessToken } from '@/service/share'
  11. import Loading from '@/app/components/base/loading'
  12. import { AccessMode } from '@/models/access-control'
  13. const Splash: FC<PropsWithChildren> = ({ children }) => {
  14. const { t } = useTranslation()
  15. const shareCode = useWebAppStore(s => s.shareCode)
  16. const webAppAccessMode = useWebAppStore(s => s.webAppAccessMode)
  17. const searchParams = useSearchParams()
  18. const router = useRouter()
  19. const redirectUrl = searchParams.get('redirect_url')
  20. const tokenFromUrl = searchParams.get('web_sso_token')
  21. const message = searchParams.get('message')
  22. const code = searchParams.get('code')
  23. const getSigninUrl = useCallback(() => {
  24. const params = new URLSearchParams(searchParams)
  25. params.delete('message')
  26. params.delete('code')
  27. return `/webapp-signin?${params.toString()}`
  28. }, [searchParams])
  29. const backToHome = useCallback(() => {
  30. removeAccessToken()
  31. const url = getSigninUrl()
  32. router.replace(url)
  33. }, [getSigninUrl, router])
  34. useEffect(() => {
  35. (async () => {
  36. if (message)
  37. return
  38. if (shareCode && tokenFromUrl && redirectUrl) {
  39. localStorage.setItem('webapp_access_token', tokenFromUrl)
  40. const tokenResp = await fetchAccessToken({ appCode: shareCode, webAppAccessToken: tokenFromUrl })
  41. await setAccessToken(shareCode, tokenResp.access_token)
  42. router.replace(decodeURIComponent(redirectUrl))
  43. return
  44. }
  45. if (shareCode && redirectUrl && localStorage.getItem('webapp_access_token')) {
  46. const tokenResp = await fetchAccessToken({ appCode: shareCode, webAppAccessToken: localStorage.getItem('webapp_access_token') })
  47. await setAccessToken(shareCode, tokenResp.access_token)
  48. router.replace(decodeURIComponent(redirectUrl))
  49. return
  50. }
  51. if (webAppAccessMode === AccessMode.PUBLIC && redirectUrl) {
  52. await checkOrSetAccessToken(shareCode)
  53. router.replace(decodeURIComponent(redirectUrl))
  54. }
  55. })()
  56. }, [shareCode, redirectUrl, router, tokenFromUrl, message, webAppAccessMode])
  57. if (message) {
  58. return <div className='flex h-full flex-col items-center justify-center gap-y-4'>
  59. <AppUnavailable className='h-auto w-auto' code={code || t('share.common.appUnavailable')} unknownReason={message} />
  60. <span className='system-sm-regular cursor-pointer text-text-tertiary' onClick={backToHome}>{code === '403' ? t('common.userProfile.logout') : t('share.login.backToHome')}</span>
  61. </div>
  62. }
  63. if (tokenFromUrl) {
  64. return <div className='flex h-full items-center justify-center'>
  65. <Loading />
  66. </div>
  67. }
  68. if (webAppAccessMode === AccessMode.PUBLIC && redirectUrl) {
  69. return <div className='flex h-full items-center justify-center'>
  70. <Loading />
  71. </div>
  72. }
  73. return <>{children}</>
  74. }
  75. export default Splash