Added book sort helper buttons

This commit is contained in:
Dan Brown 2019-02-17 11:44:02 +00:00
parent f1e571a57c
commit 8445304fe9
No known key found for this signature in database
GPG Key ID: 46D9F943C24A2EF9
5 changed files with 181 additions and 40 deletions

View File

@ -145,26 +145,41 @@
// Sortable Lists // Sortable Lists
.sortable-page-list, .sortable-page-list ul { .sortable-page-list, .sortable-page-list ul {
list-style: none; list-style: none;
background-color: #FFF;
} }
.sort-box { .sort-box {
margin-bottom: $-m; margin-bottom: $-m;
padding: 0 $-l 0 $-l; border: 2px solid rgba($color-book, 0.6);
border-left: 4px solid $color-book; padding: $-m $-xl;
border-radius: 4px;
}
.sort-box-options {
display: flex;
flex-wrap: wrap;
justify-content: space-between;
}
.sort-box-options .button {
margin-left: 0;
} }
.sortable-page-list { .sortable-page-list {
margin-left: 0; margin-left: 0;
padding: 0;
.entity-list-item > span:first-child {
align-self: flex-start;
}
.entity-list-item > div {
display: block;
flex: 1;
}
> ul { > ul {
margin-left: 0; margin-left: 0;
} }
ul { ul {
margin-bottom: 0; margin-bottom: $-m;
margin-top: 0; margin-top: 0;
box-shadow: 0 0 1px 0 rgba(0, 0, 0, 0.1); padding-left: $-m;
} }
li { li {
border: 1px solid #DDD; border: 1px solid #DDD;
padding: $-xs $-s;
margin-top: -1px; margin-top: -1px;
min-height: 38px; min-height: 38px;
&.text-chapter { &.text-chapter {
@ -278,7 +293,6 @@ ul.pagination {
align-items: center; align-items: center;
background-color: transparent; background-color: transparent;
border: 0; border: 0;
cursor: pointer;
width: 100%; width: 100%;
position: relative; position: relative;
h4 a { h4 a {
@ -293,14 +307,18 @@ ul.pagination {
flex: 1; flex: 1;
text-align: left; text-align: left;
} }
&:hover { &:not(.no-hover) {
cursor: pointer;
}
&:not(.no-hover):hover {
text-decoration: none; text-decoration: none;
background-color: #DDD; background-color: #DDD;
border-radius: 4px; border-radius: 4px;
} }
} }
.card .entity-list-item:hover {
.card .entity-list-item:not(.no-hover):hover {
background-color: #F2F2F2; background-color: #F2F2F2;
} }
.card .entity-list-item .entity-list-item:hover { .card .entity-list-item .entity-list-item:hover {

View File

@ -125,6 +125,11 @@ return [
'books_navigation' => 'Book Navigation', 'books_navigation' => 'Book Navigation',
'books_sort' => 'Sort Book Contents', 'books_sort' => 'Sort Book Contents',
'books_sort_named' => 'Sort Book :bookName', 'books_sort_named' => 'Sort Book :bookName',
'books_sort_name' => 'Sort by Name',
'books_sort_created' => 'Sort by Created Date',
'books_sort_updated' => 'Sort by Updated Date',
'books_sort_chapters_first' => 'Chapters First',
'books_sort_chapters_last' => 'Chapters Last',
'books_sort_show_other' => 'Show Other Books', 'books_sort_show_other' => 'Show Other Books',
'books_sort_save' => 'Save New Order', 'books_sort_save' => 'Save New Order',

View File

@ -1,20 +1,48 @@
<div class="sort-box" data-type="book" data-id="{{ $book->id }}"> <div class="sort-box" data-type="book" data-id="{{ $book->id }}">
<h3 class="text-book">@icon('book'){{ $book->name }}</h3> <h5 class="text-book entity-list-item no-hover py-xs pl-none">
<span>@icon('book')</span>
<span>{{ $book->name }}</span>
</h5>
<div class="sort-box-options pb-sm">
<a href="#" data-sort="name" class="button outline small">{{ trans('entities.books_sort_name') }}</a>
<a href="#" data-sort="created" class="button outline small">{{ trans('entities.books_sort_created') }}</a>
<a href="#" data-sort="updated" class="button outline small">{{ trans('entities.books_sort_updated') }}</a>
<a href="#" data-sort="chaptersFirst" class="button outline small">{{ trans('entities.books_sort_chapters_first') }}</a>
<a href="#" data-sort="chaptersLast" class="button outline small">{{ trans('entities.books_sort_chapters_last') }}</a>
</div>
<ul class="sortable-page-list sort-list"> <ul class="sortable-page-list sort-list">
@foreach($bookChildren as $bookChild) @foreach($bookChildren as $bookChild)
<li data-id="{{$bookChild->id}}" data-type="{{ $bookChild->getClassName() }}" class="text-{{ $bookChild->getClassName() }}"> <li class="text-{{ $bookChild->getClassName() }}"
@icon($bookChild->isA('chapter') ? 'chapter' : 'page'){{ $bookChild->name }} data-id="{{$bookChild->id}}" data-type="{{ $bookChild->getClassName() }}"
data-name="{{ $bookChild->name }}" data-created="{{ $bookChild->created_at->timestamp }}"
data-updated="{{ $bookChild->updated_at->timestamp }}">
<div class="entity-list-item">
<span>@icon($bookChild->getType()) </span>
<div>
{{ $bookChild->name }}
<div>
</div>
</div>
</div>
@if($bookChild->isA('chapter')) @if($bookChild->isA('chapter'))
<ul> <ul>
@foreach($bookChild->pages as $page) @foreach($bookChild->pages as $page)
<li data-id="{{$page->id}}" class="text-page" data-type="page"> <li class="text-page"
@icon('page') data-id="{{$page->id}}" data-type="page"
{{ $page->name }} data-name="{{ $page->name }}" data-created="{{ $page->created_at->timestamp }}"
data-updated="{{ $page->updated_at->timestamp }}">
<div class="entity-list-item">
<span>@icon('page')</span>
<span>{{ $page->name }}</span>
</div>
</li> </li>
@endforeach @endforeach
</ul> </ul>
@endif @endif
</li> </li>
@endforeach @endforeach
</ul> </ul>
</div> </div>

View File

@ -1,5 +1,7 @@
@extends('simple-layout') @extends('simple-layout')
{{--TODO - Load books in via selector interface--}}
@section('body') @section('body')
<div class="container"> <div class="container">
@ -7,7 +9,10 @@
<div class="my-l"> <div class="my-l">
@include('partials.breadcrumbs', ['crumbs' => [ @include('partials.breadcrumbs', ['crumbs' => [
$book, $book,
$book->getUrl('/sort') => trans('entities.books_sort') $book->getUrl('/sort') => [
'text' => trans('entities.books_sort'),
'icon' => 'sort',
]
]]) ]])
</div> </div>
@ -16,7 +21,7 @@
<div class="card content-wrap"> <div class="card content-wrap">
<h1 class="list-heading">{{ trans('entities.books_sort') }}</h1> <h1 class="list-heading">{{ trans('entities.books_sort') }}</h1>
<div id="sort-boxes"> <div id="sort-boxes">
@include('books/sort-box', ['book' => $book, 'bookChildren' => $bookChildren]) @include('books.sort-box', ['book' => $book, 'bookChildren' => $bookChildren])
</div> </div>
<form action="{{ $book->getUrl('/sort') }}" method="POST"> <form action="{{ $book->getUrl('/sort') }}" method="POST">
@ -58,49 +63,66 @@
<script> <script>
$(document).ready(function() { $(document).ready(function() {
var sortableOptions = { const $container = $('#sort-boxes');
// Sortable options
const sortableOptions = {
group: 'serialization', group: 'serialization',
onDrop: function($item, container, _super) { containerSelector: 'ul',
var pageMap = buildEntityMap(); itemPath: '',
$('#sort-tree-input').val(JSON.stringify(pageMap)); itemSelector: 'li',
onDrop: function ($item, container, _super) {
updateMapInput();
_super($item, container); _super($item, container);
}, },
isValidTarget: function ($item, container) { isValidTarget: function ($item, container) {
// Prevent nested chapters // Prevent nested chapters
return !($item.is('[data-type="chapter"]') && container.target.closest('li').attr('data-type') == 'chapter'); return !($item.is('[data-type="chapter"]') && container.target.closest('li').attr('data-type') === 'chapter');
} }
}; };
var group = $('.sort-list').sortable(sortableOptions); // Create our sortable group
let group = $('.sort-list').sortable(sortableOptions);
// Add additional books into the view on select.
$('#additional-books').on('click', 'a', function(e) { $('#additional-books').on('click', 'a', function(e) {
e.preventDefault(); e.preventDefault();
var $link = $(this);
var url = $link.attr('href'); const $link = $(this);
const url = $link.attr('href');
$.get(url, function(data) { $.get(url, function(data) {
$('#sort-boxes').append(data); $container.append(data);
group.sortable("destroy"); group.sortable("destroy");
$('.sort-list').sortable(sortableOptions); group = $('.sort-list').sortable(sortableOptions);
}); });
$link.remove(); $link.remove();
}); });
/**
* Update the input with our sort data.
*/
function updateMapInput() {
const pageMap = buildEntityMap();
$('#sort-tree-input').val(JSON.stringify(pageMap));
}
/** /**
* Build up a mapping of entities with their ordering and nesting. * Build up a mapping of entities with their ordering and nesting.
* @returns {Array} * @returns {Array}
*/ */
function buildEntityMap() { function buildEntityMap() {
var entityMap = []; const entityMap = [];
var $lists = $('.sort-list'); const $lists = $('.sort-list');
$lists.each(function(listIndex) { $lists.each(function(listIndex) {
var list = $(this); const $list = $(this);
var bookId = list.closest('[data-type="book"]').attr('data-id'); const bookId = $list.closest('[data-type="book"]').attr('data-id');
var $directChildren = list.find('> [data-type="page"], > [data-type="chapter"]'); const $directChildren = $list.find('> [data-type="page"], > [data-type="chapter"]');
$directChildren.each(function(directChildIndex) { $directChildren.each(function(directChildIndex) {
var $childElem = $(this); const $childElem = $(this);
var type = $childElem.attr('data-type'); const type = $childElem.attr('data-type');
var parentChapter = false; const parentChapter = false;
var childId = $childElem.attr('data-id'); const childId = $childElem.attr('data-id');
entityMap.push({ entityMap.push({
id: childId, id: childId,
sort: directChildIndex, sort: directChildIndex,
@ -108,8 +130,9 @@
type: type, type: type,
book: bookId book: bookId
}); });
$chapterChildren = $childElem.find('[data-type="page"]').each(function(pageIndex) {
var $chapterChild = $(this); $childElem.find('[data-type="page"]').each(function(pageIndex) {
const $chapterChild = $(this);
entityMap.push({ entityMap.push({
id: $chapterChild.attr('data-id'), id: $chapterChild.attr('data-id'),
sort: pageIndex, sort: pageIndex,
@ -118,11 +141,74 @@
book: bookId book: bookId
}); });
}); });
}); });
}); });
return entityMap; return entityMap;
} }
// 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-update'));
const bTime = Number(b.getAttribute('data-update'));
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);
},
};
let lastSort = '';
let reverse = false;
const reversableTypes = ['name', 'created', 'updated'];
$container.on('click', '.sort-box-options [data-sort]', function(event) {
event.preventDefault();
const $sortLists = $(this).closest('.sort-box').find('ul');
const sort = $(this).attr('data-sort');
reverse = (lastSort === sort) ? !reverse : false;
let sortFunction = sortOperations[sort];
if (reverse && reversableTypes.includes(sort)) {
sortFunction = function(a, b) {
return 0 - sortOperations[sort](a, b)
};
}
$sortLists.each(function() {
const $list = $(this);
$list.children('li').sort(sortFunction).appendTo($list);
});
lastSort = sort;
updateMapInput();
});
}); });
</script> </script>
@stop @stop

View File

@ -12,7 +12,11 @@
<a href="{{ baseUrl($key) }}"> <a href="{{ baseUrl($key) }}">
{{ $crumb }} {{ $crumb }}
</a> </a>
@else @elseif (is_array($crumb))
<a href="{{ baseUrl($key) }}">
@icon($crumb['icon']) {{ $crumb['text'] }}
</a>
@elseif($crumb instanceof \BookStack\Entities\Entity)
<a href="{{ $crumb->getUrl() }}" class="text-{{$crumb->getType()}}"> <a href="{{ $crumb->getUrl() }}" class="text-{{$crumb->getType()}}">
@icon($crumb->getType()){{ $crumb->getShortName() }} @icon($crumb->getType()){{ $crumb->getShortName() }}
</a> </a>