import { DisconnectOutlined, LoadingOutlined } from '@ant-design/icons';
import { css } from '@emotion/css';
import { observer } from '@formily/reactive-react';
import { getSubAppName } from '@nocobase/sdk';
import { Button, Modal, Result, Spin } from 'antd';
import React, { FC } from 'react';
import { Navigate, useNavigate } from 'react-router-dom';
import { ACLPlugin } from '../acl';
import { useAPIClient } from '../api-client';
import { Application } from '../application';
import { Plugin } from '../application/Plugin';
import { BlockSchemaComponentPlugin } from '../block-provider';
import { CollectionPlugin } from '../collection-manager';
import { RemoteDocumentTitlePlugin } from '../document-title';
import { PinnedListPlugin } from '../plugin-manager';
import { PMPlugin } from '../pm';
import { AdminLayoutPlugin, RouteSchemaComponent } from '../route-switch';
import { AntdSchemaComponentPlugin, SchemaComponentPlugin } from '../schema-component';
import { ErrorFallback } from '../schema-component/antd/error-fallback';
import { AssociationFilterPlugin, SchemaInitializerPlugin } from '../schema-initializer';
import { SchemaSettingsPlugin } from '../schema-settings';
import { BlockTemplateDetails, BlockTemplatePage } from '../schema-templates';
import { SystemSettingsPlugin } from '../system-settings';
import { CurrentUserProvider, CurrentUserSettingsMenuProvider } from '../user';
import { LocalePlugin } from './plugins/LocalePlugin';
const AppSpin = () => {
return (
);
};
const useErrorProps = (app: Application, error: any) => {
const api = useAPIClient();
if (!error) {
return {};
}
const err = error?.response?.data?.errors?.[0] || error;
const subApp = getSubAppName(app.getPublicPath());
switch (err.code) {
case 'USER_HAS_NO_ROLES_ERR':
return {
title: app.i18n.t('Permission denied'),
subTitle: err.message,
extra: [
,
subApp ? (
) : null,
],
};
default:
return {};
}
};
const AppError: FC<{ error: Error; app: Application }> = observer(
({ app, error }) => {
const props = useErrorProps(app, error);
return (
window.location.reload()}>
{app.i18n.t('Try again')}
,
]}
{...props}
/>
);
},
{ displayName: 'AppError' },
);
const getProps = (app: Application) => {
if (app.ws.serverDown) {
return {
status: 'error',
icon: ,
title: "You're offline",
subTitle: 'Please check the server status or network connection status',
};
}
if (!app.error) {
return {};
}
if (app.error.code === 'APP_NOT_FOUND') {
return {
status: 'warning',
title: 'App not found',
subTitle: app.error?.message,
};
}
if (app.error.code === 'APP_INITIALIZING') {
return {
status: 'info',
icon: ,
title: 'App initializing',
subTitle: app.error?.message,
};
}
if (app.error.code === 'APP_INITIALIZED') {
return {
status: 'warning',
title: 'App initialized',
subTitle: app.error?.message,
};
}
if (app.error.code === 'APP_ERROR' || app.error.code === 'LOAD_ERROR') {
return {
status: 'error',
title: 'App error',
subTitle: app.error?.message,
};
}
if (app.error.code === 'APP_NOT_INSTALLED_ERROR') {
return {
status: 'warning',
title: 'App not installed',
subTitle: app.error?.message,
};
}
if (app.error.code === 'APP_STOPPED') {
return {
status: 'warning',
title: 'App stopped',
subTitle: app.error?.message,
};
}
if (app.error.code === 'APP_COMMANDING') {
const props = {
status: 'info',
icon: ,
title: app.error?.command?.name,
subTitle: app.error?.message,
};
const commands = {
start: {
title: 'App starting',
},
restart: {
title: 'App restarting',
},
install: {
title: 'App installing',
},
upgrade: {
title: 'App upgrading',
},
'pm.add': {
title: 'Adding plugin',
},
'pm.update': {
title: 'Updating plugin',
},
'pm.enable': {
title: 'Enabling plugin',
},
'pm.disable': {
title: 'Disabling plugin',
},
'pm.remove': {
title: 'Removing plugin',
},
};
return { ...props, ...commands[app.error?.command?.name] };
}
return {};
};
const AppMaintaining: FC<{ app: Application; error: Error }> = observer(
({ app }) => {
const { icon, status, title, subTitle } = getProps(app);
return (
{app.i18n.t(subTitle)}
}
// extra={[
// ,
// ]}
/>
);
},
{ displayName: 'AppMaintaining' },
);
const AppMaintainingDialog: FC<{ app: Application; error: Error }> = observer(
({ app }) => {
const { icon, status, title, subTitle } = getProps(app);
return (
);
},
{ displayName: 'AppMaintainingDialog' },
);
const AppNotFound = () => {
const navigate = useNavigate();
return (
navigate('/', { replace: true })} type="primary">
Back Home
}
/>
);
};
export class NocoBaseBuildInPlugin extends Plugin {
async afterAdd() {
this.app.addComponents({
AppSpin,
AppError,
AppMaintaining,
AppMaintainingDialog,
AppNotFound,
});
await this.addPlugins();
}
async load() {
this.addComponents();
this.addRoutes();
this.app.use(CurrentUserProvider);
this.app.use(CurrentUserSettingsMenuProvider);
}
addRoutes() {
this.router.add('root', {
path: '/',
element: ,
});
this.router.add('not-found', {
path: '*',
Component: AppNotFound,
});
this.router.add('admin', {
path: '/admin',
Component: 'AdminLayout',
});
this.router.add('admin.page', {
path: '/admin/:name',
Component: 'RouteSchemaComponent',
});
}
addComponents() {
this.app.addComponents({
ErrorFallback,
RouteSchemaComponent,
BlockTemplatePage,
BlockTemplateDetails,
});
}
async addPlugins() {
await this.app.pm.add(AssociationFilterPlugin);
await this.app.pm.add(LocalePlugin, { name: 'builtin-locale' });
await this.app.pm.add(AdminLayoutPlugin, { name: 'admin-layout' });
await this.app.pm.add(SystemSettingsPlugin, { name: 'system-setting' });
await this.app.pm.add(PinnedListPlugin, {
name: 'pinned-list',
config: {
items: {
ui: { order: 100, component: 'DesignableSwitch', pin: true, snippet: 'ui.*' },
pm: { order: 200, component: 'PluginManagerLink', pin: true, snippet: 'pm' },
sc: { order: 300, component: 'SettingsCenterDropdown', pin: true, snippet: 'pm.*' },
},
},
});
await this.app.pm.add(SchemaComponentPlugin, { name: 'schema-component' });
await this.app.pm.add(SchemaInitializerPlugin, { name: 'schema-initializer' });
await this.app.pm.add(SchemaSettingsPlugin, { name: 'schema-settings' });
await this.app.pm.add(BlockSchemaComponentPlugin, { name: 'block-schema-component' });
await this.app.pm.add(AntdSchemaComponentPlugin, { name: 'antd-schema-component' });
await this.app.pm.add(ACLPlugin, { name: 'builtin-acl' });
await this.app.pm.add(RemoteDocumentTitlePlugin, { name: 'remote-document-title' });
await this.app.pm.add(PMPlugin, { name: 'builtin-pm' });
await this.app.pm.add(CollectionPlugin, { name: 'builtin-collection' });
}
}