У меня есть две модели, Room
и Image
. Image
- это общая модель, которая может быть привязана к любой другой модели. Я хочу дать пользователям форму для загрузки изображения, когда они публикуют информацию о комнате. Я написал код, который работает, но я боюсь, что я сделал это трудным путем, и особенно таким образом, который нарушает DRY.
Надеялся, что кто-то, кто немного более знаком с формами Джанго, сможет указать, где я ошибся.
Обновление:
Я попытался уточнить, почему я выбрал этот дизайн в комментариях к текущим ответам. Подведем итог:
Я не просто поместил ImageField
на модель Room
, потому что мне нужно было более одного изображения, связанного с моделью комнаты. Я выбрал общую модель изображения, потому что хотел добавить изображения к нескольким различным моделям. Альтернативы, которые я рассматривал, были несколькими внешними ключами в одном классе Image
, который казался грязным, или несколькими классами Image
, которые, как я думал, загромождали мою схему. Я не прояснил это в своем первом посте, так что извините за это.
Поскольку ни один из ответов до сих пор не касался того, как сделать это немного более СУХИМ, я придумал свое собственное решение, которое заключалось в добавлении пути загрузки в качестве атрибута класса в модель изображения и ссылки на то, что каждый раз необходимо.
# Models
class Image(models.Model):
content_type = models.ForeignKey(ContentType)
object_id = models.PositiveIntegerField()
content_object = generic.GenericForeignKey('content_type', 'object_id')
image = models.ImageField(_('Image'),
height_field='',
width_field='',
upload_to='uploads/images',
max_length=200)
class Room(models.Model):
name = models.CharField(max_length=50)
image_set = generic.GenericRelation('Image')
# The form
class AddRoomForm(forms.ModelForm):
image_1 = forms.ImageField()
class Meta:
model = Room
# The view
def handle_uploaded_file(f):
# DRY violation, I've already specified the upload path in the image model
upload_suffix = join('uploads/images', f.name)
upload_path = join(settings.MEDIA_ROOT, upload_suffix)
destination = open(upload_path, 'wb+')
for chunk in f.chunks():
destination.write(chunk)
destination.close()
return upload_suffix
def add_room(request, apartment_id, form_class=AddRoomForm, template='apartments/add_room.html'):
apartment = Apartment.objects.get(id=apartment_id)
if request.method == 'POST':
form = form_class(request.POST, request.FILES)
if form.is_valid():
room = form.save()
image_1 = form.cleaned_data['image_1']
# Instead of writing a special function to handle the image,
# shouldn't I just be able to pass it straight into Image.objects.create
# ...but it doesn't seem to work for some reason, wrong syntax perhaps?
upload_path = handle_uploaded_file(image_1)
image = Image.objects.create(content_object=room, image=upload_path)
return HttpResponseRedirect(room.get_absolute_url())
else:
form = form_class()
context = {'form': form, }
return direct_to_template(request, template, extra_context=context)