Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

fix: use fixed event listeners to address race conditions #1566

Open
wants to merge 2 commits into
base: experimental
Choose a base branch
from
Open
Show file tree
Hide file tree
Changes from 1 commit
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
5 changes: 5 additions & 0 deletions .changeset/use-fixed-event-listeners.md
Original file line number Diff line number Diff line change
@@ -0,0 +1,5 @@
---
'@dnd-kit/dom': patch
---

Use fixed event listeners in PointerSensor to address race conditions preventing `pointermove` and `pointerup` events from firing when document changes during a drag.
122 changes: 90 additions & 32 deletions packages/dom/src/core/sensors/pointer/PointerSensor.ts
Original file line number Diff line number Diff line change
Expand Up @@ -56,6 +56,8 @@ export class PointerSensor extends Sensor<

protected initialCoordinates: Coordinates | undefined;

protected source: Draggable | undefined = undefined;

#clearTimeout: CleanupFunction | undefined;

constructor(
Expand All @@ -67,6 +69,14 @@ export class PointerSensor extends Sensor<
this.handleCancel = this.handleCancel.bind(this);
this.handlePointerUp = this.handlePointerUp.bind(this);
this.handleKeyDown = this.handleKeyDown.bind(this);

effect(() => {
const unbindGlobal = this.bindGlobal(options ?? {});

return () => {
unbindGlobal();
};
});
}

public bind(source: Draggable, options = this.options) {
Expand All @@ -92,6 +102,48 @@ export class PointerSensor extends Sensor<
return unbind;
}

protected bindGlobal(options: PointerSensorOptions) {
const documents = new Set<Document>();

for (const draggable of this.manager.registry.draggables.value) {
if (draggable.element) {
documents.add(getDocument(draggable.element));
}
}

for (const droppable of this.manager.registry.droppables.value) {
if (droppable.element) {
documents.add(getDocument(droppable.element));
}
}

Comment on lines +108 to +119
Copy link
Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Noting that this may be better elsewhere for perf reasons

const unbindFns = Array.from(documents).map((doc) =>
this.listeners.bind(doc, [
{
type: 'pointermove',
listener: (event: PointerEvent) =>
this.handlePointerMove(event, doc, options),
},
{
type: 'pointerup',
listener: this.handlePointerUp,
options: {
capture: true,
},
},
{
// Cancel activation if there is a competing Drag and Drop interaction
type: 'dragstart',
listener: this.handleDragStart,
},
])
);

return () => {
unbindFns.forEach((unbind) => unbind());
};
}

protected handlePointerDown(
event: PointerEvent,
source: Draggable,
Expand All @@ -106,11 +158,6 @@ export class PointerSensor extends Sensor<
) {
return;
}
const {target} = event;
const isNativeDraggable =
isHTMLElement(target) &&
target.draggable &&
target.getAttribute('draggable') === 'true';

const offset = getFrameTransform(source.element);

Expand All @@ -119,6 +166,8 @@ export class PointerSensor extends Sensor<
y: event.clientY * offset.scaleY + offset.y,
};

this.source = source;

const {activationConstraints} = options;
const constraints =
typeof activationConstraints === 'function'
Expand All @@ -145,48 +194,38 @@ export class PointerSensor extends Sensor<
}
}

const ownerDocument = getDocument(event.target);

const unbindListeners = this.listeners.bind(ownerDocument, [
{
type: 'pointermove',
listener: (event: PointerEvent) =>
this.handlePointerMove(event, source, options),
},
{
type: 'pointerup',
listener: this.handlePointerUp,
options: {
capture: true,
},
},
{
// Cancel activation if there is a competing Drag and Drop interaction
type: 'dragstart',
listener: isNativeDraggable ? this.handleCancel : preventDefault,
},
]);

const cleanup = () => {
setTimeout(unbindListeners);
this.#clearTimeout?.();
this.initialCoordinates = undefined;
this.source = undefined;
};

this.cleanup.add(cleanup);
}

protected handlePointerMove(
event: PointerEvent,
source: Draggable,
doc: Document,
options: PointerSensorOptions
) {
if (!this.source) {
return;
}

const ownerDocument =
this.source.element && getDocument(this.source.element);

// Event may have duplicated between documents if user is bubbling events
if (doc !== ownerDocument) {
return;
}
Comment on lines +218 to +221
Copy link
Author

@chrisvxd chrisvxd Dec 12, 2024

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Note, new safe-guard to account for an edge-case where the user bubbles the pointermove event to the parent document (as I am doing with Puck).

I can address this on my end, but a safe-guard seemed sensible.


const coordinates = {
x: event.clientX,
y: event.clientY,
};

const offset = getFrameTransform(source.element);
const offset = getFrameTransform(this.source.element);

coordinates.x = coordinates.x * offset.scaleX + offset.x;
coordinates.y = coordinates.y * offset.scaleY + offset.y;
Expand All @@ -210,7 +249,7 @@ export class PointerSensor extends Sensor<
const {activationConstraints} = options;
const constraints =
typeof activationConstraints === 'function'
? activationConstraints(event, source)
? activationConstraints(event, this.source)
: activationConstraints;
const {distance, delay} = constraints ?? {};

Expand All @@ -222,7 +261,7 @@ export class PointerSensor extends Sensor<
return this.handleCancel();
}
if (exceedsDistance(delta, distance.value)) {
return this.handleStart(source, event);
return this.handleStart(this.source, event);
}
}

Expand Down Expand Up @@ -304,6 +343,25 @@ export class PointerSensor extends Sensor<
this.cleanup.add(unbind);
}

protected handleDragStart(event: DragEvent) {
const {target} = event;

if (!isElement(target)) {
return;
}

const isNativeDraggable =
isHTMLElement(target) &&
target.draggable &&
target.getAttribute('draggable') === 'true';

if (isNativeDraggable) {
this.handleCancel();
} else {
preventDefault(event);
}
}

protected handleCancel() {
const {dragOperation} = this.manager;

Expand Down