Skip to content

Instantly share code, notes, and snippets.

@ivanbarlog
Created November 2, 2015 17:10
Show Gist options
  • Save ivanbarlog/09ac3b3e5d6a9d55ab5e to your computer and use it in GitHub Desktop.
Save ivanbarlog/09ac3b3e5d6a9d55ab5e to your computer and use it in GitHub Desktop.
Nestable
/*!
* Nestable jQuery Plugin - Copyright (c) 2014 Ramon Smit - https://github.com/RamonSmit/Nestable
*/
;
(function($, window, document, undefined) {
var hasTouch = 'ontouchstart' in window;
/**
* Detect CSS pointer-events property
* events are normally disabled on the dragging element to avoid conflicts
* https://github.com/ausi/Feature-detection-technique-for-pointer-events/blob/master/modernizr-pointerevents.js
*/
var hasPointerEvents = (function() {
var el = document.createElement('div'),
docEl = document.documentElement;
if(!('pointerEvents' in el.style)) {
return false;
}
el.style.pointerEvents = 'auto';
el.style.pointerEvents = 'x';
docEl.appendChild(el);
var supports = window.getComputedStyle && window.getComputedStyle(el, '').pointerEvents === 'auto';
docEl.removeChild(el);
return !!supports;
})();
var eStart = hasTouch ? 'touchstart' : 'mousedown',
eMove = hasTouch ? 'touchmove' : 'mousemove',
eEnd = hasTouch ? 'touchend' : 'mouseup',
eCancel = hasTouch ? 'touchcancel' : 'mouseup';
var defaults = {
contentCallback: function(item) {return item.content || '' ? item.content : item.id;},
listNodeName: 'ol',
itemNodeName: 'li',
handleNodeName: 'div',
contentNodeName: 'span',
rootClass: 'dd',
listClass: 'dd-list',
itemClass: 'dd-item',
dragClass: 'dd-dragel',
handleClass: 'dd-handle',
contentClass: 'dd-content',
collapsedClass: 'dd-collapsed',
placeClass: 'dd-placeholder',
noDragClass: 'dd-nodrag',
noChildrenClass: 'dd-nochildren',
emptyClass: 'dd-empty',
expandBtnHTML: '<button class="dd-expand" data-action="expand" type="button">Expand</button>',
collapseBtnHTML: '<button class="dd-collapse" data-action="collapse" type="button">Collapse</button>',
group: 0,
maxDepth: 5,
threshold: 20,
fixedDepth: false, //fixed item's depth
fixed: false,
includeContent: false,
callback: function(l, e) {},
onDragStart: function(l, e) {},
listRenderer: function(children, options) {
var html = '<' + options.listNodeName + ' class="' + options.listClass + '">';
html += children;
html += '</' + options.listNodeName + '>';
return html;
},
itemRenderer: function(item_attrs, content, children, options, item) {
var item_attrs_string = $.map(item_attrs, function(value, key) {
return ' ' + key + '="' + value + '"';
}).join(' ');
var html = '<' + options.itemNodeName + item_attrs_string + '>';
html += '<' + options.handleNodeName + ' class="' + options.handleClass + '">';
html += '<' + options.contentNodeName + ' class="' + options.contentClass + '">';
html += content;
html += '</' + options.contentNodeName + '>';
html += '</' + options.handleNodeName + '>';
html += children;
html += '</' + options.itemNodeName + '>';
return html;
}
};
function Plugin(element, options) {
this.w = $(document);
this.el = $(element);
if(!options) {
options = defaults;
}
if(options.rootClass !== undefined && options.rootClass !== 'dd') {
options.listClass = options.listClass ? options.listClass : options.rootClass + '-list';
options.itemClass = options.itemClass ? options.itemClass : options.rootClass + '-item';
options.dragClass = options.dragClass ? options.dragClass : options.rootClass + '-dragel';
options.handleClass = options.handleClass ? options.handleClass : options.rootClass + '-handle';
options.collapsedClass = options.collapsedClass ? options.collapsedClass : options.rootClass + '-collapsed';
options.placeClass = options.placeClass ? options.placeClass : options.rootClass + '-placeholder';
options.noDragClass = options.noDragClass ? options.noDragClass : options.rootClass + '-nodrag';
options.noChildrenClass = options.noChildrenClass ? options.noChildrenClass : options.rootClass + '-nochildren';
options.emptyClass = options.emptyClass ? options.emptyClass : options.rootClass + '-empty';
}
this.options = $.extend({}, defaults, options);
// build HTML from serialized JSON if passed
if(this.options.json !== undefined) {
this._build();
}
this.init();
}
Plugin.prototype = {
init: function() {
var list = this;
list.reset();
list.el.data('nestable-group', this.options.group);
list.placeEl = $('<div class="' + list.options.placeClass + '"/>');
$.each(this.el.find(list.options.itemNodeName), function(k, el) {
var item = $(el),
parent = item.parent();
list.setParent(item);
if(parent.hasClass(list.options.collapsedClass)) {
list.collapseItem(parent.parent());
}
});
list.el.on('click', 'button', function(e) {
if(list.dragEl || (!hasTouch && e.button !== 0)) {
return;
}
var target = $(e.currentTarget),
action = target.data('action'),
item = target.parent(list.options.itemNodeName);
if(action === 'collapse') {
list.collapseItem(item);
}
if(action === 'expand') {
list.expandItem(item);
}
});
var onStartEvent = function(e) {
var handle = $(e.target);
if(!handle.hasClass(list.options.handleClass)) {
if(handle.closest('.' + list.options.noDragClass).length) {
return;
}
handle = handle.closest('.' + list.options.handleClass);
}
if(!handle.length || list.dragEl || (!hasTouch && e.which !== 1) || (hasTouch && e.touches.length !== 1)) {
return;
}
e.preventDefault();
list.dragStart(hasTouch ? e.touches[0] : e);
};
var onMoveEvent = function(e) {
if(list.dragEl) {
e.preventDefault();
list.dragMove(hasTouch ? e.touches[0] : e);
}
};
var onEndEvent = function(e) {
if(list.dragEl) {
e.preventDefault();
list.dragStop(hasTouch ? e.touches[0] : e);
}
};
if(hasTouch) {
list.el[0].addEventListener(eStart, onStartEvent, false);
window.addEventListener(eMove, onMoveEvent, false);
window.addEventListener(eEnd, onEndEvent, false);
window.addEventListener(eCancel, onEndEvent, false);
}
else {
list.el.on(eStart, onStartEvent);
list.w.on(eMove, onMoveEvent);
list.w.on(eEnd, onEndEvent);
}
},
_build: function() {
function escapeHtml(text) {
var map = {
'&': '&amp;',
'<': '&lt;',
'>': '&gt;',
'"': '&quot;',
"'": '&#039;'
};
return text + "".replace(/[&<>"']/g, function(m) { return map[m]; });
}
function filterClasses(classes) {
var new_classes = {};
for(var k in classes) {
// Remove duplicates
new_classes[classes[k]] = classes[k];
}
return new_classes;
}
function createClassesString(item, options) {
var classes = item.classes || {};
if(typeof classes == 'string') {
classes = [classes];
}
var item_classes = filterClasses(classes);
item_classes[options.itemClass] = options.itemClass;
// create class string
return $.map(item_classes, function(val) {
return val;
}).join(' ');
}
function createDataAttrs(attr) {
attr = $.extend({}, attr);
delete attr.children;
delete attr.classes;
delete attr.content;
var data_attrs = {};
$.each(attr, function(key, value) {
if(typeof value == 'object') {
value = JSON.stringify(value);
}
data_attrs["data-" + key] = escapeHtml(value);
});
return data_attrs;
}
function buildList(items, options) {
if(!items) {
return '';
}
var children = '';
$.each(items, function(index, sub) {
children += buildItem(sub, options);
});
return options.listRenderer(children, options);
}
function buildItem(item, options) {
var item_attrs = createDataAttrs(item);
item_attrs["class"] = createClassesString(item, options);
var content = options.contentCallback(item);
var children = buildList(item.children, options);
return options.itemRenderer(item_attrs, content, children, options, item);
}
var json = this.options.json;
if(typeof json == 'string') {
json = JSON.parse(json);
}
$(this.el).html(buildList(json, this.options));
},
serialize: function() {
var data, list = this, step = function(level) {
var array = [],
items = level.children(list.options.itemNodeName);
items.each(function() {
var li = $(this),
item = $.extend({}, li.data()),
sub = li.children(list.options.listNodeName);
if(list.options.includeContent) {
var content = li.find('.' + list.options.contentClass).html();
if(content) {
item.content = content;
}
}
if(sub.length) {
item.children = step(sub);
}
array.push(item);
});
return array;
};
data = step(list.el.find(list.options.listNodeName).first());
return data;
},
returnOptions: function() {
return this.options;
},
serialise: function() {
return this.serialize();
},
reset: function() {
this.mouse = {
offsetX: 0,
offsetY: 0,
startX: 0,
startY: 0,
lastX: 0,
lastY: 0,
nowX: 0,
nowY: 0,
distX: 0,
distY: 0,
dirAx: 0,
dirX: 0,
dirY: 0,
lastDirX: 0,
lastDirY: 0,
distAxX: 0,
distAxY: 0
};
this.moving = false;
this.dragEl = null;
this.dragRootEl = null;
this.dragDepth = 0;
this.hasNewRoot = false;
this.pointEl = null;
},
expandItem: function(li) {
li.removeClass(this.options.collapsedClass);
},
collapseItem: function(li) {
var lists = li.children(this.options.listNodeName);
if(lists.length) {
li.addClass(this.options.collapsedClass);
}
},
expandAll: function() {
var list = this;
list.el.find(list.options.itemNodeName).each(function() {
list.expandItem($(this));
});
},
collapseAll: function() {
var list = this;
list.el.find(list.options.itemNodeName).each(function() {
list.collapseItem($(this));
});
},
setParent: function(li) {
if(li.children(this.options.listNodeName).length) {
li.prepend($(this.options.expandBtnHTML));
li.prepend($(this.options.collapseBtnHTML));
}
},
unsetParent: function(li) {
li.removeClass(this.options.collapsedClass);
li.children('[data-action]').remove();
li.children(this.options.listNodeName).remove();
},
dragStart: function(e) {
var mouse = this.mouse,
target = $(e.target),
dragItem = target.closest(this.options.itemNodeName);
this.options.onDragStart.call(this, this.el, dragItem);
this.placeEl.css('height', dragItem.height());
mouse.offsetX = e.pageX - dragItem.offset().left;
mouse.offsetY = e.pageY - dragItem.offset().top;
mouse.startX = mouse.lastX = e.pageX;
mouse.startY = mouse.lastY = e.pageY;
this.dragRootEl = this.el;
this.dragEl = $(document.createElement(this.options.listNodeName)).addClass(this.options.listClass + ' ' + this.options.dragClass);
this.dragEl.css('width', dragItem.outerWidth());
this.setIndexOfItem(dragItem);
// fix for zepto.js
//dragItem.after(this.placeEl).detach().appendTo(this.dragEl);
dragItem.after(this.placeEl);
dragItem[0].parentNode.removeChild(dragItem[0]);
dragItem.appendTo(this.dragEl);
$(document.body).append(this.dragEl);
this.dragEl.css({
'left': e.pageX - mouse.offsetX,
'top': e.pageY - mouse.offsetY
});
// total depth of dragging item
var i, depth,
items = this.dragEl.find(this.options.itemNodeName);
for(i = 0; i < items.length; i++) {
depth = $(items[i]).parents(this.options.listNodeName).length;
if(depth > this.dragDepth) {
this.dragDepth = depth;
}
}
},
setIndexOfItem: function(item, index) {
if((typeof index) === 'undefined') {
index = [];
}
index.unshift(item.index());
if($(item[0].parentNode)[0] !== this.dragRootEl[0]) {
this.setIndexOfItem($(item[0].parentNode), index);
}
else {
this.dragEl.data('indexOfItem', index);
}
},
restoreItemAtIndex: function(dragElement) {
var indexArray = this.dragEl.data('indexOfItem'),
currentEl = this.el;
for(i = 0; i < indexArray.length; i++) {
if((indexArray.length - 1) === parseInt(i)) {
placeElement(currentEl, dragElement);
return
}
currentEl = currentEl[0].children[indexArray[i]];
}
function placeElement(currentEl, dragElement) {
if(indexArray[indexArray.length - 1] === 0) {
$(currentEl).prepend(dragElement.clone());
}
else {
$(currentEl.children[indexArray[indexArray.length - 1] - 1]).after(dragElement.clone());
}
}
},
dragStop: function(e) {
// fix for zepto.js
//this.placeEl.replaceWith(this.dragEl.children(this.options.itemNodeName + ':first').detach());
var el = this.dragEl.children(this.options.itemNodeName).first();
el[0].parentNode.removeChild(el[0]);
this.placeEl.replaceWith(el);
if(this.hasNewRoot) {
if(this.options.fixed === true) {
this.restoreItemAtIndex(el);
}
else {
this.el.trigger('lostItem');
}
this.dragRootEl.trigger('gainedItem');
}
else {
this.dragRootEl.trigger('change');
}
this.dragEl.remove();
this.options.callback.call(this, this.dragRootEl, el);
this.reset();
},
dragMove: function(e) {
var list, parent, prev, next, depth,
opt = this.options,
mouse = this.mouse;
this.dragEl.css({
'left': e.pageX - mouse.offsetX,
'top': e.pageY - mouse.offsetY
});
// mouse position last events
mouse.lastX = mouse.nowX;
mouse.lastY = mouse.nowY;
// mouse position this events
mouse.nowX = e.pageX;
mouse.nowY = e.pageY;
// distance mouse moved between events
mouse.distX = mouse.nowX - mouse.lastX;
mouse.distY = mouse.nowY - mouse.lastY;
// direction mouse was moving
mouse.lastDirX = mouse.dirX;
mouse.lastDirY = mouse.dirY;
// direction mouse is now moving (on both axis)
mouse.dirX = mouse.distX === 0 ? 0 : mouse.distX > 0 ? 1 : -1;
mouse.dirY = mouse.distY === 0 ? 0 : mouse.distY > 0 ? 1 : -1;
// axis mouse is now moving on
var newAx = Math.abs(mouse.distX) > Math.abs(mouse.distY) ? 1 : 0;
// do nothing on first move
if(!mouse.moving) {
mouse.dirAx = newAx;
mouse.moving = true;
return;
}
// calc distance moved on this axis (and direction)
if(mouse.dirAx !== newAx) {
mouse.distAxX = 0;
mouse.distAxY = 0;
}
else {
mouse.distAxX += Math.abs(mouse.distX);
if(mouse.dirX !== 0 && mouse.dirX !== mouse.lastDirX) {
mouse.distAxX = 0;
}
mouse.distAxY += Math.abs(mouse.distY);
if(mouse.dirY !== 0 && mouse.dirY !== mouse.lastDirY) {
mouse.distAxY = 0;
}
}
mouse.dirAx = newAx;
/**
* move horizontal
*/
if(mouse.dirAx && mouse.distAxX >= opt.threshold) {
// reset move distance on x-axis for new phase
mouse.distAxX = 0;
prev = this.placeEl.prev(opt.itemNodeName);
// increase horizontal level if previous sibling exists, is not collapsed, and can have children
if(mouse.distX > 0 && prev.length && !prev.hasClass(opt.collapsedClass) && !prev.hasClass(opt.noChildrenClass)) {
// cannot increase level when item above is collapsed
list = prev.find(opt.listNodeName).last();
// check if depth limit has reached
depth = this.placeEl.parents(opt.listNodeName).length;
if(depth + this.dragDepth <= opt.maxDepth) {
// create new sub-level if one doesn't exist
if(!list.length) {
list = $('<' + opt.listNodeName + '/>').addClass(opt.listClass);
list.append(this.placeEl);
prev.append(list);
this.setParent(prev);
}
else {
// else append to next level up
list = prev.children(opt.listNodeName).last();
list.append(this.placeEl);
}
}
}
// decrease horizontal level
if(mouse.distX < 0) {
// we can't decrease a level if an item preceeds the current one
next = this.placeEl.next(opt.itemNodeName);
if(!next.length) {
parent = this.placeEl.parent();
this.placeEl.closest(opt.itemNodeName).after(this.placeEl);
if(!parent.children().length) {
this.unsetParent(parent.parent());
}
}
}
}
var isEmpty = false;
// find list item under cursor
if(!hasPointerEvents) {
this.dragEl[0].style.visibility = 'hidden';
}
this.pointEl = $(document.elementFromPoint(e.pageX - document.body.scrollLeft, e.pageY - (window.pageYOffset || document.documentElement.scrollTop)));
if(!hasPointerEvents) {
this.dragEl[0].style.visibility = 'visible';
}
if(this.pointEl.hasClass(opt.handleClass)) {
this.pointEl = this.pointEl.closest(opt.itemNodeName);
}
if(this.pointEl.hasClass(opt.emptyClass)) {
isEmpty = true;
}
else if(!this.pointEl.length || !this.pointEl.hasClass(opt.itemClass)) {
return;
}
// find parent list of item under cursor
var pointElRoot = this.pointEl.closest('.' + opt.rootClass),
isNewRoot = this.dragRootEl.data('nestable-id') !== pointElRoot.data('nestable-id');
/**
* move vertical
*/
if(!mouse.dirAx || isNewRoot || isEmpty) {
// check if groups match if dragging over new root
if(isNewRoot && opt.group !== pointElRoot.data('nestable-group')) {
return;
}
// fixed item's depth, use for some list has specific type, eg:'Volume, Section, Chapter ...'
if(this.options.fixedDepth && this.dragDepth + 1 !== this.pointEl.parents(opt.listNodeName).length) {
return;
}
// check depth limit
depth = this.dragDepth - 1 + this.pointEl.parents(opt.listNodeName).length;
if(depth > opt.maxDepth) {
return;
}
var before = e.pageY < (this.pointEl.offset().top + this.pointEl.height() / 2);
parent = this.placeEl.parent();
// if empty create new list to replace empty placeholder
if(isEmpty) {
list = $(document.createElement(opt.listNodeName)).addClass(opt.listClass);
list.append(this.placeEl);
this.pointEl.replaceWith(list);
}
else if(before) {
this.pointEl.before(this.placeEl);
}
else {
this.pointEl.after(this.placeEl);
}
if(!parent.children().length) {
this.unsetParent(parent.parent());
}
if(!this.dragRootEl.find(opt.itemNodeName).length) {
this.dragRootEl.append('<div class="' + opt.emptyClass + '"/>');
}
// parent root list has changed
this.dragRootEl = pointElRoot;
if(isNewRoot) {
this.hasNewRoot = this.el[0] !== this.dragRootEl[0];
}
}
}
};
$.fn.nestable = function(params) {
var lists = this,
retval = this;
if(!('Nestable' in window)) {
window.Nestable = {};
Nestable.counter = 0;
}
lists.each(function() {
var plugin = $(this).data("nestable");
if(!plugin) {
Nestable.counter++;
$(this).data("nestable", new Plugin(this, params));
$(this).data("nestable-id", Nestable.counter);
}
else {
if(typeof params === 'string' && typeof plugin[params] === 'function') {
retval = plugin[params]();
}
}
});
return retval || lists;
};
})(window.jQuery || window.Zepto, window, document);
$("#main-menu").nestable({
maxDepth: 3,
group: 0,
onDragStart: function (l, e) {
var type = $(e).data('type');
switch (type) {
case 'singlePage':
l.find("[data-type=parentPage]").removeClass('dd-nochildren');
l.find("[data-type=pageList]").removeClass('dd-nochildren');
break;
case 'parentPage':
l.find("[data-type=parentPage]").addClass('dd-nochildren');
l.find("[data-type=pageList]").addClass('dd-nochildren');
break;
case 'pageList':
l.find("[data-type=parentPage]").removeClass('dd-nochildren');
break;
default:
console.error("Invalid type");
}
this.init();
}
});
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment