rapidforms.py 5.2 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128
  1. # coding=utf-8
  2. from __future__ import absolute_import
  3. from __future__ import division
  4. from __future__ import print_function
  5. from __future__ import unicode_literals
  6. import collections
  7. from django import forms
  8. from django.contrib.contenttypes.models import ContentType
  9. from django.core.exceptions import ValidationError
  10. from django.db import transaction
  11. from django.utils.encoding import force_text
  12. from rapid.widgets import RapidSelector, RapidRelationReadOnly, rapid_alternatives_widget
  13. from rapid.wrappers import FieldData, ModelData
  14. import gettext
  15. _ = gettext.gettext
  16. class RapidAlternativesField(forms.Field):
  17. def __init__(self, field_name, alternatives, selector_name, form, request, instance=None, *args, **kwargs):
  18. model = None
  19. if instance:
  20. model = getattr(instance, selector_name)
  21. if form.base_fields[selector_name].initial:
  22. model = ContentType.objects.get_for_id(form.base_fields[selector_name].initial)
  23. alt = []
  24. for a in alternatives:
  25. md = ModelData(a)
  26. ft = for_model(request, a)
  27. prefix = field_name + '_' + str(md.content_type().pk)
  28. selected = ContentType.objects.get_for_model(a) == model
  29. inst = getattr(instance, field_name) if selected else None
  30. if request.method == 'POST':
  31. fm = ft(request.POST, request.FILES, prefix=prefix, instance=inst)
  32. else:
  33. fm = ft(prefix=prefix, instance=inst)
  34. alt.append((md.content_type().pk, (md, fm, selected)))
  35. alt = dict(alt)
  36. kwargs['widget'] = rapid_alternatives_widget(alt, selector_name)
  37. # noinspection PyArgumentList
  38. super(RapidAlternativesField, self).__init__(field_name, *args, **kwargs)
  39. def to_python(self, value):
  40. if value is None:
  41. return None
  42. if value.is_valid():
  43. obj = value.save(commit=False)
  44. obj.save_m2m = value.save_m2m
  45. return obj
  46. raise ValidationError(_("Invalid value"), code='invalid')
  47. def for_model(request, model, default_relations=()):
  48. default_relations = list(default_relations)
  49. default_relations_request = request.GET.get('default')
  50. widgets = []
  51. default_relations_fields = []
  52. if default_relations_request:
  53. default_relations_fields = default_relations_request.split(",")
  54. default_relations += ((x, int(y)) for (x, y) in (f.split(":") for f in default_relations_fields))
  55. default_relations_fields = [x for x, y in default_relations]
  56. for (x, y) in default_relations:
  57. f = FieldData(getattr(model, x).field, request)
  58. widgets.append((x, RapidRelationReadOnly(f.related_model())))
  59. for f in ModelData(model).local_fields():
  60. if f.is_relation() and force_text(f.bare_name()) not in default_relations_fields:
  61. if f.related_model().has_permission(request, 'select'):
  62. widgets.append((f.bare_name(), RapidSelector(f)))
  63. # ModelForm.Meta has attributes with the same names, thus I'll rename them
  64. form_model = model
  65. form_widgets = dict(widgets)
  66. # noinspection PyTypeChecker
  67. class CForm(forms.ModelForm):
  68. def __init__(self, *args, **kwargs):
  69. initial = kwargs.get('initial', {})
  70. for (k, v) in default_relations:
  71. initial[k] = v
  72. if initial:
  73. kwargs['initial'] = initial
  74. instance = kwargs.get('instance')
  75. for n, fd in ModelData(model).rapid_alternative_data():
  76. ct = ModelData(model).field_by_name(fd.ct_field).field
  77. fk = ModelData(model).field_by_name(fd.fk_field).field
  78. # noinspection PyTypeChecker
  79. fl = RapidAlternativesField(n, ct.alternatives, ct.name, self, request, instance)
  80. # noinspection PyArgumentList
  81. type(self.__class__).__setattr__(self.__class__, n, fl)
  82. nd = collections.OrderedDict()
  83. for k, v in self.__class__.base_fields.items():
  84. if k == fk.name:
  85. nd[n] = fl
  86. else:
  87. nd[k] = v
  88. self.__class__.base_fields = nd
  89. super(CForm, self).__init__(*args, **kwargs)
  90. @transaction.atomic
  91. def save(self, commit=True):
  92. if not commit:
  93. return super(CForm, self).save(commit)
  94. else:
  95. obj = super(CForm, self).save(commit=False)
  96. for n, fd in ModelData(model).rapid_alternative_data():
  97. if self.instance:
  98. old_t = getattr(self.instance, fd.ct_field)
  99. new_t = getattr(obj, fd.ct_field)
  100. if old_t != new_t:
  101. getattr(self.instance, fd.bare_name).delete()
  102. fob = self.cleaned_data[n]
  103. fob.save()
  104. if hasattr(fob, 'save_m2m'):
  105. fob.save_m2m()
  106. setattr(obj, fd.fk_field, fob.pk)
  107. obj.save()
  108. self.save_m2m()
  109. return obj
  110. class Meta(object):
  111. model = form_model
  112. fields = '__all__'
  113. widgets = form_widgets
  114. return CForm