# Sanity check if !scrivito? console.error "'scrivito' is undefined, the content browser will not be available" $ -> if !scrivito? console.log "possible reason: '//= require scrivito_sdk' is missing" else console.log "possible reason: 'scrivito_sdk'/'scrivito_body_tags' is included after 'scrivito_content_browser'" return scrivito.content_browser = do -> thumbnailViewButtonSelector: '.editing-button-view' options: {} filters: {} filter_defaults: upload: {} # The content browser supports operations like +delete+ and +edit+ of resources that require a page # reload, so that all references of the resources get updated. reload: false _getBatchSize: -> @batchSize ||= 20 _getThumbnailSize: -> @thumbnailSize ||= 'normal' _setThumbnailSize: (value) -> @thumbnailSize = value _appendFilterItemTo: (list, options) -> title = options.title || 'missing title' icon = options.icon || 'scrivito-content-browser-icon-generic' query = options.query classname = options.classname || 'filter' icon = icon.replace 'editing-icon', 'scrivito-content-browser-icon' # backwards compatibility @_filterTemplate(title, icon, query) .addClass(classname) .appendTo(list) _appendSeparatorTo: (list) -> $('
  • ') .addClass('separator') .appendTo(list) _filterListTemplate: (filters) -> list = $('') .addClass('scrivito-content-browser-filter-items') @_appendFilterItemTo list, title: 'Selected ', icon: 'scrivito-content-browser-icon-ok', classname: 'selected-filter' @_appendSeparatorTo list for name, options of filters @_appendFilterItemTo list, options list _loadFilter: () -> filters = @_selectedFilters() wrapper = @modal.find('.scrivito-content-browser-filter') @_filterListTemplate(filters) .appendTo(wrapper) _activateInitialFilter: () -> @_defaultFilter().trigger('click') _selectedFilters: -> availableFilters = @filters selectedFilters = @options.filters filters = availableFilters if selectedFilters? unless $.isArray(selectedFilters) selectedFilters = new Array(selectedFilters) filters = {} for filterId, filter of availableFilters if filterId in selectedFilters filters[filterId] = filter filters _defaultFilter: -> @_filterItems().first() _getFilterQuery: (filter) -> filter.data('query') _defaultQuery: -> @_getFilterQuery(@_defaultFilter()) _activeQuery: -> filter = @_filterItems().filter('.active') @_getFilterQuery(filter) _filterItems: -> @modal.find('ul.scrivito-content-browser-filter-items li') _deactivateAllFilter: -> @_getSearch().val('') @_filterItems().removeClass('active') _triggerFilter: (filter) -> @_deactivateAllFilter() filter.addClass('active') query = @_getFilterQuery(filter) @_renderPlaceholder(query) _triggerSelectedFilter: (filter) -> if @selected.length > 0 query = @_prepareQuery(scrivito.obj_where('id', 'equals', @selected)) filter.data('query', query) else filter.removeData('query') @_triggerFilter(filter) _filterTemplate: (title, icon, query) -> filter = $('
  • ') if query? query = @_prepareQuery(query) filter.data('query', query) icon = $('') .addClass('scrivito-content-browser-icon') .addClass(icon) .appendTo(filter) title = $('') .addClass('scrivito-content-browser-filter-name') .html(title) .appendTo(filter) filter _prepareQuery: (query) -> query.clone() .order('_last_changed') .reverse_order() .format('content_browser') _save: -> @_promise.resolve(@selected) @close() _delete: -> $.each @selected, (index, id) => item = @_getItemContainer(id) @_itemLoading(item) scrivito.delete_obj(id).then => @modal.trigger('resource_change.content_browser') item.remove() @_deselectAllItems() _getItemId: (item) -> $(item).closest('li.content-browser-item').data('id') _getItemContainer: (id) -> $('li.content-browser-item').filter -> id == $(this).data('id') _selectItem: (item) -> if (item.hasClass('active')) @_removeItem(item) else if @options.selectionMode == 'single' @_deselectAllItems() @_addItem(item) _addItem: (item) -> @_activateItem(item) id = @_getItemId(item) @selected.push(id) @_setSelected($.unique(@selected)) _removeItem: (item) -> @_deactivateItem(item) selected = @selected.filter (id) => id != @_getItemId(item) @_setSelected(selected) _activateItem: (item) -> $(item).addClass('active') _deactivateItem: (item) -> $(item).removeClass('active') _changeSelectedTotal: -> @modal.find('.selected-total').html(@selected.length) _setSelected: (value) -> @selected = value || @options.selection || [] @_changeSelectedTotal() _deselectAllItems: -> @_setSelected([]) @modal.find('li.content-browser-item .select-item.active').removeClass('active') _getItems: -> @modal.find('.scrivito-content-browser-items') _getContainer: -> @modal.find('.scrivito-content-browser-thumbnails') _itemsPlaceholder: (count) -> size = @_getThumbnailSize() list = $('') .addClass('items scrivito-content-browser-thumbnails') .addClass(size) content = for index in [0...count] by 1 itemTemplate = @_itemPlaceholderTemplate(index) list.append(itemTemplate) @_getItems().html(content) _itemPlaceholderTemplate: (index) -> item = $('
  • ') .addClass('content-browser-item') .attr('data-index', index) @_itemLoading(item) _itemLoading: (item) -> loading = @_loadingTemplate() $(item).html(loading) _renderPlaceholder: (query) -> @_getItems() .empty() if query? @_getItems().html(@_loadingTemplate()) query .batch_size(@_getBatchSize()) query.size() .done (total) => if total > 0 @_itemsPlaceholder(total) @_renderItems(query) else @_getItems().empty() _renderItems: (query, index = 0) -> query.load_batch() .done (result, next) => objects = result.hits @_replacePlaceholder(objects, index) if next? start = index + objects.length @_renderItems(next, start) _replacePlaceholder: (objects, startIndex) -> $(objects).each (index, object) => elementIndex = startIndex + index template = @_itemTemplate(object) @modal.find("li.content-browser-item[data-index=#{elementIndex}]") .html(template) .data('id', object.id) _itemTemplate: (object) -> url = object.preview title = object.title || object.id id = object.id wrapper = $('
    ') .addClass('scrivito-content-browser-item-wrapper') preview = $('
    ') .addClass('scrivito-content-browser-preview') .appendTo(wrapper) inspect = $('') .addClass('scrivito-content-browser-inspect') .appendTo(wrapper) image = if url? $('') .attr('src', url) else $('') .addClass('scrivito-content-browser-icon') .addClass('scrivito-content-browser-icon-generic') image.appendTo(preview) meta = $('
    ') .addClass('scrivito-content-browser-meta') .appendTo(wrapper) title = $('') .addClass('scrivito-content-browser-thumbnails-name') .html(title) .appendTo(meta) select = $('') .addClass('scrivito-content-browser-thumbnails-select select-item') .appendTo(title) if id in @selected select.addClass('active') wrapper _getSearch: -> @modal.find('input.search-field') _triggerSearch: -> term = @_getSearch().val() query = @_prepareQuery(@_activeQuery()) if term? && term.length > 0 query.and('*', 'contains_prefix', term) @_renderPlaceholder(query) _initializeBindings: -> $(window).resize => @_center(@modal) @modal.on 'keyup', 'input.search-field', (event) => if event.keyCode == 13 @_triggerSearch() @modal.on 'click', 'button.search-field-button', (event) => event.preventDefault() @_triggerSearch() @modal.on 'click', 'li.content-browser-item', (event) => unless $(event.target).hasClass('scrivito-content-browser-inspect') item = $(event.currentTarget).find('.select-item') @_selectItem(item) @modal.on 'click', '.content-browser-save', (event) => event.preventDefault() @_save() @modal.on 'click', '.content-browser-delete', (event) => event.preventDefault() @_delete() @modal.on 'click', '.content-browser-close', (event) => event.preventDefault() @close() @modal.on 'click', 'li.filter', (event) => event.preventDefault() @_triggerFilter($(event.currentTarget)) @modal.on 'click', 'li.selected-filter', (event) => event.preventDefault() @_triggerSelectedFilter($(event.currentTarget)) @modal.on 'click', @thumbnailViewButtonSelector, (event) => size = $(event.currentTarget).data('size') @_changeThumbnailSize(size) @modal.on 'resource_change.content_browser', (event) => @reload = true $(document).on 'keyup.content_browser', (event) => event.stopImmediatePropagation() if event.keyCode == 27 # Make sure to remove the event handler after # +stopImmediatePropagation()+, otherwise all ESC keys are caught and # not propagated. $(document).off 'keyup.content_browser' @close() _loadModal: () -> @overlay = $('
    ') .addClass('editing-overlay show') .appendTo($('body')) innerHtml = """
    """ @modal = $('
    ') .addClass('scrivito-content-browser show') .attr('id', 'scrivito-content-browser') .html(innerHtml) .appendTo($('body')) @_center(@modal) @_initializeBindings() _center: (domElement) -> if domElement domElement = $(domElement) domElement.css( marginLeft: -domElement.innerWidth() / 2 marginTop: -domElement.innerHeight() / 2 left: '50%' ) _loadingTemplate: -> icon = $('') .addClass('scrivito-content-browser-icon scrivito-content-browser-icon-refresh') $('
    ') .addClass('scrivito-content-browser-loading') .html(icon) _changeThumbnailSize: (size) -> size ||= @_getThumbnailSize() @_setThumbnailSize(size) transitionListener = 'webkitTransitionEnd.content_browser otransitionend.content_browser oTransitionEnd.content_browser msTransitionEnd.content_browser transitionend.content_browser' @modal.on transitionListener, 'li.content-browser-item', (event) => @modal.off transitionListener @_getContainer() .removeClass('small normal big large normal-list') .addClass(size) @modal.find(@thumbnailViewButtonSelector) .removeClass('active') .filter("[data-size='#{size}']") .addClass('active') close: -> @_promise.reject() if @_promise.state() == 'pending' @overlay.remove() @modal.remove() if @reload $('body').trigger('infopark_reload') open: (options) -> @options = options @_loadModal() @_loadFilter() @_setSelected() @_renderPlaceholder(@_defaultQuery()) @_changeThumbnailSize() @_activateInitialFilter() @_inspector.init(@modal) @_uploader.init(@modal) @_uploader.onUploadStart = (files) => @upload_errors = false @_uploader.onUploadFailure = (errors) => @upload_errors = errors @_uploader.onUploadSuccess = (objs) => @_renderPlaceholder(@_activeQuery()) unless @upload_errors @_promise = $.Deferred() return @_promise