Source code

Revision control

Copy as Markdown

Other Tools

/* This Source Code Form is subject to the terms of the Mozilla Public
* License, v. 2.0. If a copy of the MPL was not distributed with this file,
* You can obtain one at http://mozilla.org/MPL/2.0/. */
"use strict";
const {
createRef,
Component,
} = require("resource://devtools/client/shared/vendor/react.js");
const PropTypes = require("resource://devtools/client/shared/vendor/react-prop-types.js");
const dom = require("resource://devtools/client/shared/vendor/react-dom-factories.js");
class Draggable extends Component {
static get propTypes() {
return {
onMove: PropTypes.func.isRequired,
onDoubleClick: PropTypes.func,
onStart: PropTypes.func,
onStop: PropTypes.func,
style: PropTypes.object,
title: PropTypes.string,
className: PropTypes.string,
};
}
constructor(props) {
super(props);
this.draggableEl = createRef();
this.startDragging = this.startDragging.bind(this);
this.stopDragging = this.stopDragging.bind(this);
this.onDoubleClick = this.onDoubleClick.bind(this);
this.onMove = this.onMove.bind(this);
this.mouseX = 0;
this.mouseY = 0;
}
startDragging(ev) {
// We want to handle a drag during a mouse button is pressed. So, we can
// ignore pointer events which are caused by other devices.
if (ev.pointerType != "mouse") {
return;
}
const xDiff = Math.abs(this.mouseX - ev.clientX);
const yDiff = Math.abs(this.mouseY - ev.clientY);
// This allows for double-click.
if (this.props.onDoubleClick && xDiff + yDiff <= 1) {
return;
}
this.mouseX = ev.clientX;
this.mouseY = ev.clientY;
if (this.isDragging) {
return;
}
this.isDragging = true;
// "pointermove" is fired when the button state is changed too. Therefore,
// we should listen to "mousemove" to handle the pointer position changes.
this.draggableEl.current.addEventListener("mousemove", this.onMove);
this.draggableEl.current.setPointerCapture(ev.pointerId);
this.draggableEl.current.addEventListener(
"mousedown",
event => event.preventDefault(),
{ once: true }
);
this.props.onStart && this.props.onStart();
}
onDoubleClick() {
if (this.props.onDoubleClick) {
this.props.onDoubleClick();
}
}
onMove(ev) {
if (!this.isDragging) {
return;
}
ev.preventDefault();
// Use viewport coordinates so, moving mouse over iframes
// doesn't mangle (relative) coordinates.
this.props.onMove(ev.clientX, ev.clientY);
}
stopDragging() {
if (!this.isDragging) {
return;
}
this.isDragging = false;
this.draggableEl.current.removeEventListener("mousemove", this.onMove);
this.draggableEl.current.addEventListener(
"mouseup",
event => event.preventDefault(),
{ once: true }
);
this.props.onStop && this.props.onStop();
}
render() {
return dom.div({
ref: this.draggableEl,
role: "presentation",
style: this.props.style,
title: this.props.title,
className: this.props.className,
onPointerDown: this.startDragging,
onPointerUp: this.stopDragging,
onDoubleClick: this.onDoubleClick,
});
}
}
module.exports = Draggable;