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