Skip to content

Commit feedb5c

Browse files
authored
feat: #417 drag outside behavior (#420)
* fix: improve drag behavior outside of tree boundaries or legal drop positions (#417) * chore: use volta in core package for react workflow * chore: fix packagejson field * chore: change verify workflow to not use yarn on nested packages
1 parent 59ffd2b commit feedb5c

File tree

11 files changed

+85
-21
lines changed

11 files changed

+85
-21
lines changed

.github/workflows/verify.yml

+2-5
Original file line numberDiff line numberDiff line change
@@ -18,11 +18,8 @@ jobs:
1818
run: yarn
1919
- name: Custom React version
2020
run: |
21-
echo "enableImmutableInstalls: false" > ./.yarnrc.yml
22-
yarn add react@${{ matrix.react }} -D
23-
cd packages/core
24-
yarn add react@${{ matrix.react }} -D
25-
cd ../..
21+
echo -e "nodeLinker: node-modules\nenableImmutableInstalls: false" > ./.yarnrc.yml
22+
npm pkg set devDependencies.react=${{matrix.react}} --ws
2623
cat ./.yarnrc.yml
2724
yarn
2825
- name: Build

next-release-notes.md

+2-8
Original file line numberDiff line numberDiff line change
@@ -1,9 +1,3 @@
1-
<!--
2-
### Breaking Changes
3-
4-
### Features
5-
61
### Bug Fixes and Improvements
7-
8-
### Other Changes
9-
-->
2+
- Don't show drag line when dragging outside of the tree container (#417)
3+
- Fix a bug where items where dropped on the last valid position when dragging items on an invalid position and then dropping (#417)

package.json

+1-1
Original file line numberDiff line numberDiff line change
@@ -4,7 +4,7 @@
44
"repository": {
55
"type": "git",
66
"url": "[email protected]:lukasbach/react-complex-tree.git",
7-
"directory": "packages/docs"
7+
"directory": "."
88
},
99
"author": "Lukas Bach",
1010
"license": "MIT",

packages/core/package.json

+4
Original file line numberDiff line numberDiff line change
@@ -66,5 +66,9 @@
6666
"setupFiles": [
6767
"./test/helpers/setup.ts"
6868
]
69+
},
70+
"volta": {
71+
"node": "18.12.1",
72+
"yarn": "3.3.0"
6973
}
7074
}

packages/core/src/controlledEnvironment/layoutUtils.ts

+4-4
Original file line numberDiff line numberDiff line change
@@ -16,7 +16,7 @@ export const computeItemHeight = (treeId: string) => {
1616
};
1717

1818
export const isOutsideOfContainer = (e: DragEvent, treeBb: DOMRect) =>
19-
e.clientX < treeBb.left ||
20-
e.clientX > treeBb.right ||
21-
e.clientY < treeBb.top ||
22-
e.clientY > treeBb.bottom;
19+
e.clientX <= treeBb.left ||
20+
e.clientX >= treeBb.right ||
21+
e.clientY <= treeBb.top ||
22+
e.clientY >= treeBb.bottom;

packages/core/src/drag/DragAndDropProvider.tsx

+17
Original file line numberDiff line numberDiff line change
@@ -15,6 +15,7 @@ import { useCallSoon } from '../useCallSoon';
1515
import { useStableHandler } from '../useStableHandler';
1616
import { useGetOriginalItemOrder } from '../useGetOriginalItemOrder';
1717
import { useDraggingPosition } from './useDraggingPosition';
18+
import { isOutsideOfContainer } from '../controlledEnvironment/layoutUtils';
1819

1920
const DragAndDropContext = React.createContext<DragAndDropContextProps>(
2021
null as any
@@ -177,6 +178,20 @@ export const DragAndDropProvider: React.FC<React.PropsWithChildren> = ({
177178
}
178179
);
179180

181+
const onDragLeaveContainerHandler = useStableHandler(
182+
(
183+
e: DragEvent,
184+
containerRef: React.MutableRefObject<HTMLElement | undefined>
185+
) => {
186+
if (!containerRef.current) return;
187+
if (
188+
isOutsideOfContainer(e, containerRef.current.getBoundingClientRect())
189+
) {
190+
setDraggingPosition(undefined);
191+
}
192+
}
193+
);
194+
180195
const onDropHandler = useStableHandler(() => {
181196
if (!draggingItems || !draggingPosition || !environment.onDrop) {
182197
return;
@@ -287,6 +302,7 @@ export const DragAndDropProvider: React.FC<React.PropsWithChildren> = ({
287302
itemHeight: itemHeight.current,
288303
isProgrammaticallyDragging,
289304
onDragOverTreeHandler,
305+
onDragLeaveContainerHandler,
290306
viableDragPositions,
291307
}),
292308
[
@@ -297,6 +313,7 @@ export const DragAndDropProvider: React.FC<React.PropsWithChildren> = ({
297313
isProgrammaticallyDragging,
298314
itemHeight,
299315
onDragOverTreeHandler,
316+
onDragLeaveContainerHandler,
300317
onStartDraggingItems,
301318
programmaticDragDown,
302319
programmaticDragUp,

packages/core/src/drag/DraggingPositionEvaluation.ts

+3-3
Original file line numberDiff line numberDiff line change
@@ -227,7 +227,7 @@ export class DraggingPositionEvaluation {
227227
return true;
228228
}
229229

230-
getDraggingPosition(): DraggingPosition | undefined {
230+
getDraggingPosition(): DraggingPosition | 'invalid' | undefined {
231231
if (this.env.linearItems[this.treeId].length === 0) {
232232
return this.getEmptyTreeDragPosition();
233233
}
@@ -251,11 +251,11 @@ export class DraggingPositionEvaluation {
251251
}
252252

253253
if (this.areDraggingItemsDescendantOfTarget()) {
254-
return undefined;
254+
return 'invalid';
255255
}
256256

257257
if (!this.canDropAtCurrentTarget()) {
258-
return undefined;
258+
return 'invalid';
259259
}
260260

261261
const { parent } = this.getParentOfLinearItem(

packages/core/src/tree/TreeManager.tsx

+3
Original file line numberDiff line numberDiff line change
@@ -48,6 +48,9 @@ export const TreeManager = (): JSX.Element => {
4848
e.preventDefault(); // Allow drop. Also implicitly set by items, but needed here as well for dropping on empty space
4949
dnd.onDragOverTreeHandler(e as any, treeId, containerRef);
5050
},
51+
onDragLeave: e => {
52+
dnd.onDragLeaveContainerHandler(e as any, containerRef);
53+
},
5154
onMouseDown: () => dnd.abortProgrammaticDrag(),
5255
ref: containerRef,
5356
style: { position: 'relative' },

packages/core/src/types.ts

+4
Original file line numberDiff line numberDiff line change
@@ -328,6 +328,10 @@ export interface DragAndDropContextProps<T = any> {
328328
treeId: string,
329329
containerRef: React.MutableRefObject<HTMLElement | undefined>
330330
) => void;
331+
onDragLeaveContainerHandler: (
332+
e: DragEvent,
333+
containerRef: React.MutableRefObject<HTMLElement | undefined>
334+
) => void;
331335
}
332336

333337
export type DraggingPosition =

packages/core/test/dnd-basics.spec.tsx

+31
Original file line numberDiff line numberDiff line change
@@ -654,4 +654,35 @@ describe('dnd basics', () => {
654654
});
655655
});
656656
});
657+
658+
it('doesnt drop on last valid location when moving drop to invalid location', async () => {
659+
const test = await new TestUtil().renderOpenTree({});
660+
await test.startDrag('aab');
661+
await test.dragOver('aa');
662+
await test.dragOver('aab');
663+
await test.drop();
664+
await test.expectItemContentsUnchanged('aa');
665+
});
666+
667+
it('sets drop position correctly', async () => {
668+
const test = await new TestUtil().renderOpenTree({});
669+
await test.startDrag('target');
670+
await test.dragOver('aa');
671+
expect(test.treeRef?.dragAndDropContext?.draggingPosition).toStrictEqual({
672+
depth: 1,
673+
linearIndex: 5,
674+
parentItem: 'a',
675+
targetItem: 'aa',
676+
targetType: 'item',
677+
treeId: 'tree-1',
678+
});
679+
});
680+
681+
it('unsets drop position when dragging out', async () => {
682+
const test = await new TestUtil().renderOpenTree({});
683+
await test.startDrag('target');
684+
await test.dragOver('aa');
685+
await test.dragLeave();
686+
expect(test.treeRef?.dragAndDropContext?.draggingPosition).toBe(undefined);
687+
});
657688
});

packages/core/test/helpers/TestUtil.tsx

+14
Original file line numberDiff line numberDiff line change
@@ -171,6 +171,20 @@ export class TestUtil {
171171
});
172172
}
173173

174+
public async dragLeave() {
175+
(isOutsideOfContainer as jest.Mock).mockReturnValue(true);
176+
await act(async () => {
177+
this.environmentRef?.dragAndDropContext.onDragLeaveContainerHandler(
178+
{
179+
clientX: 9999,
180+
clientY: 9999,
181+
} as any,
182+
{ current: this.containerRef ?? undefined }
183+
);
184+
});
185+
(isOutsideOfContainer as jest.Mock).mockReturnValue(false);
186+
}
187+
174188
public async drop() {
175189
await act(async () => {
176190
fireEvent.drop(window);

0 commit comments

Comments
 (0)