- Author:
- RobertoMaurizzi
- Posted:
- May 16, 2019
- Language:
- Python
- Version:
- 1.11
- Score:
- 1 (after 1 ratings)
Small fix to make https://www.djangosnippets.org/snippets/1779/ compatible with Django 1.11 To use this you'll also need the javascript from https://www.djangosnippets.org/snippets/1780/
I added an overridden optgroups method that can handle the additional tuple and a couple minor things I gathered while investigating for a fix (a default level_indicator of "+--" and changed paths for the JS files.
All credits goes to @anentropic :-)
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 | from itertools import chain
from django import forms
from django.contrib.admin import widgets
from django.utils.encoding import smart_unicode, force_unicode, force_text
from django.utils.html import escape, conditional_escape
class MPTTModelChoiceIterator(forms.models.ModelChoiceIterator):
"""MPTT version of ModelChoiceIterator"""
def choice(self, obj):
"""Overriding choice method"""
tree_id = getattr(obj, self.queryset.model._mptt_meta.tree_id_attr, 0)
left = getattr(obj, self.queryset.model._mptt_meta.left_attr, 0)
return super(MPTTModelChoiceIterator,
self).choice(obj) + ((tree_id, left),)
class MPTTModelMultipleChoiceField(forms.ModelMultipleChoiceField):
"""MPTT version of ModelMultipleChoiceField"""
def __init__(self, *args, **kwargs):
self.level_indicator = kwargs.pop('level_indicator', '+--')
super(MPTTModelMultipleChoiceField, self).__init__(*args, **kwargs)
def label_from_instance(self, obj):
"""Creates labels which represent the tree level of each node
when generating option labels."""
return u'%s %s' % (self.level_indicator * getattr(
obj, obj._mptt_meta.level_attr), smart_unicode(obj))
def _get_choices(self):
"""Overriding _get_choices"""
if hasattr(self, '_choices'):
return self._choices
return MPTTModelChoiceIterator(self)
choices = property(_get_choices, forms.ChoiceField._set_choices)
class MPTTFilteredSelectMultiple(widgets.FilteredSelectMultiple):
"""MPTT version of FilteredSelectMultiple"""
def render_options(self, choices, selected_choices):
"""
This is copy'n'pasted from django.forms.widgets Select(Widget)
change to the for loop and render_option so they will unpack
and use our extra tuple of mptt sort fields (if you pass in
some default choices for this field, make sure they have the
extra tuple too!)
"""
def render_option(option_value, option_label, sort_fields):
"""Inner scope render_option"""
option_value = force_unicode(option_value)
selected_html = (option_value in selected_choices) \
and u' selected="selected"' or ''
return u'<option value="%s" data-tree-id="%s" ' \
'data-left-value="%s"%s>%s</option>' % (
escape(option_value),
sort_fields[0],
sort_fields[1],
selected_html,
conditional_escape(force_unicode(option_label)),
)
# Normalize to strings.
selected_choices = set([force_unicode(v) for v in selected_choices])
output = []
for option_value, option_label, sort_fields in chain(
self.choices, choices):
if isinstance(option_label, (list, tuple)):
output.append(u'<optgroup label="%s">' % escape(
force_unicode(option_value)))
for option in option_label:
output.append(render_option(*option))
output.append(u'</optgroup>')
else:
output.append(render_option(option_value, option_label,
sort_fields))
return u'\n'.join(output)
def optgroups(self, name, value, attrs=None):
"""Return a list of optgroups for this widget.
Adapted from django.forms.widgets.ChoiceWidget.optgroups method, the only change was
adding `(t, p)` in the for loop"""
groups = []
has_selected = False
for index, (option_value, option_label, (t, l)) in enumerate(chain(self.choices)):
if option_value is None:
option_value = ''
subgroup = []
if isinstance(option_label, (list, tuple)):
group_name = option_value
subindex = 0
choices = option_label
else:
group_name = None
subindex = None
choices = [(option_value, option_label)]
groups.append((group_name, subgroup, index))
for subvalue, sublabel in choices:
selected = (
force_text(subvalue) in value and
(has_selected is False or self.allow_multiple_selected)
)
if selected is True and has_selected is False:
has_selected = True
subgroup.append(self.create_option(
name, subvalue, sublabel, selected, index,
subindex=subindex, attrs=attrs,
))
if subindex is not None:
subindex += 1
return groups
class Media:
# extend = False
js = ("admin/js/core.js",
"js/mptt_m2m_selectbox.js",
"admin/js/SelectFilter2.js",)
|
More like this
- Template tag - list punctuation for a list of items by shapiromatron 10 months, 1 week ago
- JSONRequestMiddleware adds a .json() method to your HttpRequests by cdcarter 10 months, 2 weeks ago
- Serializer factory with Django Rest Framework by julio 1 year, 5 months ago
- Image compression before saving the new model / work with JPG, PNG by Schleidens 1 year, 6 months ago
- Help text hyperlinks by sa2812 1 year, 6 months ago
Comments
Please login first before commenting.