python - 根据 ModelChoiceField 获取 ModelMultipleChoiceField 字段的查询集

标签 python django forms django-forms

我需要表单,用户可以在其中根据所选标签选择游览。我通过 View 、自定义 html 表单和一些 AJAX 内容实现了它:

HTML 表单:

<form id="add-destination" method="post">
    {% csrf_token %}
    <select name="tag_id" id="tag_id">
        <option value=''>---</option>
        {% for tag in tags %}
        <option value={{ tag.id }}>{{ tag.name }}</option>
        {% endfor %}
    </select></br>
    <select multiple="multiple" name="tours" id="tours">
    </select></br>
    <button type="submit">Submit</button>
</form>

查看并按标签获取游览

def get_tours_by_tag(request):
    tag_id = Tag.objects.get(id=request.GET.get('tag_id', None))
    tours = Inbound.objects.filter(tags=tag_id)
    return JsonResponse({"status": "ok", "tours": serializers.serialize("json", list(tours))})

JS:

$(function(){
    $('#tag_id').change(function(e){
        $('#tours').html('');
        e.preventDefault();
        $.ajax({
            type: 'GET',
            data: $(e.currentTarget).serialize(),
            url: "/ru/tour/get_tours/",
            success: function(data){
               var tour_data = JSON.parse(data.tours);
               if(data.status=='ok'){
                   $.each(tour_data, function(i, item) {
                       $('#tours').append('<option value='+item['pk']+'>'+item['fields']['title']+'</option>');
                   });
               }else{
                 console.log("Not okay!")
               }
            }
        });
    })
});

这种方法工作正常,但我知道这不是解决此类问题的正确方法。我想通过使用 django 表单的力量来做到这一点。

所以我有模型游览:

class Tour(models.Model):
    title = models.CharField(max_length=255)
    tags = models.ManyToManyField(Tag)

和形式:

class FiveDestinationForm(forms.Form):
    tags = forms.ModelChoiceField(queryset=Tag.objects.all())
    tours = forms.ModelMultipleChoiceField(queryset=????)

    fields = ('tag', 'tours')

我发现了一些足够接近的问题,并尝试通过重写表单中的 __init__ 方法来解决我的问题,正如建议的那样:

class MyForm(forms.Form):
    tags = forms.ModelChoiceField(queryset=Tag.objects.all())
    tours = forms.ModelMultipleChoiceField(queryset=Tour.objects.none())

    def __init__(self, *args, **kwargs):
        qs = kwargs.pop('tours', None)
        super(MyForm, self).__init__(*args, **kwargs)
        self.fields['tours'].queryset = qs

    fields = ('tag', 'tours')

但我收到以下错误:

'NoneType' object has no attribute 'iterator'

所以我的问题是如何根据用户在 ModelChoiceField 中选择的内容正确设置 ModelMultipleChoiceField 的查询集?

最佳答案

您看到的错误很可能是因为表单没有收到“tours”参数(它将收到一个名为data的字典,其中包含未经验证的表单中的字段数据)因此 qs 变量将为 None。

你可能应该这样做:

class MyForm(forms.Form):
    tag = forms.ModelChoiceField(queryset=Tag.objects.all())
    tours = forms.ModelMultipleChoiceField(queryset=Tour.objects.none())

    def __init__(self, *args, **kwargs):
        super(MyForm, self).__init__(*args, **kwargs)

        if kwargs['data']:
            tag_id = kwargs['data'].get('tag', None)
            if tag_id:
                self.fields['tours'].queryset = Tour.objects.filter(tag_id=tag_id)

请注意,我们在这里稍微绕过了 django 的表单机制,需要自己解析 tag_id。不过这没什么大不了的。

从用户体验的角度来看,如果不使用 ajax,这是一个相当困难的问题。所以我也不会完全拒绝你的解决方案。

关于python - 根据 ModelChoiceField 获取 ModelMultipleChoiceField 字段的查询集,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/31000661/

相关文章:

django - Django如何重命名上传的文件?

python - 在 Django 模型字段中存储二进制哈希值

python - 同时运行多个python脚本

python - 仅调用 python 返回代码状态

django - Heroku Django-Admin 500 错误?

forms - 文本框文本更改事件 VBA 表单出错

html - Enter 上没有提交按钮的表单

forms - Symfony:使用类型 ='text' 和表单类型 'checkbox'

使用映射从类方法调用 NumPy 时出现 Python 错误

python - ValueError 函数内的 Pandas 函数