mirror of
https://github.com/toeverything/AFFiNE.git
synced 2026-02-04 00:28:33 +00:00
132 lines
3.1 KiB
TypeScript
132 lines
3.1 KiB
TypeScript
import { RequestError } from '@affine/workspace/affine/api';
|
|
import type { NextRouter } from 'next/router';
|
|
import type { ErrorInfo } from 'react';
|
|
import type React from 'react';
|
|
import { Component } from 'react';
|
|
|
|
import type { BlockSuiteWorkspace } from '../../shared';
|
|
|
|
export type AffineErrorBoundaryProps = React.PropsWithChildren<{
|
|
router: NextRouter;
|
|
}>;
|
|
|
|
export class PageNotFoundError extends TypeError {
|
|
readonly workspace: BlockSuiteWorkspace;
|
|
readonly pageId: string;
|
|
|
|
constructor(workspace: BlockSuiteWorkspace, pageId: string) {
|
|
super();
|
|
this.workspace = workspace;
|
|
this.pageId = pageId;
|
|
}
|
|
}
|
|
|
|
export class WorkspaceNotFoundError extends TypeError {
|
|
readonly workspaceId: string;
|
|
|
|
constructor(workspaceId: string) {
|
|
super();
|
|
this.workspaceId = workspaceId;
|
|
}
|
|
}
|
|
|
|
export class QueryParamError extends TypeError {
|
|
readonly targetKey: string;
|
|
readonly query: unknown;
|
|
|
|
constructor(targetKey: string, query: unknown) {
|
|
super();
|
|
this.targetKey = targetKey;
|
|
this.query = query;
|
|
}
|
|
}
|
|
|
|
export class Unreachable extends Error {
|
|
constructor(message?: string) {
|
|
super(message);
|
|
}
|
|
}
|
|
|
|
type AffineError =
|
|
| QueryParamError
|
|
| Unreachable
|
|
| WorkspaceNotFoundError
|
|
| PageNotFoundError
|
|
| RequestError
|
|
| Error;
|
|
|
|
interface AffineErrorBoundaryState {
|
|
error: AffineError | null;
|
|
}
|
|
|
|
export class AffineErrorBoundary extends Component<
|
|
AffineErrorBoundaryProps,
|
|
AffineErrorBoundaryState
|
|
> {
|
|
public state: AffineErrorBoundaryState = {
|
|
error: null,
|
|
};
|
|
|
|
public static getDerivedStateFromError(
|
|
error: AffineError
|
|
): AffineErrorBoundaryState {
|
|
return { error };
|
|
}
|
|
|
|
public componentDidCatch(error: AffineError, errorInfo: ErrorInfo) {
|
|
console.error('Uncaught error:', error, errorInfo);
|
|
}
|
|
|
|
public render() {
|
|
if (this.state.error) {
|
|
const error = this.state.error;
|
|
if (error instanceof PageNotFoundError) {
|
|
return (
|
|
<>
|
|
<h1>Sorry.. there was an error</h1>
|
|
<>
|
|
<span> Page error </span>
|
|
<span>
|
|
Cannot find page {error.pageId} in workspace{' '}
|
|
{error.workspace.meta.name}
|
|
</span>
|
|
<button
|
|
onClick={() => {
|
|
this.props.router
|
|
.replace({
|
|
pathname: '/workspace/[workspaceId]/[pageId]',
|
|
query: {
|
|
workspaceId: error.workspace.id,
|
|
pageId: error.workspace.meta.pageMetas[0].id,
|
|
},
|
|
})
|
|
.then(() => {
|
|
this.setState({ error: null });
|
|
});
|
|
}}
|
|
>
|
|
{' '}
|
|
refresh{' '}
|
|
</button>
|
|
</>
|
|
</>
|
|
);
|
|
} else if (error instanceof RequestError) {
|
|
return (
|
|
<>
|
|
<h1>Sorry.. there was an error</h1>
|
|
{error.message}
|
|
</>
|
|
);
|
|
}
|
|
return (
|
|
<>
|
|
<h1>Sorry.. there was an error</h1>
|
|
</>
|
|
);
|
|
}
|
|
|
|
return this.props.children;
|
|
}
|
|
}
|