mirror of
https://github.com/apache/superset.git
synced 2026-04-26 03:24:53 +00:00
387 lines
12 KiB
TypeScript
387 lines
12 KiB
TypeScript
/**
|
|
* Licensed to the Apache Software Foundation (ASF) under one
|
|
* or more contributor license agreements. See the NOTICE file
|
|
* distributed with this work for additional information
|
|
* regarding copyright ownership. The ASF licenses this file
|
|
* to you under the Apache License, Version 2.0 (the
|
|
* "License"); you may not use this file except in compliance
|
|
* with the License. You may obtain a copy of the License at
|
|
*
|
|
* http://www.apache.org/licenses/LICENSE-2.0
|
|
*
|
|
* Unless required by applicable law or agreed to in writing,
|
|
* software distributed under the License is distributed on an
|
|
* "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY
|
|
* KIND, either express or implied. See the License for the
|
|
* specific language governing permissions and limitations
|
|
* under the License.
|
|
*/
|
|
import { Fragment, useCallback, useState, useMemo, memo } from 'react';
|
|
import cx from 'classnames';
|
|
import { t, css, styled, SupersetTheme } from '@apache-superset/core/ui';
|
|
import { Icons } from '@superset-ui/core/components/Icons';
|
|
import type { LayoutItem } from 'src/dashboard/types';
|
|
import type { DropResult } from 'src/dashboard/components/dnd/dragDroppableConfig';
|
|
import DashboardComponent from 'src/dashboard/containers/DashboardComponent';
|
|
import DeleteComponentButton from 'src/dashboard/components/DeleteComponentButton';
|
|
import {
|
|
Draggable,
|
|
Droppable,
|
|
} from 'src/dashboard/components/dnd/DragDroppable';
|
|
import DragHandle from 'src/dashboard/components/dnd/DragHandle';
|
|
import HoverMenu from 'src/dashboard/components/menu/HoverMenu';
|
|
import IconButton from 'src/dashboard/components/IconButton';
|
|
import ResizableContainer from 'src/dashboard/components/resizable/ResizableContainer';
|
|
import BackgroundStyleDropdown from 'src/dashboard/components/menu/BackgroundStyleDropdown';
|
|
import WithPopoverMenu from 'src/dashboard/components/menu/WithPopoverMenu';
|
|
import backgroundStyleOptions from 'src/dashboard/util/backgroundStyleOptions';
|
|
import { BACKGROUND_TRANSPARENT } from 'src/dashboard/util/constants';
|
|
import { EMPTY_CONTAINER_Z_INDEX } from 'src/dashboard/constants';
|
|
|
|
interface ColumnProps {
|
|
id: string;
|
|
parentId: string;
|
|
component: LayoutItem;
|
|
parentComponent: LayoutItem;
|
|
index: number;
|
|
depth: number;
|
|
editMode: boolean;
|
|
|
|
// grid related
|
|
availableColumnCount: number;
|
|
columnWidth: number;
|
|
minColumnWidth: number;
|
|
onResizeStart: import('re-resizable').ResizeStartCallback;
|
|
onResize: import('re-resizable').ResizeCallback;
|
|
onResizeStop: import('re-resizable').ResizeCallback;
|
|
|
|
// dnd
|
|
deleteComponent: (id: string, parentId: string) => void;
|
|
handleComponentDrop: (dropResult: DropResult) => void;
|
|
updateComponents: (components: Record<string, LayoutItem>) => void;
|
|
|
|
// optional
|
|
onChangeTab?: (params: { pathToTabIndex: string[] }) => void;
|
|
isComponentVisible?: boolean;
|
|
}
|
|
|
|
interface DragChildProps {
|
|
dragSourceRef: React.RefCallback<HTMLElement>;
|
|
}
|
|
|
|
const ColumnStyles = styled.div<{ editMode: boolean }>`
|
|
${({ theme, editMode }) => css`
|
|
&.grid-column {
|
|
width: 100%;
|
|
position: relative;
|
|
|
|
& > :not(.hover-menu):not(:last-child) {
|
|
${!editMode && `margin-bottom: ${theme.sizeUnit * 4}px;`}
|
|
}
|
|
}
|
|
|
|
.dashboard--editing &:after {
|
|
content: '';
|
|
position: absolute;
|
|
width: 100%;
|
|
height: 100%;
|
|
top: 0;
|
|
left: 0;
|
|
z-index: 1;
|
|
pointer-events: none;
|
|
border: 1px dashed ${theme.colorBorder};
|
|
}
|
|
.dashboard--editing .resizable-container--resizing:hover > &:after,
|
|
.dashboard--editing .hover-menu:hover + &:after {
|
|
border: 1px dashed ${theme.colorPrimary};
|
|
z-index: 2;
|
|
}
|
|
|
|
& .empty-droptarget {
|
|
&.droptarget-edge {
|
|
position: absolute;
|
|
z-index: ${EMPTY_CONTAINER_Z_INDEX};
|
|
width: 100%;
|
|
height: ${theme.sizeUnit * 4}px;
|
|
&:first-child {
|
|
inset-block-start: 0;
|
|
}
|
|
}
|
|
&:first-child:not(.droptarget-edge) {
|
|
position: absolute;
|
|
top: 0;
|
|
left: 0;
|
|
z-index: ${EMPTY_CONTAINER_Z_INDEX};
|
|
width: 100%;
|
|
height: 100%;
|
|
}
|
|
&:not(:first-child):not(.droptarget-edge) {
|
|
width: 100%;
|
|
min-height: ${theme.sizeUnit * 4}px;
|
|
}
|
|
}
|
|
`}
|
|
`;
|
|
|
|
const emptyColumnContentStyles = (theme: SupersetTheme) => css`
|
|
min-height: ${theme.sizeUnit * 25}px;
|
|
width: 100%;
|
|
height: 100%;
|
|
display: flex;
|
|
align-items: center;
|
|
justify-content: center;
|
|
color: ${theme.colorTextLabel};
|
|
`;
|
|
|
|
interface DropIndicatorChildProps {
|
|
dropIndicatorProps?: Record<string, unknown>;
|
|
}
|
|
|
|
const Column = (props: ColumnProps) => {
|
|
const {
|
|
component: columnComponent,
|
|
parentComponent,
|
|
index,
|
|
availableColumnCount,
|
|
columnWidth,
|
|
minColumnWidth,
|
|
depth,
|
|
onResizeStart,
|
|
onResize,
|
|
onResizeStop,
|
|
handleComponentDrop,
|
|
editMode,
|
|
onChangeTab,
|
|
isComponentVisible,
|
|
deleteComponent,
|
|
id,
|
|
parentId,
|
|
updateComponents,
|
|
} = props;
|
|
|
|
const [isFocused, setIsFocused] = useState(false);
|
|
|
|
const handleDeleteComponent = useCallback(() => {
|
|
deleteComponent(id, parentId);
|
|
}, [deleteComponent, id, parentId]);
|
|
|
|
const handleChangeFocus = useCallback((nextFocus: boolean | number) => {
|
|
setIsFocused(Boolean(nextFocus));
|
|
}, []);
|
|
|
|
const handleChangeBackground = useCallback(
|
|
(nextValue: string) => {
|
|
const metaKey = 'background';
|
|
if (nextValue && columnComponent.meta[metaKey] !== nextValue) {
|
|
updateComponents({
|
|
[columnComponent.id]: {
|
|
...columnComponent,
|
|
meta: {
|
|
...columnComponent.meta,
|
|
[metaKey]: nextValue,
|
|
},
|
|
},
|
|
});
|
|
}
|
|
},
|
|
[columnComponent, updateComponents],
|
|
);
|
|
|
|
const columnItems = useMemo(
|
|
() => columnComponent.children || [],
|
|
[columnComponent.children],
|
|
);
|
|
|
|
const backgroundStyle = backgroundStyleOptions.find(
|
|
opt =>
|
|
opt.value === (columnComponent.meta.background || BACKGROUND_TRANSPARENT),
|
|
);
|
|
|
|
const renderChild = useCallback(
|
|
({ dragSourceRef }: DragChildProps) => (
|
|
<ResizableContainer
|
|
id={columnComponent.id}
|
|
adjustableWidth
|
|
adjustableHeight={false}
|
|
widthStep={columnWidth}
|
|
widthMultiple={columnComponent.meta.width ?? 0}
|
|
heightMultiple={0}
|
|
minWidthMultiple={minColumnWidth}
|
|
maxWidthMultiple={
|
|
availableColumnCount + (columnComponent.meta.width || 0)
|
|
}
|
|
onResizeStart={onResizeStart}
|
|
onResize={onResize}
|
|
onResizeStop={onResizeStop}
|
|
editMode={editMode}
|
|
>
|
|
<WithPopoverMenu
|
|
isFocused={isFocused}
|
|
onChangeFocus={handleChangeFocus}
|
|
disableClick
|
|
menuItems={[
|
|
<BackgroundStyleDropdown
|
|
key={`${columnComponent.id}-background`}
|
|
id={`${columnComponent.id}-background`}
|
|
value={
|
|
(columnComponent.meta.background as string) ||
|
|
BACKGROUND_TRANSPARENT
|
|
}
|
|
onChange={handleChangeBackground}
|
|
/>,
|
|
]}
|
|
editMode={editMode}
|
|
>
|
|
{editMode && (
|
|
<HoverMenu
|
|
innerRef={
|
|
dragSourceRef as unknown as React.RefObject<HTMLDivElement>
|
|
}
|
|
position="top"
|
|
>
|
|
<DragHandle position="top" />
|
|
<DeleteComponentButton
|
|
iconSize="m"
|
|
onDelete={handleDeleteComponent}
|
|
/>
|
|
<IconButton
|
|
onClick={() => handleChangeFocus(true)}
|
|
icon={<Icons.SettingOutlined iconSize="m" />}
|
|
/>
|
|
</HoverMenu>
|
|
)}
|
|
<ColumnStyles
|
|
className={cx('grid-column', backgroundStyle?.className)}
|
|
editMode={editMode}
|
|
>
|
|
{editMode && (
|
|
<Droppable
|
|
component={columnComponent}
|
|
parentComponent={columnComponent}
|
|
{...(columnItems.length === 0
|
|
? {
|
|
dropToChild: true,
|
|
}
|
|
: {
|
|
component: columnItems[0],
|
|
})}
|
|
depth={depth}
|
|
index={0}
|
|
orientation="column"
|
|
onDrop={handleComponentDrop}
|
|
className={cx(
|
|
'empty-droptarget',
|
|
columnItems.length > 0 && 'droptarget-edge',
|
|
)}
|
|
editMode
|
|
>
|
|
{({ dropIndicatorProps }: DropIndicatorChildProps) =>
|
|
dropIndicatorProps && <div {...dropIndicatorProps} />
|
|
}
|
|
</Droppable>
|
|
)}
|
|
{columnItems.length === 0 ? (
|
|
<div css={emptyColumnContentStyles}>{t('Empty column')}</div>
|
|
) : (
|
|
columnItems.map((componentId: string, itemIndex: number) => (
|
|
<Fragment key={componentId}>
|
|
<DashboardComponent
|
|
id={componentId}
|
|
parentId={columnComponent.id}
|
|
depth={depth + 1}
|
|
index={itemIndex}
|
|
availableColumnCount={columnComponent.meta.width ?? 0}
|
|
columnWidth={columnWidth}
|
|
onResizeStart={
|
|
onResizeStart as unknown as (
|
|
event: MouseEvent | TouchEvent,
|
|
direction: string,
|
|
elementRef: HTMLElement,
|
|
) => void
|
|
}
|
|
onResize={
|
|
onResize as unknown as (
|
|
event: MouseEvent | TouchEvent,
|
|
direction: string,
|
|
elementRef: HTMLElement,
|
|
delta: { width: number; height: number },
|
|
) => void
|
|
}
|
|
onResizeStop={
|
|
onResizeStop as unknown as (
|
|
event: MouseEvent | TouchEvent,
|
|
direction: string,
|
|
elementRef: HTMLElement,
|
|
delta: { width: number; height: number },
|
|
id: string,
|
|
) => void
|
|
}
|
|
isComponentVisible={isComponentVisible}
|
|
onChangeTab={onChangeTab}
|
|
/>
|
|
{editMode && (
|
|
<Droppable
|
|
component={columnItems}
|
|
parentComponent={columnComponent}
|
|
depth={depth}
|
|
index={itemIndex + 1}
|
|
orientation="column"
|
|
onDrop={handleComponentDrop}
|
|
className={cx(
|
|
'empty-droptarget',
|
|
itemIndex === columnItems.length - 1 &&
|
|
'droptarget-edge',
|
|
)}
|
|
editMode
|
|
>
|
|
{({ dropIndicatorProps }: DropIndicatorChildProps) =>
|
|
dropIndicatorProps && <div {...dropIndicatorProps} />
|
|
}
|
|
</Droppable>
|
|
)}
|
|
</Fragment>
|
|
))
|
|
)}
|
|
</ColumnStyles>
|
|
</WithPopoverMenu>
|
|
</ResizableContainer>
|
|
),
|
|
[
|
|
availableColumnCount,
|
|
backgroundStyle?.className,
|
|
columnComponent,
|
|
columnItems,
|
|
columnWidth,
|
|
depth,
|
|
editMode,
|
|
handleChangeBackground,
|
|
handleChangeFocus,
|
|
handleComponentDrop,
|
|
handleDeleteComponent,
|
|
isComponentVisible,
|
|
isFocused,
|
|
minColumnWidth,
|
|
onChangeTab,
|
|
onResize,
|
|
onResizeStart,
|
|
onResizeStop,
|
|
],
|
|
);
|
|
|
|
return (
|
|
<Draggable
|
|
component={columnComponent}
|
|
parentComponent={parentComponent}
|
|
orientation="column"
|
|
index={index}
|
|
depth={depth}
|
|
onDrop={handleComponentDrop}
|
|
editMode={editMode}
|
|
>
|
|
{renderChild}
|
|
</Draggable>
|
|
);
|
|
};
|
|
|
|
export default memo(Column);
|