rapidforms.py 5.1 KB

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