Refactor dnd utility and add drag lifecycle events

Attach pointerup and pointermove listeners to scene.input
for reliable tracking. Emit dragstart, dragmove, and
dragend events alongside drag. Rename relativeX/Y to
deltaX/Y in DragDropEvent and integrate DisposableBag.
This commit is contained in:
hypercross 2026-04-18 23:33:13 +08:00
parent eefcef861a
commit d993d55576
1 changed files with 73 additions and 63 deletions

View File

@ -1,70 +1,80 @@
type PointerRecord = {
id: number;
x: number;
y: number;
}
import { DisposableBag } from "./disposable";
type PointerRecord = {
id: number;
x: number;
y: number;
};
export enum DragDropEventType {
DOWN,
UP,
MOVE,
DOWN,
UP,
MOVE,
}
export type DragDropEvent = {
type: DragDropEventType,
relativeX: number;
relativeY: number;
}
type: DragDropEventType;
deltaX: number;
deltaY: number;
};
export function dragDropEventEffect(
gameObject: Phaser.GameObjects.GameObject,
) {
let down: PointerRecord | null;
let up: PointerRecord | null;
gameObject: Phaser.GameObjects.GameObject,
disposables?: DisposableBag,
): () => void {
let isDragging = false;
let down: PointerRecord | null = null;
function onPointerDown(pointer: Phaser.Input.Pointer) {
down = {
id: pointer.id,
x: pointer.x,
y: pointer.y
}
up = null;
function onPointerDown(pointer: Phaser.Input.Pointer) {
if (isDragging) return;
isDragging = true;
down = { id: pointer.id, x: pointer.x, y: pointer.y };
const type = DragDropEventType.DOWN;
const relativeX = pointer.x - down.x;
const relativeY = pointer.y - down.y;
gameObject.emit('drag', {type, relativeX, relativeY});
}
const event: DragDropEvent = {
type: DragDropEventType.DOWN,
deltaX: 0,
deltaY: 0,
};
gameObject.emit("drag", event);
gameObject.emit("dragstart", event);
}
function onPointerUp(pointer: Phaser.Input.Pointer) {
if(!down) return;
up = {
id: pointer.id,
x: pointer.x,
y: pointer.y
}
function onPointerUp(pointer: Phaser.Input.Pointer) {
if (!isDragging || !down || pointer.id !== down.id) return;
const type = DragDropEventType.UP;
const relativeX = pointer.x - down.x;
const relativeY = pointer.y - down.y;
gameObject.emit('drag', {type, relativeX, relativeY});
}
isDragging = false;
const deltaX = pointer.x - down.x;
const deltaY = pointer.y - down.y;
const event: DragDropEvent = { type: DragDropEventType.UP, deltaX, deltaY };
gameObject.emit("drag", event);
gameObject.emit("dragend", event);
down = null;
}
function onPointerMove(pointer: Phaser.Input.Pointer) {
if(!down || up) return;
if(down.id !== pointer.id) return;
function onPointerMove(pointer: Phaser.Input.Pointer) {
if (!isDragging || !down || pointer.id !== down.id) return;
const type = DragDropEventType.MOVE;
const relativeX = pointer.x - down.x;
const relativeY = pointer.y - down.y;
gameObject.emit('drag', {type, relativeX, relativeY});
}
const deltaX = pointer.x - down.x;
const deltaY = pointer.y - down.y;
const event: DragDropEvent = {
type: DragDropEventType.MOVE,
deltaX,
deltaY,
};
gameObject.emit("drag", event);
gameObject.emit("dragmove", event);
}
gameObject.on('pointerdown', onPointerDown);
gameObject.on('pointerup', onPointerUp);
gameObject.scene.input.on('pointermove', onPointerMove);
return function () {
gameObject.off('pointerdown', onPointerDown);
gameObject.off('pointerup', onPointerUp);
gameObject.scene.input.off('pointermove', onPointerMove);
}
gameObject.on("pointerdown", onPointerDown);
gameObject.scene.input.on("pointerup", onPointerUp);
gameObject.scene.input.on("pointermove", onPointerMove);
const dispose = () => {
gameObject.off("pointerdown", onPointerDown);
gameObject.scene.input.off("pointerup", onPointerUp);
gameObject.scene.input.off("pointermove", onPointerMove);
};
disposables?.add(dispose);
return dispose;
}