1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84 | import django.newforms as forms
from django.db import models
import os
class DeleteCheckboxWidget(forms.CheckboxInput):
def __init__(self, *args, **kwargs):
self.is_image = kwargs.pop('is_image')
self.value = kwargs.pop('initial')
super(DeleteCheckboxWidget, self).__init__(*args, **kwargs)
def render(self, name, value, attrs=None):
value = value or self.value
if value:
s = u'<label for="%s">%s %s</label>' % (
attrs['id'],
super(DeleteCheckboxWidget, self).render(name, False, attrs),
_('Delete')
)
if self.is_image:
s += u'<br><img src="%s%s" width="50">' % (settings.MEDIA_URL, value)
else:
s += u'<br><a href="%s%s">%s</a>' % (settings.MEDIA_URL, value, os.path.basename(value))
return s
else:
return u''
class RemovableFileFormWidget(forms.MultiWidget):
def __init__(self, is_image=False, initial=None, **kwargs):
widgets = (forms.FileInput(), DeleteCheckboxWidget(is_image=is_image, initial=initial))
super(RemovableFileFormWidget, self).__init__(widgets)
def decompress(self, value):
return [None, value]
class RemovableFileFormField(forms.MultiValueField):
widget = RemovableFileFormWidget
field = forms.FileField
is_image = False
def __init__(self, *args, **kwargs):
fields = [self.field(*args, **kwargs), forms.BooleanField(required=False)]
# Compatibility with form_for_instance
if kwargs.get('initial'):
initial = kwargs['initial']
else:
initial = None
self.widget = self.widget(is_image=self.is_image, initial=initial)
super(RemovableFileFormField, self).__init__(fields, required=False)
def compress(self, data_list):
return data_list
class RemovableImageFormField(RemovableFileFormField):
field = forms.ImageField
is_image = True
class RemovableFileField(models.FileField):
def delete_file(self, instance):
if getattr(instance, self.attname):
file_name = getattr(instance, 'get_%s_filename' % self.name)()
# If the file exists and no other object of this type references it,
# delete it from the filesystem.
if os.path.exists(file_name) and \
not instance.__class__._default_manager.filter(**{'%s__exact' % self.name: getattr(instance, self.attname)}).exclude(pk=instance._get_pk_val()):
os.remove(file_name)
def get_internal_type(self):
return 'FileField'
def save_form_data(self, instance, data):
if data and data[0]: # Replace file
self.delete_file(instance)
super(RemovableFileField, self).save_form_data(instance, data[0])
if data and data[1]: # Delete file
self.delete_file(instance)
setattr(instance, self.name, None)
def formfield(self, **kwargs):
defaults = {'form_class': RemovableFileFormField}
defaults.update(kwargs)
return super(RemovableFileField, self).formfield(**defaults)
class RemovableImageField(models.ImageField, RemovableFileField):
def formfield(self, **kwargs):
defaults = {'form_class': RemovableImageFormField}
defaults.update(kwargs)
return super(RemovableFileField, self).formfield(**defaults)
|
Comments
It sounds like a great snippet, but when I make a model with the following code:
No deletbox shows up in the admin website. Am I forgetting something?
#
This doesn't show up for me as well. Anything else that needs to be done to get it to show up?
#
note: only works with ModelForm and form_for_instance under newforms.
so the admin does not YET work with this. but it will.
#