<!DOCTYPE html>
<html ng-app="AgRowDragNDrop">
<head>
<link rel="stylesheet" href="https://cdnjs.cloudflare.com/ajax/libs/ag-grid/4.1.5/styles/ag-grid.css"/>
<link rel="stylesheet" href="style.css">
</head>
<body ng-controller="DragNDropController">
<div>
<div ag-grid="agOptions" class="ag-grid ag-fresh"></div>
</div>
<br>
<div>
Id: {{movingId}}<br>
Name: {{movingName}}<br>
Type: {{movingType}}<br>
Moved from: {{movingFrom}}<br>
Moved to: {{movingTo}}
</div><br>
<hr><br>
Legend:<br>
[N] : position on the grid/tree<br>
<b>label</b> : can have children<br>
<b style="background: yellow">label</b> : currently has children<br>
<span style="color: red">label</span> : read only<br>
<!-- Vendors -->
<script src="//cdnjs.cloudflare.com/ajax/libs/angular.js/1.5.2/angular.min.js"></script>
<script src="ngDraggable.js"></script>
<script src="https://cdnjs.cloudflare.com/ajax/libs/ag-grid/4.1.5/ag-grid.js"></script>
<!-- Initialize -->
<script>
agGrid.initialiseAgGridWithAngular1(angular);
angular.module('AgRowDragNDrop', ['agGrid', 'ngDraggable']);
</script>
<!-- App -->
<script src="controller.js"></script>
</body>
</html>
angular.module('AgRowDragNDrop')
.controller('DragNDropController', dragNDropController);
if (typeof JSON.clone !== "function") {
JSON.clone = function(obj) {
return JSON.parse(JSON.stringify(obj));
};
}
function adjustPath(sourcePath, targetPath) {
var sourceIndexes = sourcePath.split("/");
var targetIndexes = targetPath.split("/");
var maxPathLength = sourceIndexes.length > targetIndexes.length ? sourceIndexes.length : targetIndexes.length;
var s, t, i;
for(i = 0; i < maxPathLength; i ++) {
s = i < sourceIndexes.length ? parseInt(sourceIndexes[i], 10) : -1;
t = i < targetIndexes.length ? parseInt(targetIndexes[i], 10) : -1;
if (s < t && i === sourceIndexes.length - 1) {
targetIndexes[i] = --t;
} else if (s > t || s === -1 || t === -1) {
break;
}
}
return targetIndexes.join("/");
}
function insertNodeByIndex(targetPath, nodeToInsert, nodesObj, position) {
var node = nodesObj;
var indexes = targetPath.split("/");
var folderId;
for (var i = 0; i < indexes.length; i++) {
var j = parseInt(indexes[i], 10);
// if position=before|after, we need to stop at the N-1 index
if (i < indexes.length - 1 || position === "inside") {
if (node.hasOwnProperty("Children")) {
node = node.Children[j];
} else if (node.length) {
node = node[j];
}
}
// can't move an element inside a read-only element
if (node.ReadOnly) {
return false;
}
// can't move an element inside itself or its children
if (nodeToInsert.Id === node.ParentFolderId) {
return false;
}
folderId = node.Id;
}
if (position === "inside") {
// can't put the element inside its current parent
if (nodeToInsert.ParentFolderId === folderId) {
return false;
}
nodeToInsert.ParentFolderId = folderId;
node.HasChildren = true;
if (node.hasOwnProperty("Children")) {
node.Children.push(nodeToInsert);
} else {
node.Children = [nodeToInsert];
}
return true;
} else if (position === "before" || position === "after") {
nodeToInsert.ParentFolderId = folderId || nodeToInsert.ParentFolderId;
if (node.hasOwnProperty("Children")) {
node = node.Children;
}
node.splice(position === "after" ? j + 1 : j, 0, nodeToInsert);
return true;
} else {
return false;
}
}
function extractNodeByIndex(sourcePath, nodesObj) {
var node = nodesObj;
var indexes = sourcePath.split("/");
for (var i = 0; i < indexes.length; i++) {
var j = parseInt(indexes[i], 10);
if (node.hasOwnProperty("Children")) {
if (i === indexes.length - 1) {
var _node = node.Children.splice(j, 1);
if (node.Children.length === 0) {
delete node.Children;
delete node.HasChildren;
}
node = _node;
} else {
node = node.Children[j];
}
} else if (node.length) {
if (i === indexes.length - 1) {
node = node.splice(j, 1);
} else {
node = node[j];
}
} else {
break;
}
}
return node[0]; //return the element, not the array
}
function dragNDropController($scope, $timeout) {
$scope.blueprint = [{
"Id": 9,
"Name": "Log into the system",
"Type": "Process",
"ParentFolderId": 0,
"CanHaveChildren": true
}, {
"Id": 1,
"Name": "Change password",
"Type": "Process",
"ParentFolderId": 0,
"HasChildren": true,
"CanHaveChildren": true,
"Children": [{
"Id": 6,
"Name": "Authenticating",
"Type": "Process",
"ParentFolderId": 1,
"HasChildren": true,
"CanHaveChildren": true,
"Children": [{
"Id": 10,
"Name": "Password criteria",
"Type": "Textual requirement",
"ParentFolderId": 6,
"HasChildren": true,
"CanHaveChildren": true,
"Children": [{
"Id": 11,
"Name": "Login screen",
"Type": "UI Mockup",
"ParentFolderId": 10
}, {
"Id": 12,
"Name": "How to change the password",
"Type": "Storyboard",
"ParentFolderId": 10
}]
}]
}, {
"Id": 7,
"Name": "User profile's fields",
"Type": "Glossary",
"ParentFolderId": 1,
"ReadOnly": true
}]
}, {
"Id": 2,
"Name": "Log out",
"Type": "Process",
"ParentFolderId": 0,
"CanHaveChildren": true
}, {
"Id": 4,
"Name": "List of users",
"Type": "Document",
"ParentFolderId": 0
}, {
"Id": 3,
"Name": "SAML login",
"Type": "Process",
"ParentFolderId": 0,
"CanHaveChildren": true
}, {
"Id": 13,
"Name": "Locked Workflow",
"Type": "User Story",
"ParentFolderId": 0,
"ReadOnly": true,
"HasChildren": true,
"CanHaveChildren": true,
"Children": [{
"Id": 14,
"Name": "User locked out",
"Type": "Process",
"ParentFolderId": 13,
"HasChildren": true,
"CanHaveChildren": true,
"Children": [{
"Id": 15,
"Name": "Denied",
"Type": "UI Mockup",
"ParentFolderId": 14
}]
}]
}, {
"Id": 5,
"Name": "Reset the password",
"Type": "Storyboard",
"ParentFolderId": 0
}, {
"Id": 8,
"Name": "General look&feel",
"Type": "UI Mockup",
"ParentFolderId": 0
}];
$scope.cols = [{
headerName: 'Folders',
cellRenderer: "group",
cellRendererParams: {
innerRenderer: function(params) {
return "<i>[" + params.rowIndex + "]</i> ID:" + params.node.data.Id + " " + params.data.Name + " (" + params.data.Type + ")";
}
},
cellClassRules: {
"has-children": function(params) {
return params.data.CanHaveChildren && params.data.HasChildren;
},
"can-have-children": function(params) {
return params.data.CanHaveChildren;
},
"read-only": function(params) {
var readOnly = params.data.ReadOnly;
// the following decides if the children of a read only parent are read only as well
/*var node = params.node;
while (node.parent) {
node = node.parent;
readOnly = node.data.ReadOnly
}*/
return readOnly;
}
},
field: 'Name'
}];
$scope.getNodeChildDetails = function(rowItem) {
if (rowItem.Children) {
return {
group: true,
expanded: rowItem.open,
children: rowItem.Children,
field: "Name",
key: rowItem.Id // the key is used by the default group cellRenderer
};
} else {
return null;
}
}
$scope.actions = {
onRowPostCreate: function(params) {
console.log(params)
var node = params.node;
var path = node.childIndex;
while (node.level) {
node = node.parent;
path = node.childIndex + "/" + path;
}
// the following decides if the children of a read only parent are read only as well
//var readOnlyChildren = !!params.eRow.querySelector(".ag-cell.read-only");
var readOnlyChildren = false;
var dragStartCallbackAsString = "actions.onDragRow('" + path + "', $data, $event)";
var dropSuccessCallbackAsString = "actions.onDropRow('" + path + "', $data, $event, 'inside')";
var dropSuccessCallbackAsStringPre = "actions.onDropRow('" + path + "', $data, $event, 'before')";
var dropSuccessCallbackAsStringPost = "actions.onDropRow('" + path + "', $data, $event, 'after')";
var $row = angular.element(params.eRow);
var $preRow = document.createElement("DIV");
$preRow.className = "pre-row";
var $postRow = document.createElement("DIV");
$postRow.className = "post-row";
$preRow.setAttribute('ng-drop', true);
$preRow.setAttribute('ng-drop-success', dropSuccessCallbackAsStringPre);
$postRow.setAttribute('ng-drop', true);
$postRow.setAttribute('ng-drop-success', dropSuccessCallbackAsStringPost);
$row[0].insertBefore($preRow, $row[0].firstChild);
$row[0].appendChild($postRow);
if (params.node.data.CanHaveChildren && (!params.node.data.ReadOnly && !readOnlyChildren)) {
$row[0].querySelector('.ag-cell').setAttribute('ng-drop', true);
$row[0].querySelector('.ag-cell').setAttribute('ng-drop-success', dropSuccessCallbackAsString);
}
if (!params.node.data.ReadOnly && !readOnlyChildren) {
$row.attr('ng-drag', true);
$row.attr('ng-drag-data', 'data');
$row.attr('ng-drag-start', dragStartCallbackAsString);
}
},
onDragRow: function(path, $data, $event) {
$scope.isMoving = true;
$scope.movingId = $data.Id;
$scope.movingName = $data.Name;
$scope.movingType = $data.Type;
$scope.movingFrom = path;
console.clear();
var data = JSON.clone($scope.blueprint);
console.log("before dragging", data);
},
onDropRow: function(path, $data, $event, position) {
if ($scope.isMoving) {
$scope.isMoving = false;
var dataBackup = JSON.clone($scope.blueprint);
var node = extractNodeByIndex($scope.movingFrom, $scope.blueprint);
var adjustedPath = adjustPath($scope.movingFrom, path);
$scope.movingTo = position + " element at pos: " + (path === adjustedPath ? path : path + " (" + adjustedPath + ")");
if (!insertNodeByIndex(adjustedPath, node, $scope.blueprint, position)) {
$scope.blueprint = dataBackup;
}
$scope.agOptions.api.setRowData($scope.blueprint);
var data = JSON.clone($scope.blueprint);
console.log("after dropping", data);
}
},
onRowGroupOpened: function(params) {
var node = params.node;
node.data.open = node.expanded;
}
};
$scope.agOptions = {
columnDefs: $scope.cols,
angularCompileRows: true,
icons: {
groupExpanded: "<button>-</button>",
groupContracted: "<button>+</button>"
},
processRowPostCreate: $scope.actions.onRowPostCreate,
getNodeChildDetails: $scope.getNodeChildDetails,
rowData: $scope.blueprint,
onRowGroupOpened: $scope.actions.onRowGroupOpened,
rowHeight: 24
};
$timeout(function() {
$scope.agOptions.api.sizeColumnsToFit();
});
}
/* Styles go here */
body {
font-family: Arial, sans-serif;
}
button {
font-weight: bold;
font-family: Times, serif;
font-size: 14px;
line-height: 16px;
height: 20px;
}
.can-have-children {
font-weight: bold;
}
.has-children {
background: yellow;
}
.read-only {
color: red;
}
.ag-cell:not(.read-only) {
cursor: move;
}
.ag-cell {
top: 2px;
height: 20px !important;
border: 0 !important;
padding: 0 10px !important;
}
.ag-row:not(.drag-over) .ag-cell.drag-enter {
background: lightblue;
}
.ag-row.drag-over {
opacity: 0.5;
cursor: move !important;
}
.pre-row,
.post-row {
position: absolute;
background: transparent;
height: 2px;
width: 100%;
font-size: 0;
}
.pre-row {
top: -1px;
z-index: 200;
}
.post-row {
bottom: -1px;
z-index: 300;
}
.ag-row:not(.drag-over) .pre-row.drag-enter,
.ag-row:not(.drag-over) .post-row.drag-enter {
background: url() left repeat-x;
}
.ag-row:not(.drag-over) .pre-row.drag-enter::before,
.ag-row:not(.drag-over) .post-row.drag-enter::before {
content: '';
border: solid 6px transparent;
border-left: solid 6px blue;
}
.ag-cell {
z-index: 100;
}
.ag-group-cell {
font-style: normal !important;
}
.ag-row-level-3 + .ag-row-level-0 .pre-row,
.ag-row-level-2 + .ag-row-level-0 .pre-row,
.ag-row-level-1 + .ag-row-level-0 .pre-row,
.ag-row-level-3 + .ag-row-level-1 .pre-row,
.ag-row-level-2 + .ag-row-level-1 .pre-row,
.ag-row-level-3 + .ag-row-level-2 .pre-row {
display: none;
}
/*
*
* https://github.com/fatlinesofcode/ngDraggable
*/
angular.module("ngDraggable", [])
.service('ngDraggable', [function() {
var scope = this;
scope.inputEvent = function(event) {
if (angular.isDefined(event.touches)) {
return event.touches[0];
}
//Checking both is not redundent. If only check if touches isDefined, angularjs isDefnied will return error and stop the remaining scripty if event.originalEvent is not defined.
else if (angular.isDefined(event.originalEvent) && angular.isDefined(event.originalEvent.touches)) {
return event.originalEvent.touches[0];
}
return event;
};
}])
.directive('ngDrag', ['$rootScope', '$parse', '$document', '$window', 'ngDraggable', function ($rootScope, $parse, $document, $window, ngDraggable) {
return {
restrict: 'A',
link: function (scope, element, attrs) {
scope.value = attrs.ngDrag;
var offset,_centerAnchor=false,_mx,_my,_tx,_ty,_mrx,_mry;
var _hasTouch = ('ontouchstart' in window) || window.DocumentTouch && document instanceof DocumentTouch;
var _pressEvents = 'touchstart mousedown';
var _moveEvents = 'touchmove mousemove';
var _releaseEvents = 'touchend mouseup';
var _dragHandle;
// to identify the element in order to prevent getting superflous events when a single element has both drag and drop directives on it.
var _myid = scope.$id;
var _data = null;
var _dragOffset = null;
var _dragEnabled = false;
var _pressTimer = null;
var onDragStartCallback = $parse(attrs.ngDragStart) || null;
var onDragStopCallback = $parse(attrs.ngDragStop) || null;
var onDragSuccessCallback = $parse(attrs.ngDragSuccess) || null;
var allowTransform = angular.isDefined(attrs.allowTransform) ? scope.$eval(attrs.allowTransform) : true;
var getDragData = $parse(attrs.ngDragData);
// deregistration function for mouse move events in $rootScope triggered by jqLite trigger handler
var _deregisterRootMoveListener = angular.noop;
var initialize = function () {
element.attr('draggable', 'false'); // prevent native drag
// check to see if drag handle(s) was specified
// if querySelectorAll is available, we use this instead of find
// as JQLite find is limited to tagnames
if (element[0].querySelectorAll) {
var dragHandles = angular.element(element[0].querySelectorAll('[ng-drag-handle]'));
} else {
var dragHandles = element.find('[ng-drag-handle]');
}
if (dragHandles.length) {
_dragHandle = dragHandles;
}
toggleListeners(true);
};
var toggleListeners = function (enable) {
if (!enable)return;
// add listeners.
scope.$on('$destroy', onDestroy);
scope.$watch(attrs.ngDrag, onEnableChange);
scope.$watch(attrs.ngCenterAnchor, onCenterAnchor);
// wire up touch events
if (_dragHandle) {
// handle(s) specified, use those to initiate drag
_dragHandle.on(_pressEvents, onpress);
} else {
// no handle(s) specified, use the element as the handle
element.on(_pressEvents, onpress);
}
if(! _hasTouch && element[0].nodeName.toLowerCase() == "img"){
element.on('mousedown', function(){ return false;}); // prevent native drag for images
}
};
var onDestroy = function (enable) {
toggleListeners(false);
};
var onEnableChange = function (newVal, oldVal) {
_dragEnabled = (newVal);
};
var onCenterAnchor = function (newVal, oldVal) {
if(angular.isDefined(newVal))
_centerAnchor = (newVal || 'true');
};
var isClickableElement = function (evt) {
return (
angular.isDefined(angular.element(evt.target).attr("ng-cancel-drag"))
);
};
/*
* When the element is clicked start the drag behaviour
* On touch devices as a small delay so as not to prevent native window scrolling
*/
var onpress = function(evt) {
if(! _dragEnabled)return;
if (isClickableElement(evt)) {
return;
}
if (evt.type == "mousedown" && evt.button != 0) {
// Do not start dragging on right-click
return;
}
if(_hasTouch){
cancelPress();
_pressTimer = setTimeout(function(){
cancelPress();
onlongpress(evt);
},100);
$document.on(_moveEvents, cancelPress);
$document.on(_releaseEvents, cancelPress);
}else{
onlongpress(evt);
}
};
var cancelPress = function() {
clearTimeout(_pressTimer);
$document.off(_moveEvents, cancelPress);
$document.off(_releaseEvents, cancelPress);
};
var onlongpress = function(evt) {
if(! _dragEnabled)return;
evt.preventDefault();
offset = element[0].getBoundingClientRect();
if(allowTransform)
_dragOffset = offset;
else{
_dragOffset = {left:document.body.scrollLeft, top:document.body.scrollTop};
}
element.centerX = element[0].offsetWidth / 2;
element.centerY = element[0].offsetHeight / 2;
_mx = ngDraggable.inputEvent(evt).pageX;//ngDraggable.getEventProp(evt, 'pageX');
_my = ngDraggable.inputEvent(evt).pageY;//ngDraggable.getEventProp(evt, 'pageY');
_mrx = _mx - offset.left;
_mry = _my - offset.top;
if (_centerAnchor) {
_tx = _mx - element.centerX - $window.pageXOffset;
_ty = _my - element.centerY - $window.pageYOffset;
} else {
_tx = _mx - _mrx - $window.pageXOffset;
_ty = _my - _mry - $window.pageYOffset;
}
$document.on(_moveEvents, onmove);
$document.on(_releaseEvents, onrelease);
// This event is used to receive manually triggered mouse move events
// jqLite unfortunately only supports triggerHandler(...)
// See http://api.jquery.com/triggerHandler/
// _deregisterRootMoveListener = $rootScope.$on('draggable:_triggerHandlerMove', onmove);
_deregisterRootMoveListener = $rootScope.$on('draggable:_triggerHandlerMove', function(event, origEvent) {
onmove(origEvent);
});
};
var onmove = function (evt) {
if (!_dragEnabled)return;
evt.preventDefault();
if (!element.hasClass('dragging')) {
_data = getDragData(scope);
element.addClass('dragging');
$rootScope.$broadcast('draggable:start', {x:_mx, y:_my, tx:_tx, ty:_ty, event:evt, element:element, data:_data});
if (onDragStartCallback ){
scope.$apply(function () {
onDragStartCallback(scope, {$data: _data, $event: evt});
});
}
}
_mx = ngDraggable.inputEvent(evt).pageX;//ngDraggable.getEventProp(evt, 'pageX');
_my = ngDraggable.inputEvent(evt).pageY;//ngDraggable.getEventProp(evt, 'pageY');
if (_centerAnchor) {
_tx = _mx - element.centerX - _dragOffset.left;
_ty = _my - element.centerY - _dragOffset.top;
} else {
_tx = _mx - _mrx - _dragOffset.left;
_ty = _my - _mry - _dragOffset.top;
}
moveElement(_tx, _ty);
$rootScope.$broadcast('draggable:move', { x: _mx, y: _my, tx: _tx, ty: _ty, event: evt, element: element, data: _data, uid: _myid, dragOffset: _dragOffset });
};
var onrelease = function(evt) {
if (!_dragEnabled)
return;
evt.preventDefault();
$rootScope.$broadcast('draggable:end', {x:_mx, y:_my, tx:_tx, ty:_ty, event:evt, element:element, data:_data, callback:onDragComplete, uid: _myid});
element.removeClass('dragging');
element.parent().find('.drag-enter').removeClass('drag-enter');
reset();
$document.off(_moveEvents, onmove);
$document.off(_releaseEvents, onrelease);
if (onDragStopCallback ){
scope.$apply(function () {
onDragStopCallback(scope, {$data: _data, $event: evt});
});
}
_deregisterRootMoveListener();
};
var onDragComplete = function(evt) {
if (!onDragSuccessCallback )return;
scope.$apply(function () {
onDragSuccessCallback(scope, {$data: _data, $event: evt});
});
};
var reset = function() {
if(allowTransform)
element.css({transform:'', 'z-index':'', '-webkit-transform':'', '-ms-transform':''});
else
element.css({'position':'',top:'',left:''});
};
var moveElement = function (x, y) {
if(allowTransform) {
element.css({
transform: 'matrix3d(1, 0, 0, 0, 0, 1, 0, 0, 0, 0, 1, 0, ' + x + ', ' + y + ', 0, 1)',
'z-index': 99999,
'-webkit-transform': 'matrix3d(1, 0, 0, 0, 0, 1, 0, 0, 0, 0, 1, 0, ' + x + ', ' + y + ', 0, 1)',
'-ms-transform': 'matrix(1, 0, 0, 1, ' + x + ', ' + y + ')'
});
}else{
element.css({'left':x+'px','top':y+'px', 'position':'fixed'});
}
};
initialize();
}
};
}])
.directive('ngDrop', ['$parse', '$timeout', '$window', '$document', 'ngDraggable', function ($parse, $timeout, $window, $document, ngDraggable) {
return {
restrict: 'A',
link: function (scope, element, attrs) {
scope.value = attrs.ngDrop;
scope.isTouching = false;
var _lastDropTouch=null;
var _myid = scope.$id;
var _dropEnabled=false;
var onDropCallback = $parse(attrs.ngDropSuccess);// || function(){};
var onDragStartCallback = $parse(attrs.ngDragStart);
var onDragStopCallback = $parse(attrs.ngDragStop);
var onDragMoveCallback = $parse(attrs.ngDragMove);
var initialize = function () {
toggleListeners(true);
};
var toggleListeners = function (enable) {
// remove listeners
if (!enable)return;
// add listeners.
scope.$watch(attrs.ngDrop, onEnableChange);
scope.$on('$destroy', onDestroy);
scope.$on('draggable:start', onDragStart);
scope.$on('draggable:move', onDragMove);
scope.$on('draggable:end', onDragEnd);
};
var onDestroy = function (enable) {
toggleListeners(false);
};
var onEnableChange = function (newVal, oldVal) {
_dropEnabled=newVal;
};
var onDragStart = function(evt, obj) {
if(! _dropEnabled)return;
isTouching(obj.x,obj.y,obj.element);
if (attrs.ngDragStart) {
$timeout(function(){
onDragStartCallback(scope, {$data: obj.data, $event: obj});
});
}
};
var onDragMove = function(evt, obj) {
if(! _dropEnabled)return;
isTouching(obj.x,obj.y,obj.element);
if (attrs.ngDragMove) {
$timeout(function(){
onDragMoveCallback(scope, {$data: obj.data, $event: obj});
});
}
};
var onDragEnd = function (evt, obj) {
// don't listen to drop events if this is the element being dragged
// only update the styles and return
if (!_dropEnabled || _myid === obj.uid) {
updateDragStyles(false, obj.element);
return;
}
if (isTouching(obj.x, obj.y, obj.element)) {
// call the ngDraggable ngDragSuccess element callback
if(obj.callback){
obj.callback(obj);
}
if (attrs.ngDropSuccess) {
$timeout(function(){
onDropCallback(scope, {$data: obj.data, $event: obj, $target: scope.$eval(scope.value)});
});
}
}
if (attrs.ngDragStop) {
$timeout(function(){
onDragStopCallback(scope, {$data: obj.data, $event: obj});
});
}
updateDragStyles(false, obj.element);
};
var isTouching = function(mouseX, mouseY, dragElement) {
var touching= hitTest(mouseX, mouseY);
scope.isTouching = touching;
if(touching){
_lastDropTouch = element;
}
updateDragStyles(touching, dragElement);
return touching;
};
var updateDragStyles = function(touching, dragElement) {
if(touching){
element.addClass('drag-enter');
dragElement.addClass('drag-over');
}else if(_lastDropTouch == element){
_lastDropTouch=null;
element.removeClass('drag-enter');
dragElement.removeClass('drag-over');
}
};
var hitTest = function(x, y) {
var bounds = element[0].getBoundingClientRect();// ngDraggable.getPrivOffset(element);
x -= $document[0].body.scrollLeft + $document[0].documentElement.scrollLeft;
y -= $document[0].body.scrollTop + $document[0].documentElement.scrollTop;
return x >= bounds.left
&& x <= bounds.right
&& y <= bounds.bottom
&& y >= bounds.top;
};
initialize();
}
};
}])
.directive('ngDragClone', ['$parse', '$timeout', 'ngDraggable', function ($parse, $timeout, ngDraggable) {
return {
restrict: 'A',
link: function (scope, element, attrs) {
var img, _allowClone=true;
var _dragOffset = null;
scope.clonedData = {};
var initialize = function () {
img = element.find('img');
element.attr('draggable', 'false');
img.attr('draggable', 'false');
reset();
toggleListeners(true);
};
var toggleListeners = function (enable) {
// remove listeners
if (!enable)return;
// add listeners.
scope.$on('draggable:start', onDragStart);
scope.$on('draggable:move', onDragMove);
scope.$on('draggable:end', onDragEnd);
preventContextMenu();
};
var preventContextMenu = function() {
// element.off('mousedown touchstart touchmove touchend touchcancel', absorbEvent_);
img.off('mousedown touchstart touchmove touchend touchcancel', absorbEvent_);
// element.on('mousedown touchstart touchmove touchend touchcancel', absorbEvent_);
img.on('mousedown touchstart touchmove touchend touchcancel', absorbEvent_);
};
var onDragStart = function(evt, obj, elm) {
_allowClone=true;
if(angular.isDefined(obj.data.allowClone)){
_allowClone=obj.data.allowClone;
}
if(_allowClone) {
scope.$apply(function () {
scope.clonedData = obj.data;
});
element.css('width', obj.element[0].offsetWidth);
element.css('height', obj.element[0].offsetHeight);
moveElement(obj.tx, obj.ty);
}
};
var onDragMove = function(evt, obj) {
if(_allowClone) {
_tx = obj.tx + obj.dragOffset.left;
_ty = obj.ty + obj.dragOffset.top;
moveElement(_tx, _ty);
}
};
var onDragEnd = function(evt, obj) {
//moveElement(obj.tx,obj.ty);
if(_allowClone) {
reset();
}
};
var reset = function() {
element.css({left:0,top:0, position:'fixed', 'z-index':-1, visibility:'hidden'});
};
var moveElement = function(x,y) {
element.css({
transform: 'matrix3d(1, 0, 0, 0, 0, 1, 0, 0, 0, 0, 1, 0, '+x+', '+y+', 0, 1)', 'z-index': 99999, 'visibility': 'visible',
'-webkit-transform': 'matrix3d(1, 0, 0, 0, 0, 1, 0, 0, 0, 0, 1, 0, '+x+', '+y+', 0, 1)',
'-ms-transform': 'matrix(1, 0, 0, 1, '+x+', '+y+')'
//,margin: '0' don't monkey with the margin,
});
};
var absorbEvent_ = function (event) {
var e = event;//.originalEvent;
e.preventDefault && e.preventDefault();
e.stopPropagation && e.stopPropagation();
e.cancelBubble = true;
e.returnValue = false;
return false;
};
initialize();
}
};
}])
.directive('ngPreventDrag', ['$parse', '$timeout', function ($parse, $timeout) {
return {
restrict: 'A',
link: function (scope, element, attrs) {
var initialize = function () {
element.attr('draggable', 'false');
toggleListeners(true);
};
var toggleListeners = function (enable) {
// remove listeners
if (!enable)return;
// add listeners.
element.on('mousedown touchstart touchmove touchend touchcancel', absorbEvent_);
};
var absorbEvent_ = function (event) {
var e = event.originalEvent;
e.preventDefault && e.preventDefault();
e.stopPropagation && e.stopPropagation();
e.cancelBubble = true;
e.returnValue = false;
return false;
};
initialize();
}
};
}])
.directive('ngCancelDrag', [function () {
return {
restrict: 'A',
link: function (scope, element, attrs) {
element.find('*').attr('ng-cancel-drag', 'ng-cancel-drag');
}
};
}])
.directive('ngDragScroll', ['$window', '$interval', '$timeout', '$document', '$rootScope', function($window, $interval, $timeout, $document, $rootScope) {
return {
restrict: 'A',
link: function(scope, element, attrs) {
var intervalPromise = null;
var lastMouseEvent = null;
var config = {
verticalScroll: attrs.verticalScroll || true,
horizontalScroll: attrs.horizontalScroll || true,
activationDistance: attrs.activationDistance || 75,
scrollDistance: attrs.scrollDistance || 15
};
var reqAnimFrame = (function() {
return window.requestAnimationFrame ||
window.webkitRequestAnimationFrame ||
window.mozRequestAnimationFrame ||
window.oRequestAnimationFrame ||
window.msRequestAnimationFrame ||
function( /* function FrameRequestCallback */ callback, /* DOMElement Element */ element ) {
window.setTimeout(callback, 1000 / 60);
};
})();
var animationIsOn = false;
var createInterval = function() {
animationIsOn = true;
function nextFrame(callback) {
var args = Array.prototype.slice.call(arguments);
if(animationIsOn) {
reqAnimFrame(function () {
$rootScope.$apply(function () {
callback.apply(null, args);
nextFrame(callback);
});
})
}
}
nextFrame(function() {
if (!lastMouseEvent) return;
var viewportWidth = Math.max(document.documentElement.clientWidth, window.innerWidth || 0);
var viewportHeight = Math.max(document.documentElement.clientHeight, window.innerHeight || 0);
var scrollX = 0;
var scrollY = 0;
if (config.horizontalScroll) {
// If horizontal scrolling is active.
if (lastMouseEvent.clientX < config.activationDistance) {
// If the mouse is on the left of the viewport within the activation distance.
scrollX = -config.scrollDistance;
}
else if (lastMouseEvent.clientX > viewportWidth - config.activationDistance) {
// If the mouse is on the right of the viewport within the activation distance.
scrollX = config.scrollDistance;
}
}
if (config.verticalScroll) {
// If vertical scrolling is active.
if (lastMouseEvent.clientY < config.activationDistance) {
// If the mouse is on the top of the viewport within the activation distance.
scrollY = -config.scrollDistance;
}
else if (lastMouseEvent.clientY > viewportHeight - config.activationDistance) {
// If the mouse is on the bottom of the viewport within the activation distance.
scrollY = config.scrollDistance;
}
}
if (scrollX !== 0 || scrollY !== 0) {
// Record the current scroll position.
var currentScrollLeft = ($window.pageXOffset || $document[0].documentElement.scrollLeft);
var currentScrollTop = ($window.pageYOffset || $document[0].documentElement.scrollTop);
// Remove the transformation from the element, scroll the window by the scroll distance
// record how far we scrolled, then reapply the element transformation.
var elementTransform = element.css('transform');
element.css('transform', 'initial');
$window.scrollBy(scrollX, scrollY);
var horizontalScrollAmount = ($window.pageXOffset || $document[0].documentElement.scrollLeft) - currentScrollLeft;
var verticalScrollAmount = ($window.pageYOffset || $document[0].documentElement.scrollTop) - currentScrollTop;
element.css('transform', elementTransform);
lastMouseEvent.pageX += horizontalScrollAmount;
lastMouseEvent.pageY += verticalScrollAmount;
$rootScope.$emit('draggable:_triggerHandlerMove', lastMouseEvent);
}
});
};
var clearInterval = function() {
animationIsOn = false;
};
scope.$on('draggable:start', function(event, obj) {
// Ignore this event if it's not for this element.
if (obj.element[0] !== element[0]) return;
if (!animationIsOn) createInterval();
});
scope.$on('draggable:end', function(event, obj) {
// Ignore this event if it's not for this element.
if (obj.element[0] !== element[0]) return;
if (animationIsOn) clearInterval();
});
scope.$on('draggable:move', function(event, obj) {
// Ignore this event if it's not for this element.
if (obj.element[0] !== element[0]) return;
lastMouseEvent = obj.event;
});
}
};
}]);