Sortable icon indicating copy to clipboard operation
Sortable copied to clipboard

[feature] Scroll speed according to distance from edge

Open koraykural opened this issue 5 years ago • 3 comments

Currently scroll speed is either 0 or the specified scrollSpeed if dragged item is near to edge.

Requested feature is that scroll speed will increase gradually according to the distance to the edge. If it's closer, it will be faster.

This can be done probably with scrollFn property but it would be much better if there were some built in functions like 'linear', 'ease-in', 'ease-out' etc.

koraykural avatar Sep 13 '20 07:09 koraykural

@koraykural Can you post your work-around implementation of scrollFn?

gap777 avatar Mar 23 '23 01:03 gap777

@koraykural Can you post your work-around implementation of scrollFn?

Sorry, I could not find it. It was an old project.

koraykural avatar Mar 23 '23 07:03 koraykural

@gap777 Well, I made something work-around implementation that scroll speed increases gradually. It also considers the mouse position outside of window. If u need it, just adjust some codes below

const onStart = (e) => {
  startScrollFn(e);
};
const onEnd = (e) => {
  initScrollVars();
};

type Direction = "up" | "down" | "right" | "left";
let pointerId: number;
let targetScrollEl: HTMLElement | null;
let lastPointerEvent: PointerEvent;
let interval;
let scrollSpeed = 3;
const speedIncreaseFactor = 0.001;
let lastMoveDirection: Direction;
const initScrollSpeed = () => (scrollSpeed = 3);
const onMouseMove = (event: PointerEvent) => {
  if (targetScrollEl) {
    lastPointerEvent = event;
  }
};

const startScrollFn = (e) => {
  const context = e.from as HTMLDivElement;
  pointerId = e.originalEvent.pointerId;
  targetScrollEl = getClosestScrollableEl(context);
  if (targetScrollEl) {
    targetScrollEl.setPointerCapture(pointerId);
    targetScrollEl.addEventListener("pointermove", onMouseMove);
    clearInterval(interval);
    interval = setInterval(() => {
      scrollFn(lastPointerEvent, targetScrollEl);
    });
  }
};

const initScrollVars = () => {
  if (targetScrollEl?.hasPointerCapture(pointerId)) {
    targetScrollEl?.releasePointerCapture(pointerId);
  }
  targetScrollEl?.removeEventListener("pointermove", onMouseMove);
  targetScrollEl = null;
  initScrollSpeed();
  clearInterval(interval);
};

const scrollFn = (e: PointerEvent, container: HTMLElement | null) => {
  if (!container || !e) return false;

  targetScrollEl = container;
  const { moveDirection, x, y } = getScrollResult(container, e);

  container.scrollBy(x, y);
  if (e !== lastPointerEvent) {
    scrollSpeed = round(scrollSpeed + speedIncreaseFactor, 3);
  }
  if (moveDirection !== lastMoveDirection) {
    initScrollSpeed();
  }
  if (moveDirection) lastMoveDirection = moveDirection;
  return false;
};

const getScrollResult = (container: HTMLElement, e: PointerEvent) => {
  let moveDirection: Direction | null = null;
  let x = 0;
  let y = 0;
  const scrollSensitivity = props.option?.scrollSensitivity || 50;
  const { top, bottom, left, right } = container.getBoundingClientRect();
  const topDistance = e.clientY - top;
  const bottomDistance = bottom - e.clientY;
  const leftDistance = e.clientX - left;
  const rightDistance = right - e.clientX;
  if (topDistance < scrollSensitivity) {
    y = (-(scrollSensitivity - topDistance) / scrollSensitivity) * scrollSpeed;
    moveDirection = "up";
  } else if (bottomDistance < scrollSensitivity) {
    y =
      ((scrollSensitivity - bottomDistance) / scrollSensitivity) * scrollSpeed;
    moveDirection = "down";
  }
  if (leftDistance < scrollSensitivity) {
    x = (-(scrollSensitivity - leftDistance) / scrollSensitivity) * scrollSpeed;
    moveDirection = "left";
  } else if (rightDistance < scrollSensitivity) {
    x = ((scrollSensitivity - rightDistance) / scrollSensitivity) * scrollSpeed;
    moveDirection = "right";
  }
  return { moveDirection, x, y };
};

FYI, It's actually for Vue components and used customized Sortable so some function's parameters might be different from original. You also have to call the method scrollFn in above code snippet when Sortable calls original scrollFn too.

<draggable
  :scroll-fn="(e, container)=>scrollFn(e, container)"
>
</draggable>

something like this

jongmin4943 avatar Apr 04 '23 06:04 jongmin4943