204 lines
		
	
	
		
			7.9 KiB
		
	
	
	
		
			PHP
		
	
	
	
			
		
		
	
	
			204 lines
		
	
	
		
			7.9 KiB
		
	
	
	
		
			PHP
		
	
	
	
@extends('simple-layout')
 | 
						|
 | 
						|
@section('body')
 | 
						|
 | 
						|
    <div class="container">
 | 
						|
 | 
						|
        <div class="my-s">
 | 
						|
            @include('partials.breadcrumbs', ['crumbs' => [
 | 
						|
                $book,
 | 
						|
                $book->getUrl('/sort') => [
 | 
						|
                    'text' => trans('entities.books_sort'),
 | 
						|
                    'icon' => 'sort',
 | 
						|
                ]
 | 
						|
            ]])
 | 
						|
        </div>
 | 
						|
 | 
						|
        <div class="grid left-focus gap-xl">
 | 
						|
            <div>
 | 
						|
                <div class="card content-wrap">
 | 
						|
                    <h1 class="list-heading mb-l">{{ trans('entities.books_sort') }}</h1>
 | 
						|
                    <div id="sort-boxes">
 | 
						|
                        @include('books.sort-box', ['book' => $book, 'bookChildren' => $bookChildren])
 | 
						|
                    </div>
 | 
						|
 | 
						|
                    <form action="{{ $book->getUrl('/sort') }}" method="POST">
 | 
						|
                        {!! csrf_field() !!}
 | 
						|
                        <input type="hidden" name="_method" value="PUT">
 | 
						|
                        <input type="hidden" id="sort-tree-input" name="sort-tree">
 | 
						|
                        <div class="list text-right">
 | 
						|
                            <a href="{{ $book->getUrl() }}" class="button outline">{{ trans('common.cancel') }}</a>
 | 
						|
                            <button class="button primary" type="submit">{{ trans('entities.books_sort_save') }}</button>
 | 
						|
                        </div>
 | 
						|
                    </form>
 | 
						|
                </div>
 | 
						|
            </div>
 | 
						|
 | 
						|
            <div>
 | 
						|
                <div class="card content-wrap">
 | 
						|
                    <h2 class="list-heading mb-m">{{ trans('entities.books_sort_show_other') }}</h2>
 | 
						|
 | 
						|
                    @include('components.entity-selector', ['name' => 'books_list', 'selectorSize' => 'compact', 'entityTypes' => 'book', 'entityPermission' => 'update', 'showAdd' => true])
 | 
						|
 | 
						|
                </div>
 | 
						|
            </div>
 | 
						|
        </div>
 | 
						|
 | 
						|
    </div>
 | 
						|
 | 
						|
@stop
 | 
						|
 | 
						|
@section('scripts')
 | 
						|
    <script src="{{ baseUrl("/libs/jquery-sortable/jquery-sortable.min.js") }}"></script>
 | 
						|
    <script>
 | 
						|
        $(document).ready(function() {
 | 
						|
 | 
						|
            const $container = $('#sort-boxes');
 | 
						|
 | 
						|
            // Sortable options
 | 
						|
            const sortableOptions = {
 | 
						|
                group: 'serialization',
 | 
						|
                containerSelector: 'ul',
 | 
						|
                itemPath: '',
 | 
						|
                itemSelector: 'li',
 | 
						|
                onDrop: function ($item, container, _super) {
 | 
						|
                    updateMapInput();
 | 
						|
                    _super($item, container);
 | 
						|
                },
 | 
						|
                isValidTarget: function ($item, container) {
 | 
						|
                    // Prevent nested chapters
 | 
						|
                    return !($item.is('[data-type="chapter"]') && container.target.closest('li').attr('data-type') === 'chapter');
 | 
						|
                }
 | 
						|
            };
 | 
						|
 | 
						|
            // Create our sortable group
 | 
						|
            let group = $('.sort-list').sortable(sortableOptions);
 | 
						|
 | 
						|
            // Add book on selection confirm
 | 
						|
            window.$events.listen('entity-select-confirm', function(entityInfo) {
 | 
						|
                const alreadyAdded = $container.find(`[data-type="book"][data-id="${entityInfo.id}"]`).length > 0;
 | 
						|
                if (alreadyAdded) return;
 | 
						|
 | 
						|
                const entitySortItemUrl = entityInfo.link + '/sort-item';
 | 
						|
                window.$http.get(entitySortItemUrl).then(resp => {
 | 
						|
                    $container.append(resp.data);
 | 
						|
                    group.sortable("destroy");
 | 
						|
                    group = $('.sort-list').sortable(sortableOptions);
 | 
						|
                });
 | 
						|
            });
 | 
						|
 | 
						|
            /**
 | 
						|
             * 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.
 | 
						|
             * @returns {Array}
 | 
						|
             */
 | 
						|
            function buildEntityMap() {
 | 
						|
                const entityMap = [];
 | 
						|
                const $lists = $('.sort-list');
 | 
						|
                $lists.each(function(listIndex) {
 | 
						|
                    const $list = $(this);
 | 
						|
                    const bookId = $list.closest('[data-type="book"]').attr('data-id');
 | 
						|
                    const $directChildren = $list.find('> [data-type="page"], > [data-type="chapter"]');
 | 
						|
                    $directChildren.each(function(directChildIndex) {
 | 
						|
                        const $childElem = $(this);
 | 
						|
                        const type = $childElem.attr('data-type');
 | 
						|
                        const parentChapter = false;
 | 
						|
                        const childId = $childElem.attr('data-id');
 | 
						|
 | 
						|
                        entityMap.push({
 | 
						|
                            id: childId,
 | 
						|
                            sort: directChildIndex,
 | 
						|
                            parentChapter: parentChapter,
 | 
						|
                            type: type,
 | 
						|
                            book: bookId
 | 
						|
                        });
 | 
						|
 | 
						|
                        $childElem.find('[data-type="page"]').each(function(pageIndex) {
 | 
						|
                            const $chapterChild = $(this);
 | 
						|
                            entityMap.push({
 | 
						|
                                id: $chapterChild.attr('data-id'),
 | 
						|
                                sort: pageIndex,
 | 
						|
                                parentChapter: childId,
 | 
						|
                                type: 'page',
 | 
						|
                                book: bookId
 | 
						|
                            });
 | 
						|
                        });
 | 
						|
 | 
						|
                    });
 | 
						|
                });
 | 
						|
                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 reversibleTypes = ['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 && reversibleTypes.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>
 | 
						|
@stop
 |