Как обновить поле в модели, подключенной через внешний ключ

#python #django #forms #view #django-class-based-views

#python #django #формы #Вид #представления на основе классов django

Вопрос:

Я работаю над представлением обновления, где я хочу обновить размер футболки, щелкнув по элементу в списке просмотра заказа. Я сложил в точку, где я могу отображать только имя элемента, но не обновлять размер футболки. Это django 3.

Пожалуйста, дайте подсказку, как я могу обновить размер футболки, связанной внешним ключом, с моделью заказа.

models.py

 class Size(models.TextChoices):
    SMALL = 'S', 'Small'
    MEDIUM = 'M', 'Medium'
    LARGE = 'L', 'Large'


class Tshirt(models.Model):
    name = models.CharField(max_length=255)
    size = models.CharField(max_length=2, choices=Size.choices, default=Size.MEDIUM)

    def __str__(self):
        return self.name

class Order(models.Model):
    nr = models.PositiveIntegerField(unique=True)
    item = models.ForeignKey(Tshirt, related_name='order_tshirts', on_delete=models.CASCADE)
    send_date = models.DateTimeField(null=True, blank=True)
    
    def get_update_size_url(self):
        return reverse('size_update', kwargs={'pk': self.pk})
 

urls.py

 path('update/<int:pk>/', TshirtSizeUpdateView.as_view(), name='size_update'),
 

views.py

 class TshirtSizeUpdateView(UpdateView):
    model = Order
    template_name = 'size_update.html'
    fields = ['item']
    success_url = '/'
 

forms.py

 class OrderForm(forms.ModelForm):
    class Meta:
        model = Order
        fields = '__all__'
```


update template
```
    <form method="post">
        {% csrf_token %}
        {% include 'includes/bs4_form.html' with form=form %}
        <input type="submit" class="btn btn-primary mb-2" value="Update">
    </form>
```
 

Комментарии:

1. пожалуйста, покажите свой шаблон

2. @ha-neul пожалуйста, проверьте мое редактирование

3. Ваш шаблон не предназначен для обновления. Вы вставили неправильный?

4. Кроме того, пожалуйста, покажите свой вид создания и шаблон создания

5. @ha-neul исправлено, да, это был шаблон списка. Нет CreateView, я управлял этим с помощью панели администратора.

Ответ №1:

Для этого случая вы можете использовать formset.

в forms.py

 from django.forms.models import inlineformset_factory

TshirtFormset = inlineformset_factory(
    Order, Tshirt, fields=('name', 'size')
)
 

в views.py

 class OrderUpdateView(UpdateView):
    model = Order
    fields = ["nr", "item", "send_date"]

    template_name = 'size_update.html'
    success_url = '/'

    def get_context_data(self, **kwargs):
        
        context = super(OrderUpdateView, self).get_context_data(**kwargs)
        if self.request.POST:
            context["tshirt_form"] = TshirtFormset(self.request.POST, instance=self.object)
        else:
            context["tshirt_form"] = TshirtFormset(instance=self.object)

        return context

    def form_valid(self, form):
        context = self.get_context_data()
        tshirt_form = context["tshirt_form"]
        self.object = form.save()
        if tshirt_form.is_valid():
            tshirt_form.instance = self.object
            tshirt_form.save()
        return super(OrderUpdateView, self).form_valid(form)

 

в вашем шаблоне поместите как форму заказа, так и набор форм для футболки.

 <form method="post">{% csrf_token %}
    {{ form.as_p }}
    <h2>T-shirt detail</h2>
    {{ tshirt_form.as_p }}
    <input type="submit" value="Save">
</form>