Add new list templates toolbar

Add the new generic list and generic list items toolbar which allow
switching the list display mode.

This toolbar also includes a stylized button to select and deselect
all items emulating the check-all checkbox.

Signed-off-by: Roberto Rosario <Roberto.Rosario@mayan-edms.com>
This commit is contained in:
Roberto Rosario
2018-12-21 23:51:20 -04:00
parent 9784798118
commit 14f31d5614
9 changed files with 129 additions and 42 deletions

View File

@@ -301,9 +301,23 @@ class MayanApp {
app.lastChecked = null;
$('body').on('click', '.check-all', function (event) {
var $this = $(this);
var checked = $(event.target).prop('checked');
var $checkBoxes = $('.check-all-slave');
if (checked === undefined) {
checked = $this.data('checked');
checked = !checked;
$this.data('checked', checked);
console.log($this.data('icon-checked'));
if (checked) {
$this.find('[data-fa-i2svg]').addClass($this.data('icon-checked')).removeClass($this.data('icon-unchecked'));
} else {
$this.find('[data-fa-i2svg]').addClass($this.data('icon-unchecked')).removeClass($this.data('icon-checked'));
}
}
$checkBoxes.prop('checked', checked);
$checkBoxes.trigger('change');
});
@@ -353,6 +367,7 @@ class MayanApp {
setupPanelSelection () {
var app = this;
// Setup panel highlighting on check
$('body').on('change', '.check-all-slave', function (event) {
var checked = $(event.target).prop('checked');
if (checked) {
@@ -366,9 +381,10 @@ class MayanApp {
var $this = $(this);
var targetSrc = $(event.target).prop('src');
var targetHref = $(event.target).prop('href');
var targetIsButton = event.target.tagName === 'BUTTON';
var lastChecked = null;
if ((targetSrc === undefined) && (targetHref === undefined)) {
if ((targetSrc === undefined) && (targetHref === undefined) && (targetIsButton === false)) {
var $checkbox = $this.find('.check-all-slave');
var checked = $checkbox.prop('checked');

View File

@@ -11,7 +11,6 @@
{% include 'appearance/no_results.html' %}
</div>
{% else %}
<h4>
{% if page_obj %}
{% if page_obj.paginator.num_pages != 1 %}
@@ -24,17 +23,23 @@
{% endif %}
</h4>
<hr>
<div class="well center-block">
<div class="clearfix">
{% if object_list %}
{% if not hide_multi_item_actions %}
{% get_multi_item_links_form object_list %}
{% endif %}
{% endif %}
{% include 'appearance/list_toolbar.html' %}
<div class="pull-right">
<form action="{% url 'common:multi_object_action_view' %}" class="form-multi-object-action" method="get">
{% if object_list %}
{% if not hide_multi_item_actions %}
{% get_multi_item_links_form object_list %}
{% endif %}
{% if multi_item_actions %}
<fieldset style="margin-top: -10px;">
<input class="check-all" type="checkbox"/>&nbsp;
{{ multi_item_form }}
</fieldset>
{% endif %}
@@ -65,15 +70,14 @@
<span style="color: white; word-break: break-all; overflow-wrap: break-word;">
{% if not hide_object %}
{% if main_object %}
{% with object|object_property:main_object as object %}
{% if not hide_link %}<a href="{{ object.get_absolute_url }}">{{ object }}</a>{% else %}{{ object }}{% endif %}
{% endwith %}
{% if not hide_link %}
<a href="{{ object.get_absolute_url }}">{{ object }}</a>
{% else %}
{% if not hide_link %}<a href="{{ object.get_absolute_url }}">{{ object }}</a>{% else %}{{ object }}{% endif %}
{{ object }}
{% endif %}
{% else %}
{% source_column_resolve column=object|get_source_columns:"only_identifier" %}
{% endif %}
</span>
</label>
</div>
@@ -81,10 +85,9 @@
</div>
<div class="panel-body">
{% if not hide_columns %}
{% for column in object|get_source_columns %}
<div class="text-center" style="">{% source_column_resolve column=column %}{{ column_result }}</div>
{% for column in object|get_source_columns:"exclude_identifier" %}
<div class="text-center" style="">{% source_column_resolve column=column as column_value %}{% if column_value != '' %}{{ column.label }}: {{ column_value }}{% endif %}</div>
{% endfor %}
{% endif %}

View File

@@ -26,12 +26,18 @@
<div class="well center-block">
<div class="clearfix">
{% if object_list %}
{% if not hide_multi_item_actions %}
{% get_multi_item_links_form object_list %}
{% endif %}
{% endif %}
{% include 'appearance/list_toolbar.html' %}
<div class="pull-right">
<form action="{% url 'common:multi_object_action_view' %}" class="form-multi-object-action" method="get">
{% if object_list %}
{% if not hide_multi_item_actions %}
{% get_multi_item_links_form object_list %}
{% endif %}
{% if multi_item_actions %}
<fieldset style="margin-top: -10px; margin-bottom: 10px;">
{{ multi_item_form }}
@@ -42,13 +48,17 @@
</div>
</div>
{% if object_list %}
<hr style="border-bottom: 1px solid lightgrey;">
{% endif %}
<div class="table-responsive">
<table class="table table-condensed table-striped">
<tbody>
<thead>
{% if not hide_header %}
<tr>
{% if multi_item_actions %}
<th class="first"><input class="checkbox check-all" type="checkbox" /></th>
<th class="first"></th>
{% endif %}
{% if not hide_object %}
@@ -70,6 +80,8 @@
{% endif %}
</tr>
{% endif %}
</thead>
<tbody>
{% for object in object_list %}
<tr>
{% if multi_item_actions %}
@@ -82,13 +94,7 @@
</td>
{% endif %}
{% if not hide_object %}
{% if main_object %}
{% with object|object_property:main_object as object %}
<td>{% if not hide_link %}<a href="{{ object.get_absolute_url }}">{{ object }}</a>{% else %}{{ object }}{% endif %}</td>
{% endwith %}
{% else %}
<td>{% if not hide_link %}<a href="{{ object.get_absolute_url }}">{{ object }}</a>{% else %}{{ object }}{% endif %}</td>
{% endif %}
<td>{% if not hide_link %}<a href="{{ object.get_absolute_url }}">{{ object }}</a>{% else %}{{ object }}{% endif %}</td>
{% endif %}
{% if not hide_columns %}
{% for column in object|get_source_columns %}

View File

@@ -0,0 +1,16 @@
{% load i18n %}
<div class="pull-left">
<div class="btn-toolbar" role="toolbar">
<div class="btn-group">
{% if multi_item_actions %}
<a class="btn btn-default btn-sm check-all" data-checked=false data-icon-checked="fa fa-check-square" data-icon-unchecked="far fa-square" title="{% trans 'Select/Deselect all' %}">
<i class="far fa-square"></i>
</a>
{% endif %}
<a class="btn btn-default btn-sm" href="?_list_mode={% if list_as_items %}list{% else %}items{% endif %}" title="{% trans 'Toggle list display mode' %}">
<i class="fa {% if list_as_items %}fa-list{% else %}fa-th-large{% endif %}"></i>
</a>
</div>
</div>
</div>

View File

@@ -92,7 +92,7 @@
<script src="{% static 'appearance/node_modules/select2/dist/js/select2.min.js' %}" type="text/javascript"></script>
<script src="{% static 'appearance/node_modules/toastr/build/toastr.min.js' %}" type="text/javascript"></script>
<script src="{% static 'appearance/node_modules/jquery-match-height/dist/jquery.matchHeight-min.js' %}" type="text/javascript"></script>
<script src="{% static 'appearance/node_modules/@fortawesome/fontawesome-free/js/all.min.js' %}" type="text/javascript"></script>
<script src="{% static 'appearance/node_modules/@fortawesome/fontawesome-free/js/all.min.js' %}" data-auto-replace-svg="nest" type="text/javascript"></script>
<script src="{% static 'appearance/node_modules/urijs/src/IPv6.js' %}" type="text/javascript"></script>
<script src="{% static 'appearance/node_modules/urijs/src/punycode.js' %}" type="text/javascript"></script>
<script src="{% static 'appearance/node_modules/urijs/src/SecondLevelDomains.js' %}" type="text/javascript"></script>

View File

@@ -498,6 +498,18 @@ class SingleObjectListView(PaginationMixin, ViewPermissionCheckMixin, ObjectList
return result
def get_context_data(self, **kwargs):
context = super(SingleObjectListView, self).get_context_data(**kwargs)
if context.get('list_as_items'):
default_mode = 'items'
else:
default_mode = 'list'
list_mode = self.request.GET.get('_list_mode', default_mode)
context.update({'list_as_items': list_mode=='items'})
return context
def get_paginate_by(self, queryset):
return setting_paginate_by.value

View File

@@ -495,34 +495,46 @@ class SourceColumn(object):
return sorted(columns, key=lambda x: x.order)
@classmethod
def get_for_source(cls, source):
def get_for_source(cls, source, exclude_identifier=False, only_identifier=False):
try:
return SourceColumn.sort(columns=cls._registry[source])
result = SourceColumn.sort(columns=cls._registry[source])
except KeyError:
try:
# Try it as a queryset
return SourceColumn.sort(columns=cls._registry[source.model])
result = SourceColumn.sort(columns=cls._registry[source.model])
except AttributeError:
try:
# It seems to be an instance, try its class
return SourceColumn.sort(columns=cls._registry[source.__class__])
result = SourceColumn.sort(columns=cls._registry[source.__class__])
except KeyError:
try:
# Special case for queryset items produced from
# .defer() or .only() optimizations
return SourceColumn.sort(columns=cls._registry[list(source._meta.parents.items())[0][0]])
result = SourceColumn.sort(columns=cls._registry[list(source._meta.parents.items())[0][0]])
except (AttributeError, KeyError, IndexError):
return ()
result = ()
except TypeError:
# unhashable type: list
return ()
result = ()
def __init__(self, source, label=None, attribute=None, func=None, order=None):
if exclude_identifier:
return [item for item in result if not item.is_identifier]
else:
if only_identifier:
for item in result:
if item.is_identifier:
return item
else:
return result
def __init__(self, source, label=None, attribute=None, func=None, kwargs=None, order=None, is_identifier=False):
self.source = source
self._label = label
self.attribute = attribute
self.func = func
self.kwargs = kwargs or {}
self.order = order or 0
self.is_identifier = is_identifier
self.__class__._registry.setdefault(source, [])
self.__class__._registry[source].append(self)
@@ -543,9 +555,12 @@ class SourceColumn(object):
def resolve(self, context):
if self.attribute:
result = resolve_attribute(context['object'], self.attribute)
result = resolve_attribute(
obj=context['object'], attribute=self.attribute,
kwargs=self.kwargs
)
elif self.func:
result = self.func(context=context)
result = self.func(context=context, **self.kwargs)
return result

View File

@@ -0,0 +1,4 @@
from __future__ import unicode_literals
TEXT_ONLY_IDENTIFIER = 'only_identifier'
TEXT_EXCLUDE_IDENTIFIER = 'exclude_identifier'

View File

@@ -4,6 +4,7 @@ from django.template import Library
from ..classes import Link, Menu, SourceColumn
from ..forms import MultiItemForm
from ..literals import TEXT_EXCLUDE_IDENTIFIER, TEXT_ONLY_IDENTIFIER
register = Library()
@@ -48,7 +49,15 @@ def get_multi_item_links_form(context, object_list):
@register.filter
def get_source_columns(source):
def get_source_columns(source, args=None):
exclude_identifier = False
only_identifier = False
if args == TEXT_EXCLUDE_IDENTIFIER:
exclude_identifier = True
elif args == TEXT_ONLY_IDENTIFIER:
only_identifier = True
try:
# Is it a query set?
source = source.model
@@ -67,7 +76,10 @@ def get_source_columns(source):
# It a list and it's empty
pass
return SourceColumn.get_for_source(source)
return SourceColumn.get_for_source(
source=source, exclude_identifier=exclude_identifier,
only_identifier=only_identifier
)
@register.simple_tag(takes_context=True)
@@ -78,5 +90,8 @@ def resolve_link(context, link):
@register.simple_tag(takes_context=True)
def source_column_resolve(context, column):
context['column_result'] = column.resolve(context=context)
return ''
if column:
result = column.resolve(context=context)
return result
else:
return ''