diff --git a/package-lock.json b/package-lock.json
index eab4ce9b3..d6e30513e 100644
--- a/package-lock.json
+++ b/package-lock.json
@@ -3435,21 +3435,6 @@
"resolved": "https://registry.npmjs.org/jquery/-/jquery-3.4.1.tgz",
"integrity": "sha512-36+AdBzCL+y6qjw5Tx7HgzeGCzC81MDDgaUP8ld2zhx58HdqXGoBd+tHdrBMiyjGQs0Hxs/MLZTu/eHNJJuWPw=="
},
- "jquery-sortable": {
- "version": "0.9.13",
- "resolved": "https://registry.npmjs.org/jquery-sortable/-/jquery-sortable-0.9.13.tgz",
- "integrity": "sha1-HL+2VQE6B0c3BXHwbiL1JKAP+6I=",
- "requires": {
- "jquery": "^2.1.2"
- },
- "dependencies": {
- "jquery": {
- "version": "2.2.4",
- "resolved": "https://registry.npmjs.org/jquery/-/jquery-2.2.4.tgz",
- "integrity": "sha1-LInWiJterFIqfuoywUUhVZxsvwI="
- }
- }
- },
"js-base64": {
"version": "2.5.1",
"resolved": "https://registry.npmjs.org/js-base64/-/js-base64-2.5.1.tgz",
diff --git a/package.json b/package.json
index fb06afc83..0198f0ccc 100644
--- a/package.json
+++ b/package.json
@@ -26,7 +26,6 @@
"codemirror": "^5.47.0",
"dropzone": "^5.5.1",
"jquery": "^3.4.1",
- "jquery-sortable": "^0.9.13",
"markdown-it": "^8.4.2",
"markdown-it-task-lists": "^2.1.1",
"sortablejs": "^1.9.0",
diff --git a/public/libs/jquery-sortable/jquery-sortable.min.js b/public/libs/jquery-sortable/jquery-sortable.min.js
deleted file mode 100644
index 4b483e5e8..000000000
--- a/public/libs/jquery-sortable/jquery-sortable.min.js
+++ /dev/null
@@ -1,19 +0,0 @@
-!function(d,B,m,f){function v(a,b){var c=Math.max(0,a[0]-b[0],b[0]-a[1]),e=Math.max(0,a[2]-b[1],b[1]-a[3]);return c+e}function w(a,b,c,e){var k=a.length;e=e?"offset":"position";for(c=c||0;k--;){var g=a[k].el?a[k].el:d(a[k]),l=g[e]();l.left+=parseInt(g.css("margin-left"),10);l.top+=parseInt(g.css("margin-top"),10);b[k]=[l.left-c,l.left+g.outerWidth()+c,l.top-c,l.top+g.outerHeight()+c]}}function p(a,b){var c=b.offset();return{left:a.left-c.left,top:a.top-c.top}}function x(a,b,c){b=[b.left,b.top];c=
-c&&[c.left,c.top];for(var e,k=a.length,d=[];k--;)e=a[k],d[k]=[k,v(e,b),c&&v(e,c)];return d=d.sort(function(a,b){return b[1]-a[1]||b[2]-a[2]||b[0]-a[0]})}function q(a){this.options=d.extend({},n,a);this.containers=[];this.options.rootGroup||(this.scrollProxy=d.proxy(this.scroll,this),this.dragProxy=d.proxy(this.drag,this),this.dropProxy=d.proxy(this.drop,this),this.placeholder=d(this.options.placeholder),a.isValidTarget||(this.options.isValidTarget=f))}function t(a,b){this.el=a;this.options=d.extend({},
-z,b);this.group=q.get(this.options);this.rootGroup=this.options.rootGroup||this.group;this.handle=this.rootGroup.options.handle||this.rootGroup.options.itemSelector;var c=this.rootGroup.options.itemPath;this.target=c?this.el.find(c):this.el;this.target.on(r.start,this.handle,d.proxy(this.dragInit,this));this.options.drop&&this.group.containers.push(this)}var r,z={drag:!0,drop:!0,exclude:"",nested:!0,vertical:!0},n={afterMove:function(a,b,c){},containerPath:"",containerSelector:"ol, ul",distance:0,
-delay:0,handle:"",itemPath:"",itemSelector:"li",bodyClass:"dragging",draggedClass:"dragged",isValidTarget:function(a,b){return!0},onCancel:function(a,b,c,e){},onDrag:function(a,b,c,e){a.css(b)},onDragStart:function(a,b,c,e){a.css({height:a.outerHeight(),width:a.outerWidth()});a.addClass(b.group.options.draggedClass);d("body").addClass(b.group.options.bodyClass)},onDrop:function(a,b,c,e){a.removeClass(b.group.options.draggedClass).removeAttr("style");d("body").removeClass(b.group.options.bodyClass)},
-onMousedown:function(a,b,c){if(!c.target.nodeName.match(/^(input|select|textarea)$/i))return c.preventDefault(),!0},placeholderClass:"placeholder",placeholder:'
',pullPlaceholder:!0,serialize:function(a,b,c){a=d.extend({},a.data());if(c)return[b];b[0]&&(a.children=b);delete a.subContainers;delete a.sortable;return a},tolerance:0},s={},y=0,A={left:0,top:0,bottom:0,right:0};r={start:"touchstart.sortable mousedown.sortable",drop:"touchend.sortable touchcancel.sortable mouseup.sortable",
-drag:"touchmove.sortable mousemove.sortable",scroll:"scroll.sortable"};q.get=function(a){s[a.group]||(a.group===f&&(a.group=y++),s[a.group]=new q(a));return s[a.group]};q.prototype={dragInit:function(a,b){this.$document=d(b.el[0].ownerDocument);var c=d(a.target).closest(this.options.itemSelector);c.length&&(this.item=c,this.itemContainer=b,!this.item.is(this.options.exclude)&&this.options.onMousedown(this.item,n.onMousedown,a)&&(this.setPointer(a),this.toggleListeners("on"),this.setupDelayTimer(),
-this.dragInitDone=!0))},drag:function(a){if(!this.dragging){if(!this.distanceMet(a)||!this.delayMet)return;this.options.onDragStart(this.item,this.itemContainer,n.onDragStart,a);this.item.before(this.placeholder);this.dragging=!0}this.setPointer(a);this.options.onDrag(this.item,p(this.pointer,this.item.offsetParent()),n.onDrag,a);a=this.getPointer(a);var b=this.sameResultBox,c=this.options.tolerance;(!b||b.top-c>a.top||b.bottom+ca.left||b.right+c=this.options.distance},getPointer:function(a){var b=
-a.originalEvent||a.originalEvent.touches&&a.originalEvent.touches[0];return{left:a.pageX||b.pageX,top:a.pageY||b.pageY}},setupDelayTimer:function(){var a=this;this.delayMet=!this.options.delay;this.delayMet||(clearTimeout(this._mouseDelayTimer),this._mouseDelayTimer=setTimeout(function(){a.delayMet=!0},this.options.delay))},scroll:function(a){this.clearDimensions();this.clearOffsetParent()},toggleListeners:function(a){var b=this;d.each(["drag","drop","scroll"],function(c,e){b.$document[a](r[e],b[e+
-"Proxy"])})},clearOffsetParent:function(){this.offsetParent=f},clearDimensions:function(){this.traverse(function(a){a._clearDimensions()})},traverse:function(a){a(this);for(var b=this.containers.length;b--;)this.containers[b].traverse(a)},_clearDimensions:function(){this.containerDimensions=f},_destroy:function(){s[this.options.group]=f}};t.prototype={dragInit:function(a){var b=this.rootGroup;!this.disabled&&!b.dragInitDone&&this.options.drag&&this.isValidDrag(a)&&b.dragInit(a,this)},isValidDrag:function(a){return 1==
-a.which||"touchstart"==a.type&&1==a.originalEvent.touches.length},searchValidTarget:function(a,b){var c=x(this.getItemDimensions(),a,b),e=c.length,d=this.rootGroup,g=!d.options.isValidTarget||d.options.isValidTarget(d.item,this);if(!e&&g)return d.movePlaceholder(this,this.target,"append"),!0;for(;e--;)if(d=c[e][0],!c[e][1]&&this.hasChildGroup(d)){if(this.getContainerGroup(d).searchValidTarget(a,b))return!0}else if(g)return this.movePlaceholder(d,a),!0},movePlaceholder:function(a,b){var c=d(this.items[a]),
-e=this.itemDimensions[a],k="after",g=c.outerWidth(),f=c.outerHeight(),h=c.offset(),h={left:h.left,right:h.left+g,top:h.top,bottom:h.top+f};this.options.vertical?b.top<=(e[2]+e[3])/2?(k="before",h.bottom-=f/2):h.top+=f/2:b.left<=(e[0]+e[1])/2?(k="before",h.right-=g/2):h.left+=g/2;this.hasChildGroup(a)&&(h=A);this.rootGroup.movePlaceholder(this,c,k,h)},getItemDimensions:function(){this.itemDimensions||(this.items=this.$getChildren(this.el,"item").filter(":not(."+this.group.options.placeholderClass+
-", ."+this.group.options.draggedClass+")").get(),w(this.items,this.itemDimensions=[],this.options.tolerance));return this.itemDimensions},getItemOffsetParent:function(){var a=this.el;return"relative"===a.css("position")||"absolute"===a.css("position")||"fixed"===a.css("position")?a:a.offsetParent()},hasChildGroup:function(a){return this.options.nested&&this.getContainerGroup(a)},getContainerGroup:function(a){var b=d.data(this.items[a],"subContainers");if(b===f){var c=this.$getChildren(this.items[a],
-"container"),b=!1;c[0]&&(b=d.extend({},this.options,{rootGroup:this.rootGroup,group:y++}),b=c[m](b).data(m).group);d.data(this.items[a],"subContainers",b)}return b},$getChildren:function(a,b){var c=this.rootGroup.options,e=c[b+"Path"],c=c[b+"Selector"];a=d(a);e&&(a=a.find(e));return a.children(c)},_serialize:function(a,b){var c=this,e=this.$getChildren(a,b?"item":"container").not(this.options.exclude).map(function(){return c._serialize(d(this),!b)}).get();return this.rootGroup.options.serialize(a,
-e,b)},traverse:function(a){d.each(this.items||[],function(b){(b=d.data(this,"subContainers"))&&b.traverse(a)});a(this)},_clearDimensions:function(){this.itemDimensions=f},_destroy:function(){var a=this;this.target.off(r.start,this.handle);this.el.removeData(m);this.options.drop&&(this.group.containers=d.grep(this.group.containers,function(b){return b!=a}));d.each(this.items||[],function(){d.removeData(this,"subContainers")})}};var u={enable:function(){this.traverse(function(a){a.disabled=!1})},disable:function(){this.traverse(function(a){a.disabled=
-!0})},serialize:function(){return this._serialize(this.el,!0)},refresh:function(){this.traverse(function(a){a._clearDimensions()})},destroy:function(){this.traverse(function(a){a._destroy()})}};d.extend(t.prototype,u);d.fn[m]=function(a){var b=Array.prototype.slice.call(arguments,1);return this.map(function(){var c=d(this),e=c.data(m);if(e&&u[a])return u[a].apply(e,b)||this;e||a!==f&&"object"!==typeof a||c.data(m,new t(c,a));return this})}}(jQuery,window,"sortable");
diff --git a/readme.md b/readme.md
index 940deb04c..012a6bda4 100644
--- a/readme.md
+++ b/readme.md
@@ -142,7 +142,7 @@ These are the great open-source projects used to help build BookStack:
* [CodeMirror](https://codemirror.net)
* [Vue.js](http://vuejs.org/)
* [Axios](https://github.com/mzabriskie/axios)
-* [jQuery Sortable](https://johnny.github.io/jquery-sortable/)
+* [Sortable](https://github.com/SortableJS/Sortable) & [Vue.Draggable](https://github.com/SortableJS/Vue.Draggable)
* [Google Material Icons](https://material.io/icons/)
* [Dropzone.js](http://www.dropzonejs.com/)
* [clipboard.js](https://clipboardjs.com/)
diff --git a/resources/assets/js/components/book-sort.js b/resources/assets/js/components/book-sort.js
new file mode 100644
index 000000000..da2b28d8e
--- /dev/null
+++ b/resources/assets/js/components/book-sort.js
@@ -0,0 +1,204 @@
+import Sortable from "sortablejs";
+
+// Auto sort control
+const sortOperations = {
+ name: function(a, b) {
+ const aName = a.getAttribute('data-name').trim().toLowerCase();
+ const bName = b.getAttribute('data-name').trim().toLowerCase();
+ return aName.localeCompare(bName);
+ },
+ created: function(a, b) {
+ const aTime = Number(a.getAttribute('data-created'));
+ const bTime = Number(b.getAttribute('data-created'));
+ return bTime - aTime;
+ },
+ updated: function(a, b) {
+ const aTime = Number(a.getAttribute('data-updated'));
+ const bTime = Number(b.getAttribute('data-updated'));
+ return bTime - aTime;
+ },
+ chaptersFirst: function(a, b) {
+ const aType = a.getAttribute('data-type');
+ const bType = b.getAttribute('data-type');
+ if (aType === bType) {
+ return 0;
+ }
+ return (aType === 'chapter' ? -1 : 1);
+ },
+ chaptersLast: function(a, b) {
+ const aType = a.getAttribute('data-type');
+ const bType = b.getAttribute('data-type');
+ if (aType === bType) {
+ return 0;
+ }
+ return (aType === 'chapter' ? 1 : -1);
+ },
+};
+
+class BookSort {
+
+ constructor(elem) {
+ this.elem = elem;
+ this.sortContainer = elem.querySelector('[book-sort-boxes]');
+ this.input = elem.querySelector('[book-sort-input]');
+
+ const initialSortBox = elem.querySelector('.sort-box');
+ this.setupBookSortable(initialSortBox);
+ this.setupSortPresets();
+
+ window.$events.listen('entity-select-confirm', this.bookSelect.bind(this));
+ }
+
+ /**
+ * Setup the handlers for the preset sort type buttons.
+ */
+ setupSortPresets() {
+ let lastSort = '';
+ let reverse = false;
+ const reversibleTypes = ['name', 'created', 'updated'];
+
+ this.sortContainer.addEventListener('click', event => {
+ const sortButton = event.target.closest('.sort-box-options [data-sort]');
+ if (!sortButton) return;
+
+ event.preventDefault();
+ const sortLists = sortButton.closest('.sort-box').querySelectorAll('ul');
+ const sort = sortButton.getAttribute('data-sort');
+
+ reverse = (lastSort === sort) ? !reverse : false;
+ let sortFunction = sortOperations[sort];
+ if (reverse && reversibleTypes.includes(sort)) {
+ sortFunction = function(a, b) {
+ return 0 - sortOperations[sort](a, b)
+ };
+ }
+
+ for (let list of sortLists) {
+ const directItems = Array.from(list.children).filter(child => child.matches('li'));
+ directItems.sort(sortFunction).forEach(sortedItem => {
+ list.appendChild(sortedItem);
+ });
+ }
+
+ lastSort = sort;
+ this.updateMapInput();
+ });
+ }
+
+ /**
+ * Handle book selection from the entity selector.
+ * @param {Object} entityInfo
+ */
+ bookSelect(entityInfo) {
+ const alreadyAdded = this.elem.querySelector(`[data-type="book"][data-id="${entityInfo.id}"]`) !== null;
+ if (alreadyAdded) return;
+
+ const entitySortItemUrl = entityInfo.link + '/sort-item';
+ window.$http.get(entitySortItemUrl).then(resp => {
+ const wrap = document.createElement('div');
+ wrap.innerHTML = resp.data;
+ const newBookContainer = wrap.children[0];
+ this.sortContainer.append(newBookContainer);
+ this.setupBookSortable(newBookContainer);
+ });
+ }
+
+ /**
+ * Setup the given book container element to have sortable items.
+ * @param {Element} bookContainer
+ */
+ setupBookSortable(bookContainer) {
+ const sortElems = [bookContainer.querySelector('.sort-list')];
+ sortElems.push(...bookContainer.querySelectorAll('.entity-list-item + ul'));
+
+ const bookGroupConfig = {
+ name: 'book',
+ pull: ['book', 'chapter'],
+ put: ['book', 'chapter'],
+ };
+
+ const chapterGroupConfig = {
+ name: 'chapter',
+ pull: ['book', 'chapter'],
+ put: function(toList, fromList, draggedElem) {
+ return draggedElem.getAttribute('data-type') === 'page';
+ }
+ };
+
+ for (let sortElem of sortElems) {
+ new Sortable(sortElem, {
+ group: sortElem.classList.contains('sort-list') ? bookGroupConfig : chapterGroupConfig,
+ animation: 150,
+ fallbackOnBody: true,
+ swapThreshold: 0.65,
+ onSort: this.updateMapInput.bind(this),
+ dragClass: 'bg-white',
+ ghostClass: 'primary-background-light',
+ });
+ }
+ }
+
+ /**
+ * Update the input with our sort data.
+ */
+ updateMapInput() {
+ const pageMap = this.buildEntityMap();
+ this.input.value = JSON.stringify(pageMap);
+ }
+
+ /**
+ * Build up a mapping of entities with their ordering and nesting.
+ * @returns {Array}
+ */
+ buildEntityMap() {
+ const entityMap = [];
+ const lists = this.elem.querySelectorAll('.sort-list');
+
+ for (let list of lists) {
+ const bookId = list.closest('[data-type="book"]').getAttribute('data-id');
+ const directChildren = Array.from(list.children)
+ .filter(elem => elem.matches('[data-type="page"], [data-type="chapter"]'));
+ for (let i = 0; i < directChildren.length; i++) {
+ this.addBookChildToMap(directChildren[i], i, bookId, entityMap);
+ }
+ }
+
+ return entityMap;
+ }
+
+ /**
+ * Parse a sort item and add it to a data-map array.
+ * Parses sub0items if existing also.
+ * @param {Element} childElem
+ * @param {Number} index
+ * @param {Number} bookId
+ * @param {Array} entityMap
+ */
+ addBookChildToMap(childElem, index, bookId, entityMap) {
+ const type = childElem.getAttribute('data-type');
+ const parentChapter = false;
+ const childId = childElem.getAttribute('data-id');
+
+ entityMap.push({
+ id: childId,
+ sort: index,
+ parentChapter: parentChapter,
+ type: type,
+ book: bookId
+ });
+
+ const subPages = childElem.querySelectorAll('[data-type="page"]');
+ for (let i = 0; i < subPages.length; i++) {
+ entityMap.push({
+ id: subPages[i].getAttribute('data-id'),
+ sort: i,
+ parentChapter: childId,
+ type: 'page',
+ book: bookId
+ });
+ }
+ }
+
+}
+
+export default BookSort;
\ No newline at end of file
diff --git a/resources/assets/js/components/index.js b/resources/assets/js/components/index.js
index bd7432b8d..b48ceb20d 100644
--- a/resources/assets/js/components/index.js
+++ b/resources/assets/js/components/index.js
@@ -24,6 +24,7 @@ import triLayout from "./tri-layout";
import breadcrumbListing from "./breadcrumb-listing";
import permissionsTable from "./permissions-table";
import customCheckbox from "./custom-checkbox";
+import bookSort from "./book-sort";
const componentMapping = {
'dropdown': dropdown,
@@ -52,6 +53,7 @@ const componentMapping = {
'breadcrumb-listing': breadcrumbListing,
'permissions-table': permissionsTable,
'custom-checkbox': customCheckbox,
+ 'book-sort': bookSort,
};
window.components = {};
diff --git a/resources/assets/sass/_colors.scss b/resources/assets/sass/_colors.scss
index 4dfc9d4c3..8f2de6c82 100644
--- a/resources/assets/sass/_colors.scss
+++ b/resources/assets/sass/_colors.scss
@@ -59,8 +59,11 @@
}
/*
- * Entity background colors
+ * Standard & Entity background colors
*/
+.bg-white {
+ background-color: #FFFFFF;
+}
.bg-book {
background-color: $color-book;
}
diff --git a/resources/views/books/sort.blade.php b/resources/views/books/sort.blade.php
index 23259a593..676e7112e 100644
--- a/resources/views/books/sort.blade.php
+++ b/resources/views/books/sort.blade.php
@@ -16,16 +16,16 @@
-
+
{{ trans('entities.books_sort') }}
-
+
@include('books.sort-box', ['book' => $book, 'bookChildren' => $bookChildren])