rapidforms.py 5.0 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126
  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 = list(default_relations)
  45. default_relations_request = request.GET.get('default')
  46. widgets = []
  47. default_relations_fields = []
  48. if default_relations_request:
  49. default_relations_fields = default_relations_request.split(",")
  50. default_relations += ((x, int(y)) for (x, y) in (f.split(":") for f in default_relations_fields))
  51. default_relations_fields = [x for x, y in default_relations]
  52. for (x, y) in default_relations:
  53. f = FieldData(getattr(model, x).field, request)
  54. widgets.append((x, RapidRelationReadOnly(f.related_model())))
  55. for f in ModelData(model).local_fields():
  56. if f.is_relation() and unicode(f.bare_name()) not in default_relations_fields:
  57. if f.related_model().has_permission(request, 'select'):
  58. widgets.append((f.bare_name(), RapidSelector(f)))
  59. # ModelForm.Meta has attributes with the same names, thus I'll rename them
  60. form_model = model
  61. form_widgets = dict(widgets)
  62. # noinspection PyTypeChecker
  63. class CForm(forms.ModelForm):
  64. def __init__(self, *args, **kwargs):
  65. initial = kwargs.get('initial', {})
  66. for (k, v) in default_relations:
  67. initial[k] = v
  68. if initial:
  69. kwargs['initial'] = initial
  70. instance = kwargs.get('instance')
  71. for n, f in ModelData(model).rapid_alternative_data():
  72. ct = ModelData(model).field_by_name(f.ct_field).field
  73. fk = ModelData(model).field_by_name(f.fk_field).field
  74. # noinspection PyTypeChecker
  75. fl = RapidAlternativesField(n, ct.alternatives, ct.name, self, request, instance)
  76. # noinspection PyArgumentList
  77. type(self.__class__).__setattr__(self.__class__, n, fl)
  78. nd = collections.OrderedDict()
  79. for k, v in self.__class__.base_fields.iteritems():
  80. if k == fk.name:
  81. nd[n] = fl
  82. else:
  83. nd[k] = v
  84. self.__class__.base_fields = nd
  85. super(CForm, self).__init__(*args, **kwargs)
  86. @transaction.atomic
  87. def save(self, commit=True):
  88. if not commit:
  89. return super(CForm, self).save(commit)
  90. else:
  91. obj = super(CForm, self).save(commit=False)
  92. for n, f in ModelData(model).rapid_alternative_data():
  93. if self.instance:
  94. old_t = getattr(self.instance, f.ct_field)
  95. new_t = getattr(obj, f.ct_field)
  96. if old_t != new_t:
  97. getattr(self.instance, f.bare_name).delete()
  98. fob = self.cleaned_data[n]
  99. fob.save()
  100. if hasattr(fob, 'save_m2m'):
  101. fob.save_m2m()
  102. setattr(obj, f.fk_field, fob.pk)
  103. obj.save()
  104. self.save_m2m()
  105. return obj
  106. class Meta(object):
  107. model = form_model
  108. fields = '__all__'
  109. widgets = form_widgets
  110. return CForm