mirror of
https://gitee.com/nocobase/nocobase.git
synced 2025-05-08 15:09:27 +08:00
* fix: form * refactor: schema-initializer * fix: bug * refactor: schema initializer * refactor: rename * fix: delete SchemaInitializerProvider * refactor: props `insert` to hooks `useSchemaInitializerV2` * fix: bug * refactor: delete `SchemaInitializer.Button` * refactor: delete old SchemaInitializer * fix: bug * fix: workflow * fix: docs * fix: bug * fix: bug * feat: style * fix: remove v2 * fix: visible * fix: bug * fix: item hook * feat: item hook * fix: add search DataBlockInitializer * fix: build bug * fix: style bug * fix: style bug * fix: test bug * fix: test bug * fix: rerender bug * fix: remove menu select * fix: bug * chore: add aria-label for SchemaInitializerButton * refactor: rename name to camel case * fix: menu height bug * fix: build errors * fix: build errors * fix: bug * fix: bug * fix: performance * test: add test for header * fix: sidebar is not refresh (T-2422) * feat(e2e): support to add group page and link page * chore: make sure the page is configurable when using page.goto * test: add tests for menu initializer * fix: imporve code * chore: fix build error * chore: optimize locator of menu item * refactor: rename testid for select * test: make tests passing * fix: make tests passing * chore: upgrade vitest to v0.34.6 * chore: increase timeout of e2e * feat: core * fix: revert schema initializer demos * test: menu, page tabs, page grid, table column * fix: schema button interface * feat: refactor: page tab settings * feat: page settings * fix: dumirc * fix: export CSSVariableProvider * feat: lazy render * fix: form-item * fix: general schema desinger * feat: filter form item settings * refactor: form-v2 schema settings * refactor: form-v1 schema settings * refactor: action schema settings * fix: action bug * fix: form-item bug * fix: types error * docs: schema settings doc * docs: schema settings * feat: schema setting item add name * fix: visible lazy render bug * fix: revert form item filter * fix: test bug * fix: test JSON.parse bug * fix: test bug * fix: improve styling * fix: styling * fix: cleanup * fix: token.borderRadiusSM * fix: bug * test: add tests * fix: style bug * fix: add chart performance * feat: add SchemaDesignerContext * fix: bug * fix: test bug * style: create record action style improve * fix: make test passing * chore: mack tests passing * chore: make tests passing * test: fix tests * style: style revert * fix: bug * fix: data selector * fix: fix tests * fix: fix tests * fix: delete PluginManagerContext * refactor: improve router and add SchemaComponentProvider & CSSVariableProvider to MainComponent * fix: add dn and field builtin to SchemaSettingWrapper * feat: update docs * refactor: application providers * fix: test bug * fix: fix tests * chore: make test passing * feat: update docs * chore: rename collection name * feat: update docs * chore: skip weird test * fix: blockInitializers media to otherBlocks * fix: cancel to skip test * fix: bug * test: add test * refactor: migrate to small files * test: add tests for form block settings * chore: format * fix: add chart scroll bug * refactor: action designer improve * refactor: formitem designer schemaSetting * feat: schemaSettingsManager and schemaInitializerManager addItem and removeItem * test: add tests for color field in creating block * test: add tests for email field in creating block * test: make tests passing * perf: reduce fields number * fix: sub menu bug * test: add tests basic in editing form * test: add tests basic in details form * fix: improve code * test: make tests passing * test(plugin-mock-collections): add color for enum options * refactor: improve code * fix: bug * fix: bug * refactor: convert parameters to destructured object * test: add tests choices * test: add tests media * test: add tests for datetime in creating form * feat(plugin-mock-collection): generate faker time * test: add tests for datetime in editing form * test: add tests for datetime in details form * fix: bug * feat: improve code * test: add tests for relation fields * fix: rename SchemaSettings * fix: type bug * refactor: useDesinger() * fix: bug * fix: bug * fix: build tip * fix: designableState * fix: bug * fix: designable * fix: designable * test: add tests for relation fields * test: add tests for relation fields * test: add tests for relation fields * feat: client api doc * test: add tests for relation fields * test: avoid errors * test: make tests passing * fix: bug * test: make tests passing * test: add tests for advanced fields * test: increase e2e timeout-minutes to 60 * fix: bug * fix: improve code * feat: add schema initailizer component demos * test: make tests passing * fix: schema settings demos * feat: shallowMerge & deepMerge * test: reduce number of tests * test: make tests passing * feat: updates * fix: add Initializer Internal * demos: useSchemaSettingsRender * test: make tests passing * test: make tests passing * fix: improve docs * fix: bug * chore: upgrade dumi theme * test: make tests passing * test: add tests for linkage rules * test: add test for form data templates * test: add tests for default value * test: reduce number of tests * fix: dn.deepMerge * fix: bug * fix: bug * fix: toolbar * fix: docs ssr * test: add tests for system fields * test: add tests for actions * fix: bug * test: add tests for lazy loading of variables * test: make testing more stable * fix: update docs * fix: bug --------- Co-authored-by: Rain <958414905@qq.com> Co-authored-by: chenos <chenlinxh@gmail.com> Co-authored-by: katherinehhh <katherine_15995@163.com>
152 lines
4.1 KiB
TypeScript
152 lines
4.1 KiB
TypeScript
import set from 'lodash/set';
|
|
import React, { ComponentType } from 'react';
|
|
import {
|
|
BrowserRouter,
|
|
BrowserRouterProps,
|
|
HashRouter,
|
|
HashRouterProps,
|
|
MemoryRouter,
|
|
MemoryRouterProps,
|
|
RouteObject,
|
|
useRoutes,
|
|
} from 'react-router-dom';
|
|
import { BlankComponent, RouterContextCleaner } from './components';
|
|
|
|
export interface BrowserRouterOptions extends Omit<BrowserRouterProps, 'children'> {
|
|
type?: 'browser';
|
|
}
|
|
export interface HashRouterOptions extends Omit<HashRouterProps, 'children'> {
|
|
type?: 'hash';
|
|
}
|
|
export interface MemoryRouterOptions extends Omit<MemoryRouterProps, 'children'> {
|
|
type?: 'memory';
|
|
}
|
|
export type RouterOptions = (HashRouterOptions | BrowserRouterOptions | MemoryRouterOptions) & {
|
|
renderComponent?: RenderComponentType;
|
|
};
|
|
export type ComponentTypeAndString<T = any> = ComponentType<T> | string;
|
|
export interface RouteType extends Omit<RouteObject, 'children' | 'Component'> {
|
|
Component?: ComponentTypeAndString;
|
|
}
|
|
export type RenderComponentType = (Component: ComponentTypeAndString, props?: any) => React.ReactNode;
|
|
|
|
export class RouterManager {
|
|
protected routes: Record<string, RouteType> = {};
|
|
|
|
constructor(protected options: RouterOptions) {
|
|
this.options = options || {};
|
|
}
|
|
|
|
getRoutesTree(): RouteObject[] {
|
|
type RouteTypeWithChildren = RouteType & { children?: RouteTypeWithChildren };
|
|
const routes: Record<string, RouteTypeWithChildren> = {};
|
|
|
|
/**
|
|
* { 'a': { name: '1' }, 'a.b': { name: '2' }, 'a.c': { name: '3' } };
|
|
* =>
|
|
* { a: { name: '1', children: { b: { name: '2' }, c: {name: '3'} } } }
|
|
*/
|
|
for (const [name, route] of Object.entries(this.routes)) {
|
|
set(routes, name.split('.').join('.children.'), route);
|
|
}
|
|
|
|
/**
|
|
* get RouteObject tree
|
|
*
|
|
* @example
|
|
* { a: { name: '1', children: { b: { name: '2' }, c: { children: { d: { name: '3' } } } } } }
|
|
* =>
|
|
* { name: '1', children: [{ name: '2' }, { name: '3' }] }
|
|
*/
|
|
const buildRoutesTree = (routes: RouteTypeWithChildren): RouteObject[] => {
|
|
return Object.values(routes).reduce<RouteObject[]>((acc, item) => {
|
|
if (Object.keys(item).length === 1 && item.children) {
|
|
acc.push(...buildRoutesTree(item.children));
|
|
} else {
|
|
const { Component, element, children, ...reset } = item;
|
|
let ele = element;
|
|
if (Component) {
|
|
if (typeof Component === 'string') {
|
|
ele = this.options.renderComponent ? this.options.renderComponent(Component) : Component;
|
|
} else {
|
|
ele = React.createElement(Component);
|
|
}
|
|
}
|
|
const res = {
|
|
...reset,
|
|
element: ele,
|
|
children: children ? buildRoutesTree(children) : undefined,
|
|
} as RouteObject;
|
|
acc.push(res);
|
|
}
|
|
return acc;
|
|
}, []);
|
|
};
|
|
|
|
return buildRoutesTree(routes);
|
|
}
|
|
|
|
getRoutes() {
|
|
return this.routes;
|
|
}
|
|
|
|
setType(type: RouterOptions['type']) {
|
|
this.options.type = type;
|
|
}
|
|
|
|
setBasename(basename: string) {
|
|
this.options.basename = basename;
|
|
}
|
|
|
|
getRouterComponent() {
|
|
const { type = 'browser', ...opts } = this.options || {};
|
|
const Routers = {
|
|
hash: HashRouter,
|
|
browser: BrowserRouter,
|
|
memory: MemoryRouter,
|
|
};
|
|
|
|
const ReactRouter = Routers[type];
|
|
|
|
const RenderRoutes = () => {
|
|
const routes = this.getRoutesTree();
|
|
const element = useRoutes(routes);
|
|
return element;
|
|
};
|
|
|
|
const RenderRouter: React.FC<{ BaseLayout?: ComponentType }> = ({ BaseLayout = BlankComponent }) => {
|
|
return (
|
|
<RouterContextCleaner>
|
|
<ReactRouter {...opts}>
|
|
<BaseLayout>
|
|
<RenderRoutes />
|
|
</BaseLayout>
|
|
</ReactRouter>
|
|
</RouterContextCleaner>
|
|
);
|
|
};
|
|
|
|
return RenderRouter;
|
|
}
|
|
|
|
add(name: string, route: RouteType) {
|
|
this.routes[name] = route;
|
|
}
|
|
|
|
get(name: string) {
|
|
return this.routes[name];
|
|
}
|
|
|
|
has(name: string) {
|
|
return !!this.get(name);
|
|
}
|
|
|
|
remove(name: string) {
|
|
delete this.routes[name];
|
|
}
|
|
}
|
|
|
|
export function createRouterManager(options?: RouterOptions) {
|
|
return new RouterManager(options);
|
|
}
|