
Use namespaces when referencing python packages. Also, as per https://review.opendev.org/#/c/527320/, replace all references to django.core.urlresolves with django.urls. Change-Id: I6460d0734aa11ffa44f259baa7155d2a202f7162
212 lines
7.6 KiB
Python
212 lines
7.6 KiB
Python
# Licensed under the Apache License, Version 2.0 (the "License"); you may
|
|
# not use this file except in compliance with the License. You may obtain
|
|
# a copy of the License at
|
|
#
|
|
# http://www.apache.org/licenses/LICENSE-2.0
|
|
#
|
|
# Unless required by applicable law or agreed to in writing, software
|
|
# distributed under the License is distributed on an "AS IS" BASIS, WITHOUT
|
|
# WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the
|
|
# License for the specific language governing permissions and limitations
|
|
# under the License.
|
|
|
|
from itertools import chain
|
|
|
|
from django import urls
|
|
|
|
from django.forms import fields
|
|
from django.forms import TextInput
|
|
from django.forms import widgets
|
|
from django.utils.encoding import force_text
|
|
from django.utils.safestring import mark_safe
|
|
|
|
from django.forms.utils import flatatt
|
|
from django.utils.html import format_html
|
|
|
|
from django.utils.translation import ugettext_lazy as _
|
|
|
|
|
|
class DynamicMultiSelectWidget(widgets.SelectMultiple):
|
|
|
|
"""A subclass of the ``Select`` widget which renders extra attributes for
|
|
use in callbacks to handle dynamic changes to the available choices.
|
|
"""
|
|
_data_add_url_attr = "data-add-item-url"
|
|
|
|
def render(self, *args, **kwargs):
|
|
add_item_url = self.get_add_item_url()
|
|
if add_item_url is not None:
|
|
self.attrs[self._data_add_url_attr] = add_item_url
|
|
return super(DynamicMultiSelectWidget, self).render(*args, **kwargs)
|
|
|
|
def get_add_item_url(self):
|
|
if callable(self.add_item_link):
|
|
return self.add_item_link()
|
|
try:
|
|
if self.add_item_link_args:
|
|
return urls.reverse(self.add_item_link,
|
|
args=self.add_item_link_args)
|
|
else:
|
|
return urls.reverse(self.add_item_link)
|
|
except urls.NoReverseMatch:
|
|
return self.add_item_link
|
|
|
|
|
|
class DynamicMultiChoiceField(fields.MultipleChoiceField):
|
|
|
|
"""A subclass of ``ChoiceField`` with additional properties that make
|
|
dynamically updating its elements easier.
|
|
|
|
Notably, the field declaration takes an extra argument, ``add_item_link``
|
|
which may be a string or callable defining the URL that should be used
|
|
for the "add" link associated with the field.
|
|
"""
|
|
widget = DynamicMultiSelectWidget
|
|
|
|
def __init__(self,
|
|
add_item_link=None,
|
|
add_item_link_args=None,
|
|
*args,
|
|
**kwargs):
|
|
super(DynamicMultiChoiceField, self).__init__(*args, **kwargs)
|
|
self.widget.add_item_link = add_item_link
|
|
self.widget.add_item_link_args = add_item_link_args
|
|
|
|
|
|
class CustomMultiChoiceField(DynamicMultiChoiceField):
|
|
def validate(self, *args, **kwargs):
|
|
return True
|
|
|
|
|
|
class DropdownEditWidget(TextInput):
|
|
def __init__(self, data_list, name, *args, **kwargs):
|
|
super(DropdownEditWidget, self).__init__(*args, **kwargs)
|
|
self._name = name
|
|
self._list = data_list
|
|
self.attrs.update({'list': 'list__%s' % self._name})
|
|
|
|
def render(self, name, value, attrs=None):
|
|
text_html = super(DropdownEditWidget, self).render(
|
|
name, value, attrs=attrs)
|
|
data_list = '<datalist id="list__%s">' % self._name
|
|
for item in self._list:
|
|
data_list += '<option value="%s">' % item
|
|
data_list += '</datalist>'
|
|
return mark_safe(text_html + data_list)
|
|
|
|
|
|
class TransferTableWidget(widgets.SelectMultiple):
|
|
actions_list = []
|
|
add_item_link = None
|
|
max_items = None
|
|
allocated_filter = False
|
|
|
|
allocated_help_text = None
|
|
available_help_text = None
|
|
no_allocated_text = None
|
|
no_available_text = None
|
|
|
|
def render(self, name, value, attrs=None, choices=()):
|
|
# css class currently breaks the layout for some reason,
|
|
self.attrs.pop('class', None)
|
|
|
|
final_attrs = self.build_attrs(attrs, name=name)
|
|
|
|
selected = [] if value is None else value
|
|
|
|
options = self.render_options(choices, selected)
|
|
|
|
if self.add_item_link is not None:
|
|
final_attrs['add_item_link'] = urls.reverse(
|
|
self.add_item_link
|
|
)
|
|
|
|
if self.max_items is not None:
|
|
final_attrs['max_items'] = self.max_items
|
|
|
|
if self.allocated_filter:
|
|
final_attrs['allocated_filter'] = "True"
|
|
|
|
final_attrs['allocated_help_text'] = self.allocated_help_text
|
|
final_attrs['available_help_text'] = self.available_help_text
|
|
final_attrs['no_allocated_text'] = self.no_allocated_text
|
|
final_attrs['no_available_text'] = self.no_available_text
|
|
|
|
open_tag = format_html('<d-table {}>', flatatt(final_attrs))
|
|
|
|
output = [open_tag, options, '</d-table>']
|
|
|
|
return mark_safe('\n'.join(output))
|
|
|
|
def build_attrs(self, extra_attrs=None, **kwargs):
|
|
"Helper function for building an attribute dictionary."
|
|
attrs = dict(self.attrs, **kwargs)
|
|
if extra_attrs:
|
|
attrs.update(extra_attrs)
|
|
return attrs
|
|
|
|
def render_option(self, selected_choices, option_value, option_label):
|
|
if option_value is None:
|
|
option_value = ''
|
|
option_value = force_text(option_value)
|
|
if option_value in selected_choices:
|
|
selected_html = mark_safe(' selected="selected"')
|
|
if not self.allow_multiple_selected:
|
|
# Only allow for a single selection.
|
|
selected_choices.remove(option_value)
|
|
else:
|
|
selected_html = ''
|
|
return format_html('<option value="{}"{}>{}</option>',
|
|
option_value,
|
|
selected_html,
|
|
force_text(option_label))
|
|
|
|
def render_options(self, choices, selected_choices):
|
|
# Normalize to strings.
|
|
selected_choices = set(force_text(v) for v in selected_choices)
|
|
output = []
|
|
for option_value, option_label in chain(self.choices, choices):
|
|
if isinstance(option_label, (list, tuple)):
|
|
output.append(format_html('<optgroup label="{}">',
|
|
force_text(option_value)))
|
|
for option in option_label:
|
|
output.append(self.render_option(selected_choices,
|
|
*option))
|
|
output.append('</optgroup>')
|
|
else:
|
|
output.append(self.render_option(selected_choices,
|
|
option_value,
|
|
option_label))
|
|
return '\n'.join(output)
|
|
|
|
# ...this adds the 'add item button' just by existing and returning a
|
|
# true-y value
|
|
def get_add_item_url(self):
|
|
return None
|
|
|
|
|
|
class TransferTableField(fields.MultipleChoiceField):
|
|
widget = TransferTableWidget
|
|
|
|
def __init__(self, add_item_link=None, max_items=-1,
|
|
allocated_filter=False,
|
|
allocated_help_text="",
|
|
available_help_text="",
|
|
no_allocated_text=_("Select items from bellow"),
|
|
no_available_text=_("No available items"),
|
|
*args, **kwargs):
|
|
super(TransferTableField, self).__init__(*args, **kwargs)
|
|
|
|
self.widget.add_item_link = add_item_link
|
|
self.widget.max_items = max_items
|
|
self.widget.allocated_filter = allocated_filter
|
|
|
|
self.widget.allocated_help_text = allocated_help_text
|
|
self.widget.available_help_text = available_help_text
|
|
|
|
self.widget.no_allocated_text = no_allocated_text
|
|
self.widget.no_available_text = no_available_text
|
|
|
|
def validate(self, *args, **kwargs):
|
|
return True
|