1

假设我正在制作一个“如何”的Django webapp,用户可以在其中发布关于如何做不同事情的帖子。

  • “如何”制作绳索
  • “如何”制作一个土锅
  • “如何”学习骑自行车

你明白了。我已经为此创建了帖子创建视图。现在,当成员发布帖子时。他们在帖子中添加了其他图像

示例: “如何”制作绳索

  • 这有帖子标题=如何制作绳索
  • 帖子描述=“一些描述”
  • 发布图片 = 主图片

现在他们必须一步一步地展示绳子的制作过程

  • 图 1:第一次执行此操作
  • 图 2:第二次执行此操作

我正在使用 Django 表单集和我的帖子模型来实现这一点。在创建视图中一切正常。没问题。但在更新视图中,事情发生了变化。

问题

问题是当用户想要编辑他们的帖子并将图像号 2. 从他们的帖子切换到不同的图像时。即使他们改变了第二张图片。该图像现在位于列表的最后。使用户重新上传所有图像。恢复秩序。让我的应用看起来有问题。

示例:假设用户有以下帖子

main post Title 
" Some description "
Main Image = Post_image.jpg  

1st Image = A.jpg
   Image Title
   Image description
2nd Image = B.jpg
   Image Title
   Image description
3rd Image = C.jpg
   Image Title
   Image description
4st Image = D.jpg
    Image Title
     Image description
5th Image = E.jpg
     Image Title
     Image description
6th Image = F.img
     Image Title
     Image description

现在,如果我将第二张图片更改B.jpgb.jpg b.jpg 移动到列表的最后,并且您的顺序为 A、C、D、E、F、b

以下是我的模型:

 class Post(models.Model):
    user = models.ForeignKey(User, related_name='posts')
    created_at = models.DateTimeField(auto_now_add=True)
    title = models.CharField(max_length=250, unique=True)
    slug = models.SlugField(allow_unicode=True, unique=True,max_length=500)
    post_image = models.ImageField()
    message = models.TextField()

class Prep (models.Model): #(Images)
    post = models.ForeignKey(Post, on_delete=models.CASCADE, related_name='post_prep')
    image = models.ImageField(upload_to='images/', blank=True, null=True, default='')
    image_title = models.CharField(max_length=100, default='')
    image_description = models.CharField(max_length=250, default='')
    sequence = models.SmallIntegerField() ###########################ADDED THIS

    class Meta:    ###########################ADDED THIS
    unique_together = (('post', 'sequence'),) ###########################ADDED THIS
    ordering = ['sequence']  ###########################ADDED THIS

我的帖子创建视图

def post_create(request):
    ImageFormSet = modelformset_factory(Prep, fields=('image', 'image_title', 'image_description'), extra=12, max_num=12,
                                        min_num=2)
    if request.method == "POST":
        form = PostForm(request.POST or None, request.FILES or None)
        formset = ImageFormSet(request.POST or None, request.FILES or None)
        if form.is_valid() and formset.is_valid():
            instance = form.save(commit=False)
            instance.user = request.user
            instance.save()
            post_user = request.user
            for index, f in enumerate(formset.cleaned_data): #######CHANGED THIS
                try: ##############CHANGED THIS
                    photo = Prep(sequence=index, post=instance, image=f['image'], 
                             image_title=f['image_title'], image_description=f['image_description'])
                    photo.save()
                except Exception as e:
                    break

            return redirect('posts:single', username=instance.user.username, slug=instance.slug)
    else:
        form = PostForm()
        formset = ImageFormSet(queryset=Prep.objects.none())
    context = {
        'form': form,
        'formset': formset,
    }
    return render(request, 'posts/post_form.html', context)

我的帖子编辑视图:

class PostPrepUpdate(LoginRequiredMixin, UpdateView):
    model = Post
    fields = ('title', 'message', 'post_image')
    template_name = 'posts/post_edit.html'
    success_url = reverse_lazy('home')

    def get_context_data(self, **kwargs):
        data = super(PostPrepUpdate, self).get_context_data(**kwargs)
        if self.request.POST:
            data['prep'] = PrepFormSet(self.request.POST, self.request.FILES, instance=self.object)
        else:
            data['prep'] = PrepFormSet(instance=self.object)
        return data

    def form_valid(self, form):
        context = self.get_context_data()
        prep = context['prep']
        with transaction.atomic():
            self.object = form.save()

            if prep.is_valid():
                prep.instance = self.object
                prep.save()
        return super(PostPrepUpdate, self).form_valid(form)

我的表格.py

class PostEditForm(forms.ModelForm):
    class Meta:
        model = Post
        fields = ('title', 'message', 'post_image', 'group', )


class PrepForm(forms.ModelForm): #####################CHANGED THIS
    class Meta:
        model = Prep
        fields = ('image', 'image_title', 'image_description', 'sequence')


PrepFormSet = inlineformset_factory(Post, Prep, form=PrepForm, extra=5, max_num=7, min_num=2)

***需要帮助解决此问题。例如,如果他们更改图像 2。那么它应该保持在第 2 位,而不是移动到列表的末尾

4

1 回答 1

1

目前,您不保存图像的顺序,因为它们以与创建时相同的顺序显示。在图像序列中添加一个Prep包含图像位置的字段将有助于:

class Prep (models.Model):
    # ...
    nr = SmallIntegerField()
    #...
    class Meta:
        unique_together = (('post', 'nr'),)

unique_together约束确保每个数字在每个帖子中仅使用一次。这也允许在不删除和重新创建所有Prep对象的情况下对帖子中的图像进行重新排序。

在显示帖子时,您必须按 . 对Prep对象进行排序nr


至于填充新列,因为没有一个有意义的默认值,最简单的方法可能是:

  • 添加没有约束的nr字段,并且首先添加;迁移更改。unique_togethernull=True
  • 迁移后,按当前顺序循环Prep每个现有对象,Post并按升序分配它们。
  • 之后 remove null=True,添加unique_together并再次迁移。

unique_together需要字符串参数(它不能访问外部类中的字段);谢谢你明白了。


在编辑表单上,您希望包含新字段,以便用户可以通过交换两个图像的索引来交换两个图像的顺序。如果他们使用重复索引,您只需向他们提供有意义的错误消息。

但是,在创建时,您的用户不需要明确指定顺序,因为它隐含在表单集中的图像序列中。所以我建议像这样改变你的 for 循环:

for index, f in enumerate(formset.cleaned_data):
    # ...
    photo = Prep(nr=index,
                 # add all other fields
                )

用于nr = index + 1以 1 开头的人性化索引。事实上,indexorimage_index可能是比 . 更好的字段名称nr

于 2019-03-14T14:24:05.867 回答