|
| 1 | +/* eslint-disable i18next/no-literal-string -- will not support i18n in error boundaries */ |
| 2 | +import {Card, Container, Heading, Stack, Text} from '@sanity/ui' |
| 3 | +import {type ReactNode, useCallback, useEffect, useState} from 'react' |
| 4 | +import {type ViteHotContext} from 'vite/types/hot.js' |
| 5 | + |
| 6 | +const ERROR_TITLE = 'Dev server stopped' |
| 7 | +const ERROR_DESCRIPTION = |
| 8 | + 'The development server has stopped. You may need to restart it to continue working.' |
| 9 | + |
| 10 | +class ViteDevServerStoppedError extends Error { |
| 11 | + ViteDevServerStoppedError: boolean |
| 12 | + |
| 13 | + constructor() { |
| 14 | + super(ERROR_TITLE) |
| 15 | + this.name = 'ViteDevServerStoppedError' |
| 16 | + this.ViteDevServerStoppedError = true |
| 17 | + } |
| 18 | +} |
| 19 | +const serverHot = import.meta.hot |
| 20 | +const isViteServer = (hot: unknown): hot is ViteHotContext => Boolean(hot) |
| 21 | + |
| 22 | +const useDetectViteDevServerStopped = () => { |
| 23 | + const [devServerStopped, setDevServerStopped] = useState(false) |
| 24 | + |
| 25 | + const markDevServerStopped = useCallback(() => setDevServerStopped(true), []) |
| 26 | + |
| 27 | + useEffect(() => { |
| 28 | + // no early return to optimize tree-shaking |
| 29 | + if (isViteServer(serverHot)) { |
| 30 | + serverHot.on('vite:ws:disconnect', markDevServerStopped) |
| 31 | + } |
| 32 | + }, [markDevServerStopped]) |
| 33 | + |
| 34 | + return {devServerStopped} |
| 35 | +} |
| 36 | + |
| 37 | +const ThrowViteServerStopped = () => { |
| 38 | + const {devServerStopped} = useDetectViteDevServerStopped() |
| 39 | + |
| 40 | + if (devServerStopped) throw new ViteDevServerStoppedError() |
| 41 | + |
| 42 | + return null |
| 43 | +} |
| 44 | + |
| 45 | +export const DetectViteDevServerStopped = (): ReactNode => |
| 46 | + isViteServer(serverHot) ? <ThrowViteServerStopped /> : null |
| 47 | + |
| 48 | +export const DevServerStoppedErrorScreen = (): ReactNode => ( |
| 49 | + <Card |
| 50 | + height="fill" |
| 51 | + overflow="auto" |
| 52 | + paddingY={[4, 5, 6, 7]} |
| 53 | + paddingX={4} |
| 54 | + sizing="border" |
| 55 | + tone="critical" |
| 56 | + > |
| 57 | + <Container width={3}> |
| 58 | + <Stack space={4}> |
| 59 | + <Heading>{ERROR_TITLE}</Heading> |
| 60 | + |
| 61 | + <Card border radius={2} overflow="auto" padding={4} tone="inherit"> |
| 62 | + <Stack space={4}> |
| 63 | + <Text size={2}>{ERROR_DESCRIPTION}</Text> |
| 64 | + </Stack> |
| 65 | + </Card> |
| 66 | + </Stack> |
| 67 | + </Container> |
| 68 | + </Card> |
| 69 | +) |
0 commit comments