A simple component for making elements draggable.
<Draggable>
<div>I can now be moved around!</div>
</Draggable>npm install react-draggable
# or
yarn add react-draggable// ES Modules
import Draggable from 'react-draggable';
import { DraggableCore } from 'react-draggable';
// CommonJS
const Draggable = require('react-draggable');
const { DraggableCore } = require('react-draggable');TypeScript types are included.
| Version | React Version |
|---|---|
| 4.x | 16.3+ |
| 3.x | 15 - 16 |
| 2.x | 0.14 - 15 |
import React, { useRef } from 'react';
import Draggable from 'react-draggable';
function App() {
const nodeRef = useRef(null);
return (
<Draggable nodeRef={nodeRef}>
<div ref={nodeRef}>Drag me!</div>
</Draggable>
);
}View the Demo and its source for more examples.
A <Draggable> element wraps an existing element and extends it with new event handlers and styles. It does not create a wrapper element in the DOM.
Draggable items are moved using CSS Transforms. This allows items to be dragged regardless of their current positioning (relative, absolute, or static). Elements can also be moved between drags without incident.
If the item you are dragging already has a CSS Transform applied, it will be overwritten by <Draggable>. Use an intermediate wrapper (<Draggable><span>...</span></Draggable>) in this case.
type DraggableEventHandler = (e: Event, data: DraggableData) => void | false;
type DraggableData = {
node: HTMLElement,
x: number, y: number,
deltaX: number, deltaY: number,
lastX: number, lastY: number,
};| Prop | Type | Default | Description |
|---|---|---|---|
allowAnyClick |
boolean |
false |
Allow dragging on non-left-button clicks |
allowMobileScroll |
boolean |
false |
Don't prevent touchstart, allowing scrolling inside containers |
axis |
'both' | 'x' | 'y' | 'none' |
'both' |
Axis to allow dragging on |
bounds |
object | string |
- | Restrict movement. Use 'parent', a CSS selector, or {left, top, right, bottom} |
cancel |
string |
- | CSS selector for elements that should not initiate drag |
defaultClassName |
string |
'react-draggable' |
Class name applied to the element |
defaultClassNameDragging |
string |
'react-draggable-dragging' |
Class name applied while dragging |
defaultClassNameDragged |
string |
'react-draggable-dragged' |
Class name applied after drag |
defaultPosition |
{x: number, y: number} |
{x: 0, y: 0} |
Starting position |
disabled |
boolean |
false |
Disable dragging |
enableUserSelectHack |
boolean |
true |
Add user-select: none while dragging |
grid |
[number, number] |
- | Snap to grid [x, y] |
handle |
string |
- | CSS selector for the drag handle |
nodeRef |
React.RefObject |
- | Ref to the DOM element. Required for React Strict Mode |
offsetParent |
HTMLElement |
- | Custom offsetParent for drag calculations |
onDrag |
DraggableEventHandler |
- | Called while dragging |
onMouseDown |
(e: MouseEvent) => void |
- | Called on mouse down |
onStart |
DraggableEventHandler |
- | Called when dragging starts. Return false to cancel |
onStop |
DraggableEventHandler |
- | Called when dragging stops |
position |
{x: number, y: number} |
- | Controlled position |
positionOffset |
{x: number | string, y: number | string} |
- | Position offset (supports percentages) |
scale |
number |
1 |
Scale factor for dragging inside transformed parents |
Note: Setting className, style, or transform on <Draggable> will error. Set them on the child element.
For users that require full control, <DraggableCore> provides drag callbacks without managing state or styles. It does not set any transforms; you must handle positioning yourself.
See React-Resizable and React-Grid-Layout for usage examples.
<DraggableCore> accepts a subset of <Draggable> props:
allowAnyClickallowMobileScrollcanceldisabledenableUserSelectHackgridhandlenodeRefoffsetParentonDragonMouseDownonStartonStopscale
To avoid ReactDOM.findDOMNode() deprecation warnings in React Strict Mode, pass a nodeRef prop:
function App() {
const nodeRef = useRef(null);
return (
<Draggable nodeRef={nodeRef}>
<div ref={nodeRef}>Drag me!</div>
</Draggable>
);
}For custom components, forward both the ref and props:
const MyComponent = forwardRef((props, ref) => (
<div {...props} ref={ref}>Draggable content</div>
));
function App() {
const nodeRef = useRef(null);
return (
<Draggable nodeRef={nodeRef}>
<MyComponent ref={nodeRef} />
</Draggable>
);
}<Draggable> is a 'batteries-included' component that manages its own state. For complete control, use <DraggableCore>.
For programmatic repositioning while using <Draggable>'s state management, pass the position prop:
function ControlledDraggable() {
const nodeRef = useRef(null);
const [position, setPosition] = useState({ x: 0, y: 0 });
const handleDrag = (e, data) => {
setPosition({ x: data.x, y: data.y });
};
const resetPosition = () => setPosition({ x: 0, y: 0 });
return (
<>
<button onClick={resetPosition}>Reset</button>
<Draggable nodeRef={nodeRef} position={position} onDrag={handleDrag}>
<div ref={nodeRef}>Drag me or reset!</div>
</Draggable>
</>
);
}- Fork the project
- Run
yarn devto start the development server - Make changes and add tests
- Run
yarn testto ensure tests pass - Submit a PR
- Update CHANGELOG.md
- Run
make release-patch,make release-minor, ormake release-major - Run
make publish
MIT
