import React, { Component, ComponentType, PropsWithChildren } from "react"; import { ErrorFallback, ErrorFallbackProps } from "@/components/ErrorFallback"; export type ErrorBoundaryProps = PropsWithChildren<{ FallbackComponent?: ComponentType; onError?: (error: Error, stackTrace: string) => void; }>; type ErrorBoundaryState = { error: Error | null }; /** * This is a special case for for using the class components. Error boundaries must be class components because React only provides error boundary functionality through lifecycle methods (componentDidCatch and getDerivedStateFromError) which are not available in functional components. * https://react.dev/reference/react/Component#catching-rendering-errors-with-an-error-boundary */ export class ErrorBoundary extends Component< ErrorBoundaryProps, ErrorBoundaryState > { state: ErrorBoundaryState = { error: null }; static defaultProps: { FallbackComponent: ComponentType; } = { FallbackComponent: ErrorFallback, }; static getDerivedStateFromError(error: Error): ErrorBoundaryState { return { error }; } componentDidCatch(error: Error, info: { componentStack: string }): void { if (typeof this.props.onError === "function") { this.props.onError(error, info.componentStack); } } resetError = (): void => { this.setState({ error: null }); }; render() { const { FallbackComponent } = this.props; return this.state.error && FallbackComponent ? ( ) : ( this.props.children ); } }