mirror of
https://github.com/overleaf/overleaf.git
synced 2024-11-21 20:47:08 -05:00
417 lines
15 KiB
JavaScript
417 lines
15 KiB
JavaScript
|
/* ***** BEGIN LICENSE BLOCK *****
|
||
|
* Distributed under the BSD license:
|
||
|
*
|
||
|
* Copyright (c) 2010, Ajax.org B.V.
|
||
|
* All rights reserved.
|
||
|
*
|
||
|
* Redistribution and use in source and binary forms, with or without
|
||
|
* modification, are permitted provided that the following conditions are met:
|
||
|
* * Redistributions of source code must retain the above copyright
|
||
|
* notice, this list of conditions and the following disclaimer.
|
||
|
* * Redistributions in binary form must reproduce the above copyright
|
||
|
* notice, this list of conditions and the following disclaimer in the
|
||
|
* documentation and/or other materials provided with the distribution.
|
||
|
* * Neither the name of Ajax.org B.V. nor the
|
||
|
* names of its contributors may be used to endorse or promote products
|
||
|
* derived from this software without specific prior written permission.
|
||
|
*
|
||
|
* THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" AND
|
||
|
* ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
|
||
|
* WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
|
||
|
* DISCLAIMED. IN NO EVENT SHALL AJAX.ORG B.V. BE LIABLE FOR ANY
|
||
|
* DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES
|
||
|
* (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;
|
||
|
* LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND
|
||
|
* ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
|
||
|
* (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS
|
||
|
* SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
||
|
*
|
||
|
* ***** END LICENSE BLOCK ***** */
|
||
|
|
||
|
define(function(require, exports, module) {
|
||
|
"use strict";
|
||
|
|
||
|
var dom = require("../lib/dom");
|
||
|
var event = require("../lib/event");
|
||
|
var useragent = require("../lib/useragent");
|
||
|
|
||
|
var AUTOSCROLL_DELAY = 200;
|
||
|
var SCROLL_CURSOR_DELAY = 200;
|
||
|
var SCROLL_CURSOR_HYSTERESIS = 5;
|
||
|
|
||
|
function DragdropHandler(mouseHandler) {
|
||
|
|
||
|
var editor = mouseHandler.editor;
|
||
|
|
||
|
var blankImage = dom.createElement("img");
|
||
|
// Safari crashes without image data
|
||
|
blankImage.src = "";
|
||
|
if (useragent.isOpera)
|
||
|
blankImage.style.cssText = "width:1px;height:1px;position:fixed;top:0;left:0;z-index:2147483647;opacity:0;";
|
||
|
|
||
|
var exports = ["dragWait", "dragWaitEnd", "startDrag", "dragReadyEnd", "onMouseDrag"];
|
||
|
|
||
|
exports.forEach(function(x) {
|
||
|
mouseHandler[x] = this[x];
|
||
|
}, this);
|
||
|
editor.addEventListener("mousedown", this.onMouseDown.bind(mouseHandler));
|
||
|
|
||
|
|
||
|
var mouseTarget = editor.container;
|
||
|
var dragSelectionMarker, x, y;
|
||
|
var timerId, range;
|
||
|
var dragCursor, counter = 0;
|
||
|
var dragOperation;
|
||
|
var isInternal;
|
||
|
var autoScrollStartTime;
|
||
|
var cursorMovedTime;
|
||
|
var cursorPointOnCaretMoved;
|
||
|
|
||
|
this.onDragStart = function(e) {
|
||
|
// webkit workaround, see this.onMouseDown
|
||
|
if (this.cancelDrag || !mouseTarget.draggable) {
|
||
|
var self = this;
|
||
|
setTimeout(function(){
|
||
|
self.startSelect();
|
||
|
self.captureMouse(e);
|
||
|
}, 0);
|
||
|
return e.preventDefault();
|
||
|
}
|
||
|
range = editor.getSelectionRange();
|
||
|
|
||
|
var dataTransfer = e.dataTransfer;
|
||
|
dataTransfer.effectAllowed = editor.getReadOnly() ? "copy" : "copyMove";
|
||
|
if (useragent.isOpera) {
|
||
|
editor.container.appendChild(blankImage);
|
||
|
// force layout
|
||
|
blankImage._top = blankImage.offsetTop;
|
||
|
}
|
||
|
dataTransfer.setDragImage && dataTransfer.setDragImage(blankImage, 0, 0);
|
||
|
if (useragent.isOpera) {
|
||
|
editor.container.removeChild(blankImage);
|
||
|
}
|
||
|
// clear Opera garbage
|
||
|
dataTransfer.clearData();
|
||
|
dataTransfer.setData("Text", editor.session.getTextRange());
|
||
|
|
||
|
isInternal = true;
|
||
|
this.setState("drag");
|
||
|
};
|
||
|
|
||
|
this.onDragEnd = function(e) {
|
||
|
mouseTarget.draggable = false;
|
||
|
isInternal = false;
|
||
|
this.setState(null);
|
||
|
if (!editor.getReadOnly()) {
|
||
|
var dropEffect = e.dataTransfer.dropEffect;
|
||
|
if (!dragOperation && dropEffect == "move")
|
||
|
// text was dragged outside the editor
|
||
|
editor.session.remove(editor.getSelectionRange());
|
||
|
editor.renderer.$cursorLayer.setBlinking(true);
|
||
|
}
|
||
|
this.editor.unsetStyle("ace_dragging");
|
||
|
};
|
||
|
|
||
|
this.onDragEnter = function(e) {
|
||
|
if (editor.getReadOnly() || !canAccept(e.dataTransfer))
|
||
|
return;
|
||
|
if (!dragSelectionMarker)
|
||
|
addDragMarker();
|
||
|
counter++;
|
||
|
// dataTransfer object does not save dropEffect across events on IE, so we store it in dragOperation
|
||
|
e.dataTransfer.dropEffect = dragOperation = getDropEffect(e);
|
||
|
return event.preventDefault(e);
|
||
|
};
|
||
|
|
||
|
this.onDragOver = function(e) {
|
||
|
if (editor.getReadOnly() || !canAccept(e.dataTransfer))
|
||
|
return;
|
||
|
// Opera doesn't trigger dragenter event on drag start
|
||
|
if (!dragSelectionMarker) {
|
||
|
addDragMarker();
|
||
|
counter++;
|
||
|
}
|
||
|
if (onMouseMoveTimer !== null)
|
||
|
onMouseMoveTimer = null;
|
||
|
x = e.clientX;
|
||
|
y = e.clientY;
|
||
|
|
||
|
e.dataTransfer.dropEffect = dragOperation = getDropEffect(e);
|
||
|
return event.preventDefault(e);
|
||
|
};
|
||
|
|
||
|
this.onDragLeave = function(e) {
|
||
|
counter--;
|
||
|
if (counter <= 0 && dragSelectionMarker) {
|
||
|
clearDragMarker();
|
||
|
dragOperation = null;
|
||
|
return event.preventDefault(e);
|
||
|
}
|
||
|
};
|
||
|
|
||
|
this.onDrop = function(e) {
|
||
|
if (!dragSelectionMarker)
|
||
|
return;
|
||
|
var dataTransfer = e.dataTransfer;
|
||
|
if (isInternal) {
|
||
|
switch (dragOperation) {
|
||
|
case "move":
|
||
|
if (range.contains(dragCursor.row, dragCursor.column)) {
|
||
|
// clear selection
|
||
|
range = {
|
||
|
start: dragCursor,
|
||
|
end: dragCursor
|
||
|
};
|
||
|
} else {
|
||
|
// move text
|
||
|
range = editor.moveText(range, dragCursor);
|
||
|
}
|
||
|
break;
|
||
|
case "copy":
|
||
|
// copy text
|
||
|
range = editor.moveText(range, dragCursor, true);
|
||
|
break;
|
||
|
}
|
||
|
} else {
|
||
|
var dropData = dataTransfer.getData('Text');
|
||
|
range = {
|
||
|
start: dragCursor,
|
||
|
end: editor.session.insert(dragCursor, dropData)
|
||
|
};
|
||
|
editor.focus();
|
||
|
dragOperation = null;
|
||
|
}
|
||
|
clearDragMarker();
|
||
|
return event.preventDefault(e);
|
||
|
};
|
||
|
|
||
|
event.addListener(mouseTarget, "dragstart", this.onDragStart.bind(mouseHandler));
|
||
|
event.addListener(mouseTarget, "dragend", this.onDragEnd.bind(mouseHandler));
|
||
|
event.addListener(mouseTarget, "dragenter", this.onDragEnter.bind(mouseHandler));
|
||
|
event.addListener(mouseTarget, "dragover", this.onDragOver.bind(mouseHandler));
|
||
|
event.addListener(mouseTarget, "dragleave", this.onDragLeave.bind(mouseHandler));
|
||
|
event.addListener(mouseTarget, "drop", this.onDrop.bind(mouseHandler));
|
||
|
|
||
|
function scrollCursorIntoView(cursor, prevCursor) {
|
||
|
var now = new Date().getTime();
|
||
|
var vMovement = !prevCursor || cursor.row != prevCursor.row;
|
||
|
var hMovement = !prevCursor || cursor.column != prevCursor.column;
|
||
|
if (!cursorMovedTime || vMovement || hMovement) {
|
||
|
editor.$blockScrolling += 1;
|
||
|
editor.moveCursorToPosition(cursor);
|
||
|
editor.$blockScrolling -= 1;
|
||
|
cursorMovedTime = now;
|
||
|
cursorPointOnCaretMoved = {x: x, y: y};
|
||
|
} else {
|
||
|
var distance = calcDistance(cursorPointOnCaretMoved.x, cursorPointOnCaretMoved.y, x, y);
|
||
|
if (distance > SCROLL_CURSOR_HYSTERESIS) {
|
||
|
cursorMovedTime = null;
|
||
|
} else if (now - cursorMovedTime >= SCROLL_CURSOR_DELAY) {
|
||
|
editor.renderer.scrollCursorIntoView();
|
||
|
cursorMovedTime = null;
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
function autoScroll(cursor, prevCursor) {
|
||
|
var now = new Date().getTime();
|
||
|
var lineHeight = editor.renderer.layerConfig.lineHeight;
|
||
|
var characterWidth = editor.renderer.layerConfig.characterWidth;
|
||
|
var editorRect = editor.renderer.scroller.getBoundingClientRect();
|
||
|
var offsets = {
|
||
|
x: {
|
||
|
left: x - editorRect.left,
|
||
|
right: editorRect.right - x
|
||
|
},
|
||
|
y: {
|
||
|
top: y - editorRect.top,
|
||
|
bottom: editorRect.bottom - y
|
||
|
}
|
||
|
};
|
||
|
var nearestXOffset = Math.min(offsets.x.left, offsets.x.right);
|
||
|
var nearestYOffset = Math.min(offsets.y.top, offsets.y.bottom);
|
||
|
var scrollCursor = {row: cursor.row, column: cursor.column};
|
||
|
if (nearestXOffset / characterWidth <= 2) {
|
||
|
scrollCursor.column += (offsets.x.left < offsets.x.right ? -3 : +2);
|
||
|
}
|
||
|
if (nearestYOffset / lineHeight <= 1) {
|
||
|
scrollCursor.row += (offsets.y.top < offsets.y.bottom ? -1 : +1);
|
||
|
}
|
||
|
var vScroll = cursor.row != scrollCursor.row;
|
||
|
var hScroll = cursor.column != scrollCursor.column;
|
||
|
var vMovement = !prevCursor || cursor.row != prevCursor.row;
|
||
|
if (vScroll || (hScroll && !vMovement)) {
|
||
|
if (!autoScrollStartTime)
|
||
|
autoScrollStartTime = now;
|
||
|
else if (now - autoScrollStartTime >= AUTOSCROLL_DELAY)
|
||
|
editor.renderer.scrollCursorIntoView(scrollCursor);
|
||
|
} else {
|
||
|
autoScrollStartTime = null;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
function onDragInterval() {
|
||
|
var prevCursor = dragCursor;
|
||
|
dragCursor = editor.renderer.screenToTextCoordinates(x, y);
|
||
|
scrollCursorIntoView(dragCursor, prevCursor);
|
||
|
autoScroll(dragCursor, prevCursor);
|
||
|
}
|
||
|
|
||
|
function addDragMarker() {
|
||
|
range = editor.selection.toOrientedRange();
|
||
|
dragSelectionMarker = editor.session.addMarker(range, "ace_selection", editor.getSelectionStyle());
|
||
|
editor.clearSelection();
|
||
|
if (editor.isFocused())
|
||
|
editor.renderer.$cursorLayer.setBlinking(false);
|
||
|
clearInterval(timerId);
|
||
|
timerId = setInterval(onDragInterval, 20);
|
||
|
counter = 0;
|
||
|
event.addListener(document, "mousemove", onMouseMove);
|
||
|
}
|
||
|
|
||
|
function clearDragMarker() {
|
||
|
clearInterval(timerId);
|
||
|
editor.session.removeMarker(dragSelectionMarker);
|
||
|
dragSelectionMarker = null;
|
||
|
editor.$blockScrolling += 1;
|
||
|
editor.selection.fromOrientedRange(range);
|
||
|
editor.$blockScrolling -= 1;
|
||
|
if (editor.isFocused() && !isInternal)
|
||
|
editor.renderer.$cursorLayer.setBlinking(!editor.getReadOnly());
|
||
|
range = null;
|
||
|
counter = 0;
|
||
|
autoScrollStartTime = null;
|
||
|
cursorMovedTime = null;
|
||
|
event.removeListener(document, "mousemove", onMouseMove);
|
||
|
}
|
||
|
|
||
|
// sometimes other code on the page can stop dragleave event leaving editor stuck in the drag state
|
||
|
var onMouseMoveTimer = null;
|
||
|
function onMouseMove() {
|
||
|
if (onMouseMoveTimer == null) {
|
||
|
onMouseMoveTimer = setTimeout(function() {
|
||
|
if (onMouseMoveTimer != null && dragSelectionMarker)
|
||
|
clearDragMarker();
|
||
|
}, 20);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
function canAccept(dataTransfer) {
|
||
|
var types = dataTransfer.types;
|
||
|
return !types || Array.prototype.some.call(types, function(type) {
|
||
|
return type == 'text/plain' || type == 'Text';
|
||
|
});
|
||
|
}
|
||
|
|
||
|
function getDropEffect(e) {
|
||
|
var copyAllowed = ['copy', 'copymove', 'all', 'uninitialized'];
|
||
|
var moveAllowed = ['move', 'copymove', 'linkmove', 'all', 'uninitialized'];
|
||
|
|
||
|
var copyModifierState = useragent.isMac ? e.altKey : e.ctrlKey;
|
||
|
|
||
|
// IE throws error while dragging from another app
|
||
|
var effectAllowed = "uninitialized";
|
||
|
try {
|
||
|
effectAllowed = e.dataTransfer.effectAllowed.toLowerCase();
|
||
|
} catch (e) {}
|
||
|
var dropEffect = "none";
|
||
|
|
||
|
if (copyModifierState && copyAllowed.indexOf(effectAllowed) >= 0)
|
||
|
dropEffect = "copy";
|
||
|
else if (moveAllowed.indexOf(effectAllowed) >= 0)
|
||
|
dropEffect = "move";
|
||
|
else if (copyAllowed.indexOf(effectAllowed) >= 0)
|
||
|
dropEffect = "copy";
|
||
|
|
||
|
return dropEffect;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
(function() {
|
||
|
|
||
|
this.dragWait = function() {
|
||
|
var interval = (new Date()).getTime() - this.mousedownEvent.time;
|
||
|
if (interval > this.editor.getDragDelay())
|
||
|
this.startDrag();
|
||
|
};
|
||
|
|
||
|
this.dragWaitEnd = function() {
|
||
|
var target = this.editor.container;
|
||
|
target.draggable = false;
|
||
|
this.startSelect(this.mousedownEvent.getDocumentPosition());
|
||
|
this.selectEnd();
|
||
|
};
|
||
|
|
||
|
this.dragReadyEnd = function(e) {
|
||
|
this.editor.renderer.$cursorLayer.setBlinking(!this.editor.getReadOnly());
|
||
|
this.editor.unsetStyle("ace_dragging");
|
||
|
this.dragWaitEnd();
|
||
|
};
|
||
|
|
||
|
this.startDrag = function(){
|
||
|
this.cancelDrag = false;
|
||
|
var target = this.editor.container;
|
||
|
target.draggable = true;
|
||
|
this.editor.renderer.$cursorLayer.setBlinking(false);
|
||
|
this.editor.setStyle("ace_dragging");
|
||
|
this.setState("dragReady");
|
||
|
};
|
||
|
|
||
|
this.onMouseDrag = function(e) {
|
||
|
var target = this.editor.container;
|
||
|
if (useragent.isIE && this.state == "dragReady") {
|
||
|
// IE does not handle [draggable] attribute set after mousedown
|
||
|
var distance = calcDistance(this.mousedownEvent.x, this.mousedownEvent.y, this.x, this.y);
|
||
|
if (distance > 3)
|
||
|
target.dragDrop();
|
||
|
}
|
||
|
if (this.state === "dragWait") {
|
||
|
var distance = calcDistance(this.mousedownEvent.x, this.mousedownEvent.y, this.x, this.y);
|
||
|
if (distance > 0) {
|
||
|
target.draggable = false;
|
||
|
this.startSelect(this.mousedownEvent.getDocumentPosition());
|
||
|
}
|
||
|
}
|
||
|
};
|
||
|
|
||
|
this.onMouseDown = function(e) {
|
||
|
if (!this.$dragEnabled)
|
||
|
return;
|
||
|
this.mousedownEvent = e;
|
||
|
var editor = this.editor;
|
||
|
|
||
|
var inSelection = e.inSelection();
|
||
|
var button = e.getButton();
|
||
|
var clickCount = e.domEvent.detail || 1;
|
||
|
if (clickCount === 1 && button === 0 && inSelection) {
|
||
|
this.mousedownEvent.time = (new Date()).getTime();
|
||
|
var eventTarget = e.domEvent.target || e.domEvent.srcElement;
|
||
|
if ("unselectable" in eventTarget)
|
||
|
eventTarget.unselectable = "on";
|
||
|
if (editor.getDragDelay()) {
|
||
|
// https://code.google.com/p/chromium/issues/detail?id=286700
|
||
|
if (useragent.isWebKit) {
|
||
|
self.cancelDrag = true;
|
||
|
var mouseTarget = editor.container;
|
||
|
mouseTarget.draggable = true;
|
||
|
}
|
||
|
this.setState("dragWait");
|
||
|
} else {
|
||
|
this.startDrag();
|
||
|
}
|
||
|
this.captureMouse(e, this.onMouseDrag.bind(this));
|
||
|
// TODO: a better way to prevent default handler without preventing browser default action
|
||
|
e.defaultPrevented = true;
|
||
|
}
|
||
|
};
|
||
|
|
||
|
}).call(DragdropHandler.prototype);
|
||
|
|
||
|
|
||
|
function calcDistance(ax, ay, bx, by) {
|
||
|
return Math.sqrt(Math.pow(bx - ax, 2) + Math.pow(by - ay, 2));
|
||
|
}
|
||
|
|
||
|
exports.DragdropHandler = DragdropHandler;
|
||
|
|
||
|
});
|