create_update.py 8.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215
  1. from django.forms.models import ModelFormMetaclass, ModelForm
  2. from django.template import RequestContext, loader
  3. from django.http import Http404, HttpResponse, HttpResponseRedirect
  4. from django.core.xheaders import populate_xheaders
  5. from django.core.exceptions import ObjectDoesNotExist, ImproperlyConfigured
  6. from django.utils.translation import ugettext
  7. from django.contrib.auth.views import redirect_to_login
  8. from django.views.generic import GenericViewError
  9. from django.contrib import messages
  10. def apply_extra_context(extra_context, context):
  11. """
  12. Adds items from extra_context dict to context. If a value in extra_context
  13. is callable, then it is called and the result is added to context.
  14. """
  15. for key, value in extra_context.iteritems():
  16. if callable(value):
  17. context[key] = value()
  18. else:
  19. context[key] = value
  20. def get_model_and_form_class(model, form_class):
  21. """
  22. Returns a model and form class based on the model and form_class
  23. parameters that were passed to the generic view.
  24. If ``form_class`` is given then its associated model will be returned along
  25. with ``form_class`` itself. Otherwise, if ``model`` is given, ``model``
  26. itself will be returned along with a ``ModelForm`` class created from
  27. ``model``.
  28. """
  29. if form_class:
  30. return form_class._meta.model, form_class
  31. if model:
  32. # The inner Meta class fails if model = model is used for some reason.
  33. tmp_model = model
  34. # TODO: we should be able to construct a ModelForm without creating
  35. # and passing in a temporary inner class.
  36. class Meta:
  37. model = tmp_model
  38. class_name = model.__name__ + 'Form'
  39. form_class = ModelFormMetaclass(class_name, (ModelForm,), {'Meta': Meta})
  40. return model, form_class
  41. raise GenericViewError("Generic view must be called with either a model or"
  42. " form_class argument.")
  43. def redirect(post_save_redirect, obj):
  44. """
  45. Returns a HttpResponseRedirect to ``post_save_redirect``.
  46. ``post_save_redirect`` should be a string, and can contain named string-
  47. substitution place holders of ``obj`` field names.
  48. If ``post_save_redirect`` is None, then redirect to ``obj``'s URL returned
  49. by ``get_absolute_url()``. If ``obj`` has no ``get_absolute_url`` method,
  50. then raise ImproperlyConfigured.
  51. This function is meant to handle the post_save_redirect parameter to the
  52. ``create_object`` and ``update_object`` views.
  53. """
  54. if post_save_redirect:
  55. return HttpResponseRedirect(post_save_redirect % obj.__dict__)
  56. elif hasattr(obj, 'get_absolute_url'):
  57. return HttpResponseRedirect(obj.get_absolute_url())
  58. else:
  59. raise ImproperlyConfigured(
  60. "No URL to redirect to. Either pass a post_save_redirect"
  61. " parameter to the generic view or define a get_absolute_url"
  62. " method on the Model.")
  63. def lookup_object(model, object_id, slug, slug_field):
  64. """
  65. Return the ``model`` object with the passed ``object_id``. If
  66. ``object_id`` is None, then return the object whose ``slug_field``
  67. equals the passed ``slug``. If ``slug`` and ``slug_field`` are not passed,
  68. then raise Http404 exception.
  69. """
  70. lookup_kwargs = {}
  71. if object_id:
  72. lookup_kwargs['%s__exact' % model._meta.pk.name] = object_id
  73. elif slug and slug_field:
  74. lookup_kwargs['%s__exact' % slug_field] = slug
  75. else:
  76. raise GenericViewError(
  77. "Generic view must be called with either an object_id or a"
  78. " slug/slug_field.")
  79. try:
  80. return model.objects.get(**lookup_kwargs)
  81. except ObjectDoesNotExist:
  82. raise Http404("No %s found for %s"
  83. % (model._meta.verbose_name, lookup_kwargs))
  84. def create_object(request, model=None, template_name=None,
  85. template_loader=loader, extra_context=None, post_save_redirect=None,
  86. login_required=False, context_processors=None, form_class=None):
  87. """
  88. Generic object-creation function.
  89. Templates: ``<app_label>/<model_name>_form.html``
  90. Context:
  91. form
  92. the form for the object
  93. """
  94. if extra_context is None: extra_context = {}
  95. if login_required and not request.user.is_authenticated():
  96. return redirect_to_login(request.path)
  97. model, form_class = get_model_and_form_class(model, form_class)
  98. if request.method == 'POST':
  99. form = form_class(request.POST, request.FILES)
  100. if form.is_valid():
  101. new_object = form.save()
  102. msg = ugettext("The %(verbose_name)s was created successfully.") %\
  103. {"verbose_name": model._meta.verbose_name}
  104. messages.success(request, msg, fail_silently=True)
  105. return redirect(post_save_redirect, new_object)
  106. else:
  107. form = form_class()
  108. # Create the template, context, response
  109. if not template_name:
  110. template_name = "%s/%s_form.html" % (model._meta.app_label, model._meta.object_name.lower())
  111. t = template_loader.get_template(template_name)
  112. c = RequestContext(request, {
  113. 'form': form,
  114. }, context_processors)
  115. apply_extra_context(extra_context, c)
  116. return HttpResponse(t.render(c))
  117. def update_object(request, model=None, object_id=None, slug=None,
  118. slug_field='slug', template_name=None, template_loader=loader,
  119. extra_context=None, post_save_redirect=None, login_required=False,
  120. context_processors=None, template_object_name='object',
  121. form_class=None):
  122. """
  123. Generic object-update function.
  124. Templates: ``<app_label>/<model_name>_form.html``
  125. Context:
  126. form
  127. the form for the object
  128. object
  129. the original object being edited
  130. """
  131. if extra_context is None: extra_context = {}
  132. if login_required and not request.user.is_authenticated():
  133. return redirect_to_login(request.path)
  134. model, form_class = get_model_and_form_class(model, form_class)
  135. obj = lookup_object(model, object_id, slug, slug_field)
  136. if request.method == 'POST':
  137. form = form_class(request.POST, request.FILES, instance=obj)
  138. if form.is_valid():
  139. obj = form.save()
  140. msg = ugettext("The %(verbose_name)s was updated successfully.") %\
  141. {"verbose_name": model._meta.verbose_name}
  142. messages.success(request, msg, fail_silently=True)
  143. return redirect(post_save_redirect, obj)
  144. else:
  145. form = form_class(instance=obj)
  146. if not template_name:
  147. template_name = "%s/%s_form.html" % (model._meta.app_label, model._meta.object_name.lower())
  148. t = template_loader.get_template(template_name)
  149. c = RequestContext(request, {
  150. 'form': form,
  151. template_object_name: obj,
  152. }, context_processors)
  153. apply_extra_context(extra_context, c)
  154. response = HttpResponse(t.render(c))
  155. populate_xheaders(request, response, model, getattr(obj, obj._meta.pk.attname))
  156. return response
  157. def delete_object(request, model, post_delete_redirect, object_id=None,
  158. slug=None, slug_field='slug', template_name=None,
  159. template_loader=loader, extra_context=None, login_required=False,
  160. context_processors=None, template_object_name='object'):
  161. """
  162. Generic object-delete function.
  163. The given template will be used to confirm deletetion if this view is
  164. fetched using GET; for safty, deletion will only be performed if this
  165. view is POSTed.
  166. Templates: ``<app_label>/<model_name>_confirm_delete.html``
  167. Context:
  168. object
  169. the original object being deleted
  170. """
  171. if extra_context is None: extra_context = {}
  172. if login_required and not request.user.is_authenticated():
  173. return redirect_to_login(request.path)
  174. obj = lookup_object(model, object_id, slug, slug_field)
  175. if request.method == 'POST':
  176. obj.delete()
  177. msg = ugettext("The %(verbose_name)s was deleted.") %\
  178. {"verbose_name": model._meta.verbose_name}
  179. messages.success(request, msg, fail_silently=True)
  180. return HttpResponseRedirect(post_delete_redirect)
  181. else:
  182. if not template_name:
  183. template_name = "%s/%s_confirm_delete.html" % (model._meta.app_label, model._meta.object_name.lower())
  184. t = template_loader.get_template(template_name)
  185. c = RequestContext(request, {
  186. template_object_name: obj,
  187. }, context_processors)
  188. apply_extra_context(extra_context, c)
  189. response = HttpResponse(t.render(c))
  190. populate_xheaders(request, response, model, getattr(obj, obj._meta.pk.attname))
  191. return response