Skip to content

Commit

Permalink
fix: renderClone for dnd
Browse files Browse the repository at this point in the history
  • Loading branch information
GermanVor committed Mar 29, 2024
1 parent 3274c04 commit 7fbb889
Show file tree
Hide file tree
Showing 3 changed files with 50 additions and 29 deletions.
Original file line number Diff line number Diff line change
Expand Up @@ -2,7 +2,11 @@ import React from 'react';

import {Gear, Grip, Lock} from '@gravity-ui/icons';
import {DragDropContext, Draggable, Droppable} from 'react-beautiful-dnd';
import type {OnDragEndResponder} from 'react-beautiful-dnd';
import type {
DraggableProvided,
DraggableStateSnapshot,
OnDragEndResponder,
} from 'react-beautiful-dnd';

import {useUniqId} from '../../../../../hooks';
import type {PopperPlacement} from '../../../../../hooks/private';
Expand Down Expand Up @@ -65,6 +69,11 @@ const prepareValue = (tableColumnItems: TableColumnSetupItem[]) => {
return selectedIds;
};

interface RenderContextProps {
provided: DraggableProvided;
snapshot: DraggableStateSnapshot;
}

interface SwitcherProps {
onKeyDown: React.KeyboardEventHandler<HTMLElement>;
onClick: React.MouseEventHandler<HTMLElement>;
Expand Down Expand Up @@ -93,7 +102,17 @@ const useDndRenderContainer = ({onDragEnd, renderControls}: UseDndRenderContaine
<React.Fragment>
<ListContainerView ref={containerRef} id={id} className={className}>
<DragDropContext onDragEnd={onDragEnd}>
<Droppable droppableId={uniqId}>
<Droppable
droppableId={uniqId}
renderClone={(provided, snapshot, rubric) => {
const renderContextProps: RenderContextProps = {provided, snapshot};
return renderItem(
visibleFlattenIds[rubric.source.index],
rubric.source.index,
renderContextProps,
);
}}
>
{(droppableProvided) => {
return (
<div
Expand All @@ -117,7 +136,12 @@ const useDndRenderContainer = ({onDragEnd, renderControls}: UseDndRenderContaine
};

const useDndRenderItem = (sortable: boolean | undefined) => {
const renderDndItem: TreeSelectRenderItem<Item> = ({data, props, index}) => {
const renderDndItem: TreeSelectRenderItem<Item, RenderContextProps> = ({
data,
props,
index,
renderContext: renderContextProps,
}) => {
const isDragDisabled = sortable === false;

const endSlot =
Expand All @@ -129,35 +153,28 @@ const useDndRenderItem = (sortable: boolean | undefined) => {
endSlot,
};

const renderItem = (provided: DraggableProvided, snapshot: DraggableStateSnapshot) => (
<ListItemView
{...commonProps}
{...provided.draggableProps}
{...provided.dragHandleProps}
ref={provided.innerRef}
active={snapshot.isDragging}
/>
);

if (renderContextProps) {
return renderItem(renderContextProps.provided, renderContextProps.snapshot);
}

return (
<Draggable
draggableId={data.id}
index={index}
key={`item-key-${data.id}`}
isDragDisabled={isDragDisabled}
>
{(provided, snapshot) => {
const style: React.CSSProperties = {
...provided.draggableProps.style,
};

// not expected offset appears, one way to fix - remove this offsets explicitly
if (snapshot.isDragging) {
style.left = undefined;
style.top = undefined;
}

return (
<ListItemView
ref={provided.innerRef}
{...commonProps}
{...provided.draggableProps}
{...provided.dragHandleProps}
style={style}
dragging={snapshot.isDragging}
/>
);
}}
{renderItem}
</Draggable>
);
};
Expand Down Expand Up @@ -186,7 +203,7 @@ export interface TableColumnSetupProps {
sortable?: boolean;

onUpdate: (newSettings: TableSetting[]) => void;
popupWidth?: TreeSelectProps<any>['popupWidth'];
popupWidth?: TreeSelectProps<unknown>['popupWidth'];
popupPlacement?: PopperPlacement;

/**
Expand Down
4 changes: 3 additions & 1 deletion src/components/TreeSelect/TreeSelect.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -264,4 +264,6 @@ export const TreeSelect = React.forwardRef(function TreeSelect<T>(
</SelectPopup>
</Flex>
);
}) as <T>(props: TreeSelectProps<T> & {ref?: React.Ref<HTMLDivElement>}) => React.ReactElement;
}) as <T, P extends {} = {}>(
props: TreeSelectProps<T, P> & {ref?: React.Ref<HTMLDivElement>},
) => React.ReactElement;
6 changes: 4 additions & 2 deletions src/components/TreeSelect/types.ts
Original file line number Diff line number Diff line change
Expand Up @@ -43,7 +43,9 @@ export type TreeSelectRenderItem<T, P extends {} = {}> = (props: {
export type TreeSelectRenderContainerProps<T> = TreeListRenderContainerProps<T>;
export type TreeSelectRenderContainer<T> = TreeListRenderContainer<T>;

export interface TreeSelectProps<T> extends QAProps, Partial<Omit<ListState, 'selectedById'>> {
export interface TreeSelectProps<T, P extends {} = {}>
extends QAProps,
Partial<Omit<ListState, 'selectedById'>> {
value?: ListItemId[];
defaultOpen?: boolean;
defaultValue?: ListItemId[];
Expand Down Expand Up @@ -94,7 +96,7 @@ export interface TreeSelectProps<T> extends QAProps, Partial<Omit<ListState, 'se
/**
* Override list item content by you custom node.
*/
renderItem?: TreeSelectRenderItem<T>;
renderItem?: TreeSelectRenderItem<T, P>;
onClose?(): void;
onUpdate?(value: ListItemId[], selectedItems: T[]): void;
onOpenChange?(open: boolean): void;
Expand Down

0 comments on commit 7fbb889

Please sign in to comment.