首页
学习
活动
专区
圈层
工具
发布
社区首页 >问答首页 >在动态设置的MultipleChoiceField上进行验证,其中字段最初是空的?

在动态设置的MultipleChoiceField上进行验证,其中字段最初是空的?
EN

Stack Overflow用户
提问于 2011-08-25 22:54:03
回答 2查看 1.8K关注 0票数 2

我有一个视图,里面有几种不同的形式。第一个表单有一个搜索字段,该字段填充多项选择字段,当用户在搜索字段中输入他们的查询时,将发送一个AJAX调用,以获取与查询匹配的所有记录。然后,用户从第一个多项选择字段中选择选项,并单击“添加”将它们移动到不同的多选框中

这可以很好地工作,但是当表单提交时,我收到一个错误,显示"Select a valid choice.1 is not of the available one .“。我尝试在接收到init形式的参数后设置选项,但似乎不起作用。

我的表单:

代码语言:javascript
复制
class SiteCoordinatorForm(forms.ModelForm):
    selected_studies = forms.MultipleChoiceField(required = False)
    site = forms.ChoiceField(required = False)    
    studies = forms.MultipleChoiceField(required = False)
    study_search = forms.CharField(max_length = 50, required = False)

    def __init__(self, *args, **kwargs):
        super(SiteCoordinatorForm, self).__init__(*args, **kwargs)        
        if args:
            study_list = []
            query_dict = args[0]
            self.fields['selected_studies'].choices = [(int(x), x) for x in   query_dict.getlist('selected_studies')]

        self.fields['site'].choices = [(x.pk, "%s (%s)" % (x.primary_name, x.primary_number)) for x in Site.objects.all().order_by('primary_name')]

    class Meta:
        model = SiteCoordinator
        exclude = ('studies', 'site', 'selected_studies')

我的AJAX函数填充了这个框:

代码语言:javascript
复制
def search_studies(request):
    return_data = {}
    studies = []
    make_query = lambda terms, fieldname: reduce(lambda x, y: x & Q(**{fieldname + '__icontains': y}), terms, Q())
    if 'search_text' in request.POST:
        terms = request.POST['search_text'].split()
        for rec in Study.objects.filter(make_query(terms, 'name')):
            studies.append({
                'study': {'id': rec.id, 'name': rec.name, 'number': rec.id}
            })

    response = {'status': 'success', 'count': len(studies), 'studies': studies}
    return HttpResponse(simplejson.dumps(response), mimetype="application/json")

填充MultiSelectBox的函数:

代码语言:javascript
复制
function show_results(data, status_code, request){
    recs = data.studies;
    var select_box = document.getElementById('id_studies');
    select_box.options.length = 0;

    for (var index = 0; index < recs.length; index ++){
        select_box.options[index] = new Option(recs[index].study.name,
                        recs[index].study.id,
                        false, false);
    }
}

然后是移动选项的JQuery:

代码语言:javascript
复制
$('#add').click(function() {
    $('#id_studies option:selected').remove().appendTo('#id_selected_studies');
    return false;

});

$('#remove').click(function() {  
    $('#id_selected_studies option:selected').remove().appendTo('#id_studies');
    return false;
});
EN

回答 2

Stack Overflow用户

回答已采纳

发布于 2011-10-31 22:40:03

很明显,我们不需要MultipleChoiceField的验证,但确实需要MultipleSelectWidget,因此解决方案只需将字段更改为

代码语言:javascript
复制
studies = forms.CharField(widget=forms.SelectMultiple)

中进行您自己的验证

代码语言:javascript
复制
def clean_studies(self):
    ...
票数 5
EN

Stack Overflow用户

发布于 2015-11-26 23:00:16

下面是一个例子,它使用了@AnuragUniyal提出的解决方案,也回答了@JohnLehmann的评论:

代码语言:javascript
复制
class MyForm(forms.Form):
    new_tags = forms.CharField(required=False, widget=forms.SelectMultiple)

    def save(self, commit=True):
        instance = super(MyForm, self).save(commit=commit)
        for tag in eval(self.cleaned_data['new_tags']):
            models.Tag.objects.get_or_create(name=tag)
        return instance

基本上,我们将unicode字符串列表的字符串转换回Python列表,并对其应用eval()

票数 0
EN
页面原文内容由Stack Overflow提供。腾讯云小微IT领域专用引擎提供翻译支持
原文链接:

https://stackoverflow.com/questions/7192540

复制
相关文章

相似问题

领券
问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档