rapidforms.py 5.0 KB

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