nk20/apps/note/tables.py

296 lines
10 KiB
Python
Raw Normal View History

Update 131 files - /apps/activity/api/serializers.py - /apps/activity/api/urls.py - /apps/activity/api/views.py - /apps/activity/tests/test_activities.py - /apps/activity/__init__.py - /apps/activity/admin.py - /apps/activity/apps.py - /apps/activity/forms.py - /apps/activity/tables.py - /apps/activity/urls.py - /apps/activity/views.py - /apps/api/__init__.py - /apps/api/apps.py - /apps/api/serializers.py - /apps/api/tests.py - /apps/api/urls.py - /apps/api/views.py - /apps/api/viewsets.py - /apps/logs/signals.py - /apps/logs/apps.py - /apps/logs/__init__.py - /apps/logs/api/serializers.py - /apps/logs/api/urls.py - /apps/logs/api/views.py - /apps/member/api/serializers.py - /apps/member/api/urls.py - /apps/member/api/views.py - /apps/member/templatetags/memberinfo.py - /apps/member/__init__.py - /apps/member/admin.py - /apps/member/apps.py - /apps/member/auth.py - /apps/member/forms.py - /apps/member/hashers.py - /apps/member/signals.py - /apps/member/tables.py - /apps/member/urls.py - /apps/member/views.py - /apps/note/api/serializers.py - /apps/note/api/urls.py - /apps/note/api/views.py - /apps/note/models/__init__.py - /apps/note/static/note/js/consos.js - /apps/note/templates/note/mails/negative_balance.txt - /apps/note/templatetags/getenv.py - /apps/note/templatetags/pretty_money.py - /apps/note/tests/test_transactions.py - /apps/note/__init__.py - /apps/note/admin.py - /apps/note/apps.py - /apps/note/forms.py - /apps/note/signals.py - /apps/note/tables.py - /apps/note/urls.py - /apps/note/views.py - /apps/permission/api/serializers.py - /apps/permission/api/urls.py - /apps/permission/api/views.py - /apps/permission/templatetags/perms.py - /apps/permission/tests/test_oauth2.py - /apps/permission/tests/test_permission_denied.py - /apps/permission/tests/test_permission_queries.py - /apps/permission/tests/test_rights_page.py - /apps/permission/__init__.py - /apps/permission/admin.py - /apps/permission/backends.py - /apps/permission/apps.py - /apps/permission/decorators.py - /apps/permission/permissions.py - /apps/permission/scopes.py - /apps/permission/signals.py - /apps/permission/tables.py - /apps/permission/urls.py - /apps/permission/views.py - /apps/registration/tests/test_registration.py - /apps/registration/__init__.py - /apps/registration/apps.py - /apps/registration/forms.py - /apps/registration/tables.py - /apps/registration/tokens.py - /apps/registration/urls.py - /apps/registration/views.py - /apps/treasury/api/serializers.py - /apps/treasury/api/urls.py - /apps/treasury/api/views.py - /apps/treasury/templatetags/escape_tex.py - /apps/treasury/tests/test_treasury.py - /apps/treasury/__init__.py - /apps/treasury/admin.py - /apps/treasury/apps.py - /apps/treasury/forms.py - /apps/treasury/signals.py - /apps/treasury/tables.py - /apps/treasury/urls.py - /apps/treasury/views.py - /apps/wei/api/serializers.py - /apps/wei/api/urls.py - /apps/wei/api/views.py - /apps/wei/forms/surveys/__init__.py - /apps/wei/forms/surveys/base.py - /apps/wei/forms/surveys/wei2021.py - /apps/wei/forms/surveys/wei2022.py - /apps/wei/forms/surveys/wei2023.py - /apps/wei/forms/__init__.py - /apps/wei/forms/registration.py - /apps/wei/management/commands/export_wei_registrations.py - /apps/wei/management/commands/import_scores.py - /apps/wei/management/commands/wei_algorithm.py - /apps/wei/templates/wei/weilist_sample.tex - /apps/wei/tests/test_wei_algorithm_2021.py - /apps/wei/tests/test_wei_algorithm_2022.py - /apps/wei/tests/test_wei_algorithm_2023.py - /apps/wei/tests/test_wei_registration.py - /apps/wei/__init__.py - /apps/wei/admin.py - /apps/wei/apps.py - /apps/wei/tables.py - /apps/wei/urls.py - /apps/wei/views.py - /note_kfet/settings/__init__.py - /note_kfet/settings/base.py - /note_kfet/settings/development.py - /note_kfet/settings/secrets_example.py - /note_kfet/static/js/base.js - /note_kfet/admin.py - /note_kfet/inputs.py - /note_kfet/middlewares.py - /note_kfet/urls.py - /note_kfet/views.py - /note_kfet/wsgi.py - /entrypoint.sh
2024-02-07 01:26:49 +00:00
# Copyright (C) 2018-2024 by BDE ENS Paris-Saclay
2020-02-18 20:30:26 +00:00
# SPDX-License-Identifier: GPL-3.0-or-later
2020-03-14 01:08:23 +00:00
import html
2019-08-15 21:08:15 +00:00
import django_tables2 as tables
from django.utils.html import format_html, mark_safe
from django_tables2.utils import A
2020-03-16 11:11:16 +00:00
from django.utils.translation import gettext_lazy as _
from note_kfet.middlewares import get_current_request
from permission.backends import PermissionBackend
2020-03-07 21:28:59 +00:00
from .models.notes import Alias, Trust
2020-03-23 14:35:24 +00:00
from .models.transactions import Transaction, TransactionTemplate
2020-03-09 17:53:13 +00:00
from .templatetags.pretty_money import pretty_money
2019-08-15 21:08:15 +00:00
2020-03-09 18:19:56 +00:00
2019-08-15 21:08:15 +00:00
class HistoryTable(tables.Table):
class Meta:
2020-02-18 11:31:15 +00:00
attrs = {
2020-08-10 16:01:39 +00:00
'class': 'table table-condensed table-striped'
2020-02-18 11:31:15 +00:00
}
2019-08-15 21:08:15 +00:00
model = Transaction
2020-03-26 13:45:48 +00:00
exclude = ("id", "polymorphic_ctype", "invalidity_reason", "source_alias", "destination_alias",)
template_name = 'django_tables2/bootstrap4.html'
sequence = ('...', 'type', 'total', 'valid',)
2020-03-14 01:08:23 +00:00
orderable = False
2019-08-15 21:08:15 +00:00
2020-03-26 13:45:48 +00:00
source = tables.Column(
attrs={
"td": {
"class": "text-nowrap",
2020-03-26 13:45:48 +00:00
"data-toggle": "tooltip",
"title": lambda record: _("used alias").capitalize() + " : " + record.source_alias,
}
}
)
destination = tables.Column(
attrs={
"td": {
"class": "text-nowrap",
2020-03-26 13:45:48 +00:00
"data-toggle": "tooltip",
"title": lambda record: _("used alias").capitalize() + " : " + record.destination_alias,
}
}
)
created_at = tables.DateTimeColumn(
format='Y-m-d H:i:s',
attrs={
"td": {
"class": "text-nowrap",
},
}
)
amount = tables.Column(
attrs={
"td": {
"class": "text-nowrap",
},
}
)
reason = tables.Column(
attrs={
"td": {
"class": "text-break",
},
}
)
2020-03-16 11:11:16 +00:00
type = tables.Column()
2019-08-15 21:08:15 +00:00
total = tables.Column( # will use Transaction.total() !!
attrs={
"td": {
"class": "text-nowrap",
},
}
)
2019-08-15 21:08:15 +00:00
valid = tables.Column(
attrs={
"td": {
"id": lambda record: "validate_" + str(record.id),
"class": lambda record:
str(record.valid).lower()
2020-08-31 18:15:48 +00:00
+ (' validate' if record.source.is_active and record.destination.is_active and PermissionBackend
.check_perm(get_current_request(), "note.change_transaction_invalidity_reason", record)
2020-08-31 18:15:48 +00:00
else ''),
2020-03-25 13:50:21 +00:00
"data-toggle": "tooltip",
"title": lambda record: (_("Click to invalidate") if record.valid else _("Click to validate"))
if PermissionBackend.check_perm(get_current_request(),
2020-08-31 18:15:48 +00:00
"note.change_transaction_invalidity_reason", record)
and record.source.is_active and record.destination.is_active else None,
"onclick": lambda record: 'de_validate(' + str(record.id) + ', ' + str(record.valid).lower()
+ ', "' + str(record.__class__.__name__) + '")'
if PermissionBackend.check_perm(get_current_request(),
2020-08-31 18:15:48 +00:00
"note.change_transaction_invalidity_reason", record)
and record.source.is_active and record.destination.is_active else None,
2020-03-25 13:50:21 +00:00
"onmouseover": lambda record: '$("#invalidity_reason_'
+ str(record.id) + '").show();$("#invalidity_reason_'
+ str(record.id) + '").focus();',
"onmouseout": lambda record: '$("#invalidity_reason_' + str(record.id) + '").hide()',
}
}
)
2020-03-14 01:08:23 +00:00
2020-03-09 18:19:56 +00:00
def render_amount(self, value):
2020-03-09 17:53:13 +00:00
return pretty_money(value)
2020-03-09 18:19:56 +00:00
def render_total(self, value):
2020-03-09 17:53:13 +00:00
return pretty_money(value)
2020-03-07 21:28:59 +00:00
2020-03-16 11:11:16 +00:00
def render_type(self, value):
return _(value)
2020-03-14 01:08:23 +00:00
# Django-tables escape strings. That's a wrong thing.
def render_reason(self, value):
return html.unescape(value)
def render_valid(self, value, record):
2020-03-25 13:54:26 +00:00
"""
When the validation status is hovered, an input field is displayed to let the user specify an invalidity reason
"""
has_perm = PermissionBackend \
.check_perm(get_current_request(), "note.change_transaction_invalidity_reason", record)
val = "" if value else ""
if value and not has_perm:
return val
2020-03-25 13:50:21 +00:00
val += "<input type='text' class='form-control' id='invalidity_reason_" + str(record.id) \
+ "' value='" + (html.escape(record.invalidity_reason)
if record.invalidity_reason else ("" if value else str(_("No reason specified")))) \
2020-08-31 18:15:48 +00:00
+ "'" + ("" if value and record.source.is_active and record.destination.is_active else " disabled") \
2020-03-25 13:50:21 +00:00
+ " placeholder='" + html.escape(_("invalidity reason").capitalize()) + "'" \
+ " style='position: absolute; width: 15em; margin-left: -15.5em; margin-top: -2em; display: none;'>"
return format_html(val)
2020-03-14 01:08:23 +00:00
2020-03-24 23:03:48 +00:00
2020-03-24 22:02:12 +00:00
# function delete_button(id) provided in template file
2020-03-24 23:03:48 +00:00
DELETE_TEMPLATE = """
2020-03-26 16:46:38 +00:00
<button id="{{ record.pk }}" class="btn btn-danger btn-sm" onclick="delete_button(this.id)"> {{ delete_trans }}</button>
2020-03-24 22:02:12 +00:00
"""
2020-03-24 23:03:48 +00:00
2020-03-24 23:12:56 +00:00
class TrustTable(tables.Table):
class Meta:
attrs = {
'class': 'table table condensed table-striped',
'id': "trust_table"
}
model = Trust
fields = ("trusted",)
template_name = 'django_tables2/bootstrap4.html'
show_header = False
trusted = tables.Column(attrs={'td': {'class': 'text-center'}})
2021-10-04 18:45:05 +00:00
delete_col = tables.TemplateColumn(
template_code=DELETE_TEMPLATE,
extra_context={"delete_trans": _('Delete')},
2021-10-04 18:45:05 +00:00
attrs={
'td': {
'class': lambda record: 'col-sm-1'
+ (' d-none' if not PermissionBackend.check_perm(
get_current_request(), "note.delete_trust", record)
else '')}},
verbose_name=_("Delete"),)
2022-09-28 10:50:11 +00:00
class TrustedTable(tables.Table):
class Meta:
attrs = {
'class': 'table table condensed table-striped',
'id': 'trusted_table'
}
Model = Trust
fields = ("trusting",)
template_name = "django_tables2/bootstrap4.html"
show_header = False
2022-09-28 08:28:47 +00:00
trusting = tables.Column(attrs={
2022-09-28 10:50:11 +00:00
'td': {'class': 'text-center', 'width': '100%'}})
trust_back = tables.Column(
verbose_name=_("Trust back"),
accessor="pk",
attrs={
'td': {
2022-09-28 08:28:47 +00:00
'class': '',
'id': lambda record: "trust_back_" + str(record.pk),
}
},
)
def render_trust_back(self, record):
user_note = record.trusted
trusting_note = record.trusting
2022-09-28 10:50:11 +00:00
if Trust.objects.filter(trusted=trusting_note, trusting=user_note):
return ""
val = '<button id="'
val += str(record.pk)
2022-09-28 08:28:47 +00:00
val += '" class="btn btn-success btn-sm text-nowrap" \
onclick="create_trust(' + str(record.trusted.pk) + ',' + \
str(record.trusting.pk) + ')">'
2022-09-28 08:28:47 +00:00
val += str(_("Add back"))
val += '</button>'
return mark_safe(val)
2020-02-28 12:37:31 +00:00
class AliasTable(tables.Table):
class Meta:
attrs = {
2020-08-10 16:01:39 +00:00
'class': 'table table condensed table-striped',
2020-03-27 13:19:55 +00:00
'id': "alias_table"
2020-02-28 12:37:31 +00:00
}
model = Alias
2020-03-07 21:28:59 +00:00
fields = ('name',)
2020-02-28 12:37:31 +00:00
template_name = 'django_tables2/bootstrap4.html'
2020-03-01 12:42:22 +00:00
show_header = False
2020-03-07 21:28:59 +00:00
name = tables.Column(attrs={'td': {'class': 'text-center'}})
2020-03-26 16:47:17 +00:00
2020-03-26 22:04:14 +00:00
delete_col = tables.TemplateColumn(template_code=DELETE_TEMPLATE,
2020-03-27 13:19:55 +00:00
extra_context={"delete_trans": _('delete')},
2020-08-31 18:15:48 +00:00
attrs={'td': {'class': lambda record: 'col-sm-1' + (
' d-none' if not PermissionBackend.check_perm(
get_current_request(), "note.delete_alias",
2020-08-31 18:15:48 +00:00
record) else '')}}, verbose_name=_("Delete"), )
2020-03-24 23:12:56 +00:00
class ButtonTable(tables.Table):
class Meta:
attrs = {
2020-08-10 16:01:39 +00:00
'class': 'table table-bordered condensed'
}
2020-03-23 19:21:25 +00:00
row_attrs = {
'class': lambda record: 'table-row ' + ('table-success' if record.display else 'table-danger'),
2020-03-24 23:12:56 +00:00
'id': lambda record: "row-" + str(record.pk),
2020-03-23 19:21:25 +00:00
}
model = TransactionTemplate
2020-04-01 18:56:24 +00:00
exclude = ('id',)
edit = tables.LinkColumn(
'note:template_update',
args=[A('pk')],
attrs={
'td': {'class': 'col-sm-1'},
'a': {
'class': 'btn btn-sm btn-primary',
'data-turbolinks': 'false',
}
},
text=_('edit'),
accessor='pk',
verbose_name=_("Edit"),
)
2020-03-24 23:03:48 +00:00
hideshow = tables.Column(
verbose_name=_("Hide/Show"),
accessor="pk",
attrs={
'td': {
'class': 'col-sm-1',
'id': lambda record: "hideshow_" + str(record.pk),
}
},
)
2020-03-26 22:04:14 +00:00
delete_col = tables.TemplateColumn(template_code=DELETE_TEMPLATE,
2020-03-27 13:19:55 +00:00
extra_context={"delete_trans": _('delete')},
2020-04-01 18:56:24 +00:00
attrs={'td': {'class': 'col-sm-1'}},
verbose_name=_("Delete"), )
2020-03-24 23:03:48 +00:00
def render_amount(self, value):
return pretty_money(value)
def order_category(self, queryset, is_descending):
return queryset.order_by(f"{'-' if is_descending else ''}category__name"), True
def render_hideshow(self, record):
val = '<button id="'
val += str(record.pk)
val += '" class="btn btn-secondary btn-sm" \
onclick="hideshow(' + str(record.id) + ',' + \
str(record.display).lower() + ')">'
val += str(_("Hide/Show"))
val += '</button>'
return mark_safe(val)