mirror of
https://gitee.com/nocobase/nocobase.git
synced 2025-05-05 13:39:24 +08:00
* feat: init * fix: mobile layout * feat: more code * feat: improve navigate bar * fix: mobile title * feat: improve code * fix: add settings and initailzer * fix: settings * fix: tabbar items settings * feat: tabbar initializer * fix: api * fix: styles * feat: navbar * feat: navigate bar tabs initializer * feat: navigate bar tab settings * feat: navigation bar actions * fix: bug * fix: bug * fix: bug * fix: tabbar active * fix: bug * fix: mobile login and layout * fix: update version * fix: build error * feat: plugin settings support link * fix: add mobile meta * fix: desktop mode * fix: remove old code and change collection name and mobile path * fix: tabbar and tabs initialer layout * fix: initializer style * fix: adjust schema position * fix: mobile style * fix: delete relation resource and home page bug * fix: support multi app * fix: not found page * fix: js bridge * fix: bug * fix: navigation bar schema flat * fix: navigation bar action style * fix: change version * fix: mobile meta and real mobile test * refactor: folder and name * fix: navigation bar sticky and zIndex * fix: full mobile schema * fix: mobile readme and package.json * fix: e2e bug * fix: bug * fix: tabbar style on productino * fix: bug * fix: rename MobileTabBar.Page * fix: support tabbar sort * fix: support page tabs sort * fix: i18n * fix: settings utils import bug * docs: api doc * fix: qrcode refresh * test: unit tests * fix: bug * fix: unit test * fix: build bug * fix: e2e test * fix: overflow scroll * fix: bug * fix: scroll and overflow * fix: bug * fix: e2e expect await * fix: e2e bug * fix: bug * fix: change name * fix: add more e2e * fix: page header * fix: tab support icon * fix: bug * fix: bug * fix: docs * fix(T-4811): scroll bar too long * fix(T-4810): desktop mode * fix: e2e * fix(T-4812): title empty * fix: unit test * feat: hide Open mode option in mobile mode * feat: change default value of Open mode on mobile * feat: add OpenModeProvider * feat: support page mode * fix: fix build * test: update unit tests * chore: remove pro-plugins * fix: bug * fix(T-4812): title is required * fix: bug * fix: bug * fix: bug * fix: bug * refactor: remove z-index * refactor: make better for subpages * fix: drag bug * fix: bug * fix: theme bug * fix(T-4859): create tab bar title empty * fix(T-4857): action too long * fix: e2e bug * fix: remove comment * fix: bug * fix: theme bug * fix: should provider modal component * fix: bug --------- Co-authored-by: chenos <chenlinxh@gmail.com> Co-authored-by: Zeke Zhang <958414905@qq.com>
368 lines
12 KiB
TypeScript
368 lines
12 KiB
TypeScript
/**
|
|
* This file is part of the NocoBase (R) project.
|
|
* Copyright (c) 2020-2024 NocoBase Co., Ltd.
|
|
* Authors: NocoBase Team.
|
|
*
|
|
* This project is dual-licensed under AGPL-3.0 and NocoBase Commercial License.
|
|
* For more information, please refer to: https://www.nocobase.com/agreement.
|
|
*/
|
|
|
|
import { DragOutlined, MenuOutlined, PlusOutlined } from '@ant-design/icons';
|
|
import { css } from '@emotion/css';
|
|
import { useField, useFieldSchema } from '@formily/react';
|
|
import { Space } from 'antd';
|
|
import classNames from 'classnames';
|
|
import React, { FC, useEffect, useMemo, useRef } from 'react';
|
|
import { useTranslation } from 'react-i18next';
|
|
import { SchemaInitializer, SchemaSettings, SchemaToolbarProvider, useSchemaInitializerRender } from '../application';
|
|
import { useSchemaSettingsRender } from '../application/schema-settings/hooks/useSchemaSettingsRender';
|
|
import { useDataSourceManager } from '../data-source/data-source/DataSourceManagerProvider';
|
|
import { useDataSource } from '../data-source/data-source/DataSourceProvider';
|
|
import { DragHandler, useCompile, useDesignable, useGridContext, useGridRowContext } from '../schema-component';
|
|
import { gridRowColWrap } from '../schema-initializer/utils';
|
|
import { SchemaSettingsDropdown } from './SchemaSettings';
|
|
import { useGetAriaLabelOfDesigner } from './hooks/useGetAriaLabelOfDesigner';
|
|
import { useStyles } from './styles';
|
|
|
|
const titleCss = css`
|
|
pointer-events: none;
|
|
position: absolute;
|
|
font-size: 12px;
|
|
/* background: var(--colorSettings);
|
|
color: #fff; */
|
|
padding: 0;
|
|
line-height: 16px;
|
|
height: 16px;
|
|
border-bottom-right-radius: 2px;
|
|
border-radius: 2px;
|
|
top: 2px;
|
|
left: 2px;
|
|
.title-tag {
|
|
padding: 0 3px;
|
|
border-radius: 2px;
|
|
background: var(--colorSettings);
|
|
color: #fff;
|
|
display: block;
|
|
}
|
|
`;
|
|
|
|
const overrideAntdCSS = css`
|
|
& .ant-space-item .anticon {
|
|
margin: 0;
|
|
}
|
|
|
|
&:hover {
|
|
display: block !important;
|
|
}
|
|
`;
|
|
|
|
export interface GeneralSchemaDesignerProps {
|
|
disableInitializer?: boolean;
|
|
title?: string;
|
|
template?: any;
|
|
schemaSettings?: string;
|
|
contextValue?: any;
|
|
/**
|
|
* @default true
|
|
*/
|
|
draggable?: boolean;
|
|
showDataSource?: boolean;
|
|
}
|
|
|
|
/**
|
|
* @deprecated use `SchemaToolbar` instead
|
|
*/
|
|
export const GeneralSchemaDesigner: FC<GeneralSchemaDesignerProps> = (props: any) => {
|
|
const fieldSchema = useFieldSchema();
|
|
const {
|
|
disableInitializer,
|
|
title,
|
|
template,
|
|
schemaSettings,
|
|
contextValue,
|
|
draggable = true,
|
|
showDataSource = true,
|
|
} = { ...props, ...(fieldSchema['x-toolbar-props'] || {}) } as GeneralSchemaDesignerProps;
|
|
const { dn, designable } = useDesignable();
|
|
const field = useField();
|
|
const { t } = useTranslation();
|
|
const compile = useCompile();
|
|
const { getAriaLabel } = useGetAriaLabelOfDesigner();
|
|
const schemaSettingsProps = {
|
|
dn,
|
|
field,
|
|
fieldSchema,
|
|
};
|
|
const { render: schemaSettingsRender, exists: schemaSettingsExists } = useSchemaSettingsRender(
|
|
fieldSchema['x-settings'] || schemaSettings,
|
|
fieldSchema['x-settings-props'],
|
|
);
|
|
const rowCtx = useGridRowContext();
|
|
const ctx = useGridContext();
|
|
const dm = useDataSourceManager();
|
|
const dataSources = dm?.getDataSources();
|
|
const dataSourceContext = useDataSource();
|
|
const dataSource = dataSources?.length > 1 && dataSourceContext;
|
|
const templateName = ['FormItem', 'ReadPrettyFormItem'].includes(template?.componentName)
|
|
? `${template?.name} ${t('(Fields only)')}`
|
|
: template?.name;
|
|
const initializerProps = useMemo(() => {
|
|
return {
|
|
insertPosition: 'afterEnd',
|
|
wrap: rowCtx?.cols?.length > 1 ? undefined : gridRowColWrap,
|
|
Component: (props: any) => (
|
|
<PlusOutlined
|
|
{...props}
|
|
role="button"
|
|
aria-label={getAriaLabel('schema-initializer')}
|
|
style={{ cursor: 'pointer', fontSize: 14 }}
|
|
/>
|
|
),
|
|
};
|
|
}, [getAriaLabel, rowCtx?.cols?.length]);
|
|
|
|
if (!designable) {
|
|
return null;
|
|
}
|
|
return (
|
|
<SchemaToolbarProvider {...contextValue}>
|
|
<div className={classNames('general-schema-designer', overrideAntdCSS)}>
|
|
{title && (
|
|
<div className={classNames('general-schema-designer-title', titleCss)}>
|
|
<Space size={2}>
|
|
<span className={'title-tag'}>
|
|
{showDataSource && dataSource
|
|
? `${compile(dataSource?.displayName)} > ${compile(title)}`
|
|
: compile(title)}
|
|
</span>
|
|
{template && (
|
|
<span className={'title-tag'}>
|
|
{t('Reference template')}: {templateName || t('Untitled')}
|
|
</span>
|
|
)}
|
|
</Space>
|
|
</div>
|
|
)}
|
|
<div className={'general-schema-designer-icons'}>
|
|
<Space size={3} align={'center'}>
|
|
{draggable && (
|
|
<DragHandler>
|
|
<DragOutlined role="button" aria-label={getAriaLabel('drag-handler')} />
|
|
</DragHandler>
|
|
)}
|
|
{!disableInitializer &&
|
|
(ctx?.InitializerComponent ? (
|
|
<ctx.InitializerComponent {...initializerProps} />
|
|
) : (
|
|
ctx?.renderSchemaInitializer?.(initializerProps)
|
|
))}
|
|
{schemaSettingsExists ? (
|
|
schemaSettingsRender(contextValue)
|
|
) : (
|
|
<SchemaSettingsDropdown
|
|
title={
|
|
<MenuOutlined
|
|
role="button"
|
|
aria-label={getAriaLabel('schema-settings')}
|
|
style={{ cursor: 'pointer', fontSize: 12 }}
|
|
/>
|
|
}
|
|
{...schemaSettingsProps}
|
|
>
|
|
{props.children}
|
|
</SchemaSettingsDropdown>
|
|
)}
|
|
</Space>
|
|
</div>
|
|
</div>
|
|
</SchemaToolbarProvider>
|
|
);
|
|
};
|
|
|
|
export interface SchemaToolbarProps {
|
|
title?: string | string[];
|
|
draggable?: boolean;
|
|
initializer?: string | SchemaInitializer<any> | false;
|
|
settings?: string | SchemaSettings<any> | false;
|
|
/**
|
|
* @default true
|
|
*/
|
|
showBorder?: boolean;
|
|
showBackground?: boolean;
|
|
toolbarClassName?: string;
|
|
toolbarStyle?: React.CSSProperties;
|
|
spaceWrapperClassName?: string;
|
|
spaceWrapperStyle?: React.CSSProperties;
|
|
spaceClassName?: string;
|
|
spaceStyle?: React.CSSProperties;
|
|
}
|
|
|
|
const InternalSchemaToolbar: FC<SchemaToolbarProps> = (props) => {
|
|
const fieldSchema = useFieldSchema();
|
|
const {
|
|
title,
|
|
initializer,
|
|
settings,
|
|
showBackground,
|
|
spaceWrapperClassName,
|
|
spaceWrapperStyle,
|
|
showBorder = true,
|
|
draggable = true,
|
|
spaceClassName,
|
|
spaceStyle,
|
|
toolbarClassName,
|
|
toolbarStyle = {},
|
|
} = {
|
|
...props,
|
|
...(fieldSchema?.['x-toolbar-props'] || {}),
|
|
} as SchemaToolbarProps;
|
|
const { designable } = useDesignable();
|
|
const compile = useCompile();
|
|
const { styles } = useStyles();
|
|
const { t } = useTranslation();
|
|
const { getAriaLabel } = useGetAriaLabelOfDesigner();
|
|
const dm = useDataSourceManager();
|
|
const dataSources = dm?.getDataSources();
|
|
const dataSourceContext = useDataSource();
|
|
const dataSource = dataSources?.length > 1 && dataSourceContext;
|
|
|
|
const titleArr = useMemo(() => {
|
|
if (!title) return undefined;
|
|
if (typeof title === 'string') return [compile(title)];
|
|
if (Array.isArray(title)) return title.map((item) => compile(item));
|
|
}, [compile, title]);
|
|
const { render: schemaSettingsRender, exists: schemaSettingsExists } = useSchemaSettingsRender(
|
|
settings || fieldSchema?.['x-settings'],
|
|
fieldSchema?.['x-settings-props'],
|
|
);
|
|
const { render: schemaInitializerRender, exists: schemaInitializerExists } = useSchemaInitializerRender(
|
|
initializer || fieldSchema?.['x-initializer'],
|
|
fieldSchema?.['x-initializer-props'],
|
|
);
|
|
const rowCtx = useGridRowContext();
|
|
const gridContext = useGridContext();
|
|
const initializerProps: any = useMemo(() => {
|
|
return {
|
|
insertPosition: 'afterEnd',
|
|
wrap: rowCtx?.cols?.length === 1 ? gridRowColWrap : undefined,
|
|
Component: (props: any) => (
|
|
<PlusOutlined
|
|
{...props}
|
|
role="button"
|
|
aria-label={getAriaLabel('schema-initializer')}
|
|
style={{ cursor: 'pointer', fontSize: 14 }}
|
|
/>
|
|
),
|
|
};
|
|
}, [getAriaLabel, rowCtx?.cols?.length]);
|
|
|
|
const dragElement = useMemo(() => {
|
|
if (draggable === false) return null;
|
|
return (
|
|
<DragHandler>
|
|
<DragOutlined role="button" aria-label={getAriaLabel('drag-handler')} />
|
|
</DragHandler>
|
|
);
|
|
}, [draggable, getAriaLabel]);
|
|
|
|
const initializerElement = useMemo(() => {
|
|
if (initializer === false) return null;
|
|
if (schemaInitializerExists) {
|
|
return schemaInitializerRender(initializerProps);
|
|
}
|
|
if (gridContext?.InitializerComponent || gridContext?.renderSchemaInitializer) {
|
|
return gridContext?.InitializerComponent ? (
|
|
<gridContext.InitializerComponent {...initializerProps} />
|
|
) : (
|
|
gridContext.renderSchemaInitializer?.(initializerProps)
|
|
);
|
|
}
|
|
}, [gridContext, initializer, initializerProps, schemaInitializerExists, schemaInitializerRender]);
|
|
|
|
const settingsElement = useMemo(() => {
|
|
return settings !== false && schemaSettingsExists ? schemaSettingsRender() : null;
|
|
}, [schemaSettingsExists, schemaSettingsRender, settings]);
|
|
|
|
const toolbarRef = useRef<HTMLDivElement>(null);
|
|
|
|
useEffect(() => {
|
|
const toolbarElement = toolbarRef.current;
|
|
let parentElement = toolbarElement?.parentElement;
|
|
while (parentElement && window.getComputedStyle(parentElement).height === '0px') {
|
|
parentElement = parentElement.parentElement;
|
|
}
|
|
if (!parentElement) {
|
|
return;
|
|
}
|
|
|
|
function show() {
|
|
if (toolbarElement) {
|
|
toolbarElement.style.display = 'block';
|
|
}
|
|
}
|
|
|
|
function hide() {
|
|
if (toolbarElement) {
|
|
toolbarElement.style.display = 'none';
|
|
}
|
|
}
|
|
|
|
const style = window.getComputedStyle(parentElement);
|
|
if (style.position === 'static') {
|
|
parentElement.style.position = 'relative';
|
|
}
|
|
|
|
parentElement.addEventListener('mouseenter', show);
|
|
parentElement.addEventListener('mouseleave', hide);
|
|
|
|
return () => {
|
|
parentElement.removeEventListener('mouseenter', show);
|
|
parentElement.removeEventListener('mouseleave', hide);
|
|
};
|
|
}, []);
|
|
|
|
if (!designable) {
|
|
return null;
|
|
}
|
|
|
|
return (
|
|
<div
|
|
ref={toolbarRef}
|
|
className={classNames(styles.toolbar, toolbarClassName, 'schema-toolbar')}
|
|
style={{ border: showBorder ? 'auto' : 0, background: showBackground ? 'auto' : 0, ...toolbarStyle }}
|
|
>
|
|
{titleArr && (
|
|
<div className={styles.toolbarTitle}>
|
|
<Space size={2}>
|
|
<span key={titleArr[0]} className={styles.toolbarTitleTag}>
|
|
{dataSource ? `${compile(dataSource?.displayName)} > ${titleArr[0]}` : titleArr[0]}
|
|
</span>
|
|
{titleArr[1] && (
|
|
<span className={styles.toolbarTitleTag}>
|
|
{`${t('Reference template')}: ${`${titleArr[1]}` || t('Untitled')}`}
|
|
</span>
|
|
)}
|
|
</Space>
|
|
</div>
|
|
)}
|
|
<div className={classNames(styles.toolbarIcons, spaceWrapperClassName)} style={spaceWrapperStyle}>
|
|
<Space size={3} align={'center'} className={spaceClassName} style={spaceStyle}>
|
|
{dragElement}
|
|
{initializerElement}
|
|
{settingsElement}
|
|
</Space>
|
|
</div>
|
|
</div>
|
|
);
|
|
};
|
|
|
|
export const SchemaToolbar: FC<SchemaToolbarProps> = (props) => {
|
|
const { designable } = useDesignable();
|
|
|
|
if (!designable) {
|
|
return null;
|
|
}
|
|
|
|
return <InternalSchemaToolbar {...props} />;
|
|
};
|