DashKit
用于构建交互式仪表板的网格组件。
DashKit
一个仪表板网格渲染库。
安装
npm i @gravity-ui/dashkit @gravity-ui/uikit
描述
该库用于在网格中排列小部件,调整它们的大小,添加新的小部件和删除它们。 小部件是一个 React 组件。例如,文本、图形和图像。
通过插件系统添加新的小部件。
插件
创建自定义小部件需要插件。
Props
type ItemManipulationCallback = (eventData: {
layout: Layouts;
oldItem: Layout;
newItem: Layout;
placeholder: Layout;
e: MouseEvent;
element: HTMLElement;
}) => void;
interface DashKitProps {
config: Config;
editMode: boolean;
onItemEdit: ({id}: {id: string}) => void;
onChange: (data: {config: Config; itemsStateAndParams: ItemsStateAndParams}) => void;
onDrop: (dropProps: ItemDropProps) => void;
onItemMountChange: (item: ConfigItem, state: {isAsync: boolead; isMounted: boolean}) => void;
onItemRender: (item: ConfigItem) => void;
onDragStart?: ItemManipulationCallback;
onDrag?: ItemManipulationCallback;
onDragStop?: ItemManipulationCallback;
onResizeStart?: ItemManipulationCallback;
onResize?: ItemManipulationCallback;
onResizeStop?: ItemManipulationCallback;
defaultGlobalParams: GlobalParams;
globalParams: GlobalParams;
itemsStateAndParams: ItemsStateAndParams;
settings: SettingsProps;
context: ContextProps;
overlayControls?: Record<string, OverlayControlItem[]> | null;
overlayMenuItems?: MenuItems[] | null;
noOverlay?: boolean;
focusable?: boolean;
onItemFocus: (item: ConfigItem) => void;
onItemBlur: (item: ConfigItem) => void;
draggableHandleClassName?: string;
getPreparedCopyItemOptions?: (options: PreparedCopyItemOptions) => PreparedCopyItemOptions;
onCopyFulfill?: (error: null | Error, data?: PreparedCopyItemOptions) => void;
}
- config: сonfig。
- editMode: 是否启用编辑模式。
- onItemEdit: 当您点击编辑小部件时调用。
- onChange: 当 config 或 itemsStateAndParams 更改时调用。
- onDrop: 当使用 (#DashKitDnDWrapper) 从 ActionPanel 拖放项目时调用
- onItemMountChange: 当项目挂载状态改变时调用
- onItemRender: 当项目渲染完成时调用
- defaultGlobalParams, globalParams: 影响所有小部件的参数。在 DataLens 中,
defaultGlobalParams
是在仪表板设置中设置的全局参数。globalParams
是可以在 url 中设置的全局参数。 - itemsStateAndParams: itemsStateAndParams。
- settings: DashKit 设置。
- context: 将在所有小部件上提供的对象。
- overlayControls: 在编辑时覆盖小部件控件的对象。如果未传输,将显示基本控件。如果传递
null
,则只显示关闭按钮或自定义菜单。 - overlayMenuItems: 自定义下拉菜单项
- noOverlay: 如果为
true
,则在编辑时不显示覆盖层和控件。 - focusable: 如果为
true
,网格项目将可聚焦。 - onItemFocus: 当
focusable
为 true 且项目被聚焦时调用。 - onItemBlur: 当
focusable
为 true 且项目失去焦点时调用。 - draggableHandleClassName: 使小部件可拖动的元素的 CSS 类名。
- onDragStart: 当项目拖动开始时调用的 ReactGridLayout
- onDrag: 项目拖动时调用的 ReactGridLayout
- onDragStop: 当项目拖动停止时调用的 ReactGridLayout
- onResizeStart: 当项目调整大小开始时调用的 ReactGridLayout
- onResize: 项目调整大小时调用的 ReactGridLayout
- onResizeStop: 当项目调整大小停止时调用的 ReactGridLayout
- getPreparedCopyItemOptions: 在将复制的项目保存到本地存储之前,调用此函数将其转换为可序列化对象。应该使用它代替已弃用的
context.getPreparedCopyItemOptions
属性 - onCopyFulfill: 当项目复制完成时,如果操作成功,则使用
error=null
和已定义的data
调用;否则,使用error: Error
而不带data
调用
使用方法
DashKit 配置
在将 DashKit
用作 React 组件之前,必须对其进行配置。
-
设置语言
import {configure, Lang} from '@gravity-ui/uikit'; configure({lang: Lang.En});
-
DashKit.setSettings
用于全局 DashKit 设置(如小部件之间的边距、默认小部件大小和小部件覆盖菜单)
import {DashKit} from '@gravity-ui/dashkit'; DashKit.setSettings({ gridLayout: {margin: [8, 8]}, isMobile: true, // menu: [] as Array<MenuItem>, });
-
DashKit.registerPlugins
注册和配置插件
import {DashKit} from '@gravity-ui/dashkit'; import {pluginTitle, pluginText} from '@gravity-ui/dashkit'; DashKit.registerPlugins( pluginTitle, pluginText.setSettings({ apiHandler({text}) { return api.getMarkdown(text); }, }), ); DashKit.registerPlugins({ type: 'custom', defaultLayout: { w: 10, h: 8, }, renderer: function CustomPlugin() { return <div>Custom widget with custom controls</div>; }, });
Config
export interface Config {
salt: string; // 用于形成唯一 id
counter: number; // 用于形成唯一 id,只增不减
items: ConfigItem[]; // 初始小部件状态
layout: ConfigLayout[]; // 网格上的小部件位置 https://github.com/react-grid-layout
aliases: ConfigAliases; // 参数的别名,参见 #Params
connections: ConfigConnection[]; // 小部件之间的链接,参见 #Params
}
配置示例:
import {DashKitProps} from '@gravity-ui/dashkit';
const config: DashKitProps['config'] = {
salt: '0.46703554571365613',
counter: 4,
items: [
{
id: 'tT',
data: {
size: 'm',
text: 'Caption',
showInTOC: true,
},
type: 'title',
namespace: 'default',
orderId: 1,
},
{
id: 'Ea',
data: {
text: 'mode _editActive',
_editActive: true,
},
type: 'text',
namespace: 'default',
},
{
id: 'zR',
data: {
text: '### Text',
},
type: 'text',
namespace: 'default',
orderId: 0,
},
{
id: 'Dk',
data: {
foo: 'bar',
},
type: 'custom',
namespace: 'default',
orderId: 5,
},
],
layout: [
{
h: 2,
i: 'tT',
w: 36,
x: 0,
y: 0,
},
{
h: 6,
i: 'Ea',
w: 12,
x: 0,
y: 2,
},
{
h: 6,
i: 'zR',
w: 12,
x: 12,
y: 2,
},
{
h: 4,
i: 'Dk',
w: 8,
x: 0,
y: 8,
},
],
aliases: {},
connections: [],
};
向配置添加新项目:
const newLayout = updateLayout: [
{
h: 6,
i: 'Ea',
w: 12,
x: 0,
y: 6,
},
{
h: 4,
i: 'Dk',
w: 8,
x: 0,
y: 12,
},
];
const newConfig = DashKit.setItem({
item: {
data: {
text: `Some text`,
},
namespace: 'default',
type: 'text',
// 可选。如果需要在当前布局中插入具有预定义尺寸的新项目
layout: { // 当前项目插入在 'Ea' 之前
h: 6,
w: 12,
x: 0,
y: 2,
},,
},
config: config,
options: {
// 可选。当从 ActionPanel 拖放新元素时,现有项目的新布局值
updateLayout: newLayout,
},
});
更改配置中的现有项目:
const newConfig = DashKit.setItem({
item: {
id: 'tT', // item.id
data: {
size: 'm',
text: `New caption`,
},
namespace: 'default',
type: 'title',
},
config: config,
});
从配置中删除项目:
import {DashKitProps} from '@gravity-ui/dashkit';
const oldItemsStateAndParams: DashKitProps['itemsStateAndParams'] = {};
const {config: newConfig, itemsStateAndParams} = DashKit.removeItem({
id: 'tT', // item.id
config: config,
itemsStateAndParams: this.state.itemsStateAndParams,
});
Params
type Params = Record<string, string | string[]>;
DashKit
根据小部件、链接和别名的默认参数生成参数。这些参数是 ChartKit 库所必需的。
生成顺序:
defaultGlobalParams
- 默认小部件参数
item.default
globalParams
- 根据队列从 itemsStateAndParams 获取的参数。
itemsStateAndParams
存储小部件参数和状态以及参数更改队列的对象。
它有一个 __meta__
字段,用于存储队列和元信息。
interface StateAndParamsMeta = {
__meta__: {
queue: {id: string}[]; // 队列
version: number; // 当前版本 itemsStateAndParams
};
}
以及小部件状态和参数:
interface ItemsStateAndParamsBase {
[itemId: string]: {
state?: Record<string, any>;
params?: Params;
};
}
type ItemsStateAndParams = StateAndParamsMeta & ItemsStateAndParamsBase;
Menu
您可以在编辑模式下指定自定义 DashKit 小部件覆盖菜单
type MenuItem = {
id: string; // 唯一 id
title?: string; // 字符串标题
icon?: ReactNode; // 图标节点
iconSize?: number | string; // 图标大小,以 px 为单位的数字或带单位的字符串
handler?: (item: ConfigItem) => void; // 自定义项目操作处理程序
visible?: (item: ConfigItem) => boolean; // 用于过滤菜单项的可选可见性处理程序
className?: string; // 自定义类属性
};
// 在设置中使用菜单项数组
<Dashkit overlayMenuItems={[] as Array<MenuItem> | null} />;
[deprecated];
// overlayMenuItems 属性优先于 setSettings 菜单
DashKit.setSettings({menu: [] as Array<MenuItem>});
从 ActionPanel 拖动项目
DashKitDnDWrapper
type DraggedOverItem = {
h: number;
w: number;
type: string;
parent: string;
i?: number;
};
interface DashKitDnDWrapperProps {
dragImageSrc?: string;
onDragStart?: (dragProps: ItemDragProps) => void;
onDragEnd?: () => void;
onDropDragOver?: (
draggedItem: DraggedOverItem,
sharedItem: DraggedOverItem | null,
) => void | boolean;
}
- dragImageSrc: 拖动图像预览,默认使用透明的 1px png base64
- onDragStart: 当从 ActionPanel 拖动元素时调用的回调
- onDragEnd: 当元素被放下或拖动取消时调用的回调
type ItemDragProps = {
type: string; // 插件类型
layout?: {
// 可选。用于预览和初始化的布局项目大小
w?: number;
h?: number;
};
extra?: any; // 自定义用户上下文
};
type ItemDropProps = {
commit: () => void; // 在完成所有配置操作后应调用的回调
dragProps: ItemDragProps; // 项目拖动属性
itemLayout: ConfigLayout; // 计算的项目布局尺寸
newLayout: ConfigLayout[]; // 元素放下后的新布局
};
示例:
const overlayMenuItems = [
{
id: 'chart',
icon: <Icon data={ChartColumn} />,
title: 'Chart',
qa: 'chart',
dragProps: {
// ItemDragProps
type: 'custom', // 已注册的插件类型
},
},
];
const onDrop = (dropProps: ItemDropProps) => {
// ... 将元素添加到您的配置中
dropProps.commit();
};
<DashKitDnDWrapper>
<DashKit editMode={true} config={config} onChange={onChange} onDrop={onDrop} />
<ActionPanel items={overlayMenuItems} />
</DashKitDnDWrapper>;
CSS API
名称 | 描述 |
---|---|
操作面板变量 | |
--dashkit-action-panel-color | 背景颜色 |
--dashkit-action-panel-border-color | 边框颜色 |
--dashkit-action-panel-border-radius | 边框圆角 |
操作面板项目变量 | |
--dashkit-action-panel-item-color | 背景颜色 |
--dashkit-action-panel-item-text-color | 文本颜色 |
--dashkit-action-panel-item-color-hover | 悬停背景颜色 |
--dashkit-action-panel-item-text-color-hover | 悬停文本颜色 |
覆盖层变量 | |
--dashkit-overlay-border-color | 边框颜色 |
--dashkit-overlay-color | 背景颜色 |
--dashkit-overlay-opacity | 不透明度 |
网格项目变量 | |
--dashkit-grid-item-edit-opacity | 不透明度 |
--dashkit-grid-item-border-radius | 边框圆角 |
占位符变量 | |
--dashkit-placeholder-color | 背景颜色 |
--dashkit-placeholder-opacity | 不透明度 |
使用示例
.custom-theme-wrapper {
--dashkit-grid-item-edit-opacit: 1;
--dashkit-overlay-color: var(--g-color-base-float);
--dashkit-overlay-border-color: var(--g-color-base-float);
--dashkit-overlay-opacity: 0.5;
--dashkit-action-panel-border-color: var(--g-color-line-info);
--dashkit-action-panel-color: var(--g-color-base-float-accent);
--dashkit-action-panel-border-radius: var(--g-border-radius-xxl);
}
// ....
const CustomThemeWrapper = (props: {
dashkitProps: DashkitProps;
actionPanelProps: ActionPanelProps;
}) => {
return (
<div className="custom-theme-wrapper">
<Dashkit {...props.dashkitProps} />
<ActionPanel {...props.actionPanelProps} />
</div>
);
};
开发
构建和监视
- 构建依赖项
npm ci
- 构建项目
npm run build
- 构建 storybook
npm run start
默认情况下,storybook 在 http://localhost:7120/
上运行。
当 storybook 运行时,不总是能捕获项目中的新更改,所以最好手动重建项目并重启 storybook。
开发机器上的 nginx 配置示例
server {
server_name dashkit.username.ru;
include common/ssl;
access_log /home/username/logs/common.access.log;
error_log /home/username/logs/common.error.log;
root /home/username/projects/dashkit;
location / {
try_files $uri @node;
}
location @node {
proxy_set_header Host $host;
proxy_set_header X-Real-IP $remote_addr;
proxy_set_header X-Forwarded-For $proxy_add_x_forwarded_for;
proxy_pass http://127.0.0.1:7120;
proxy_redirect off;
}
}