我是新的django和我试图连接不同的查询集从不同的模型基于这个答案在这里:https://stackoverflow.com/a/434755/4465450
当我试图在我的管理中查看它时,我一直得到这个错误。
'list' object has no attribute 'iterator'
我在form。py
中调用它 from django import forms
from itertools import chain
from content.models import Article, Tutorial
class DashboardChoiceForm(forms.Modelform):
dashboard_select = forms.ModelChoiceField(queryset=None)
def __init__(self, *args, **kwargs):
super(DashboardChoiceForm, self).__init__(*args, **kwargs)
article_list = Article.objects.all()
tutorial_list = Tutorial.objects.all()
self.fields['dashboard_select'].queryset = list(chain(article_list, tutorial_list))
我把它包括在我的管理页面,像这样:
from django.contrib import admin
from .models import *
from .forms import *
class RowSingleAdmin(admin.ModelAdmin):
model = RowSingle
form = DashboardChoiceForm
我想做的是在RowSingle admin上创建一个下拉菜单,以便用户可以选择一个内容项(文章或教程)显示在该行
这行不行:
self.fields['dashboard_select'].queryset = list(chain(article_list, tutorial_list))
问题不是来自右边,而是来自左边。ModelChoiceField
期望queryset
是一个实际的查询集,而不是一个列表。
如果您想提供一个列表,您应该使用另一个字段类型,即ChoiceField
。然后你需要分配选项。它们必须是一个二元组的列表,其中第一项是实际值,第二项是显示的文本。
它变得有点棘手,因为实际值必须是字符串,或者可以转换为字符串的东西。例如,我们可以为文章设置"article_
choices = chain(
('article_%d' % obj.pk, str(obj)) for obj in article_list),
('tutorial_%d' % obj.pk, str(obj)) for obj in tutorial_list),
)
self.fields['dashboard_select'].choices = choices
您可以将str(obj)
替换为向用户显示对象的任何合理方式。
然后当你的表单被验证时,你会得到值,你应该解析回:
from django.core.exceptions import ObjectDoesNotExist, ValidationError
# on your form class
def clean_dashboard_select(self):
data = self.cleaned_data['dashboard_select'] # will be, eg: "article_42"
try:
typename, value = data.split('_')
pk = int(value)
except ValueError:
raise ValidationError('cannot understand %s' % data)
try:
if typename == 'article':
return Article.objects.get(pk=pk)
if typename == 'tutorial':
return Tutorial.objects.get(pk=pk)
except ObjectDoesNotExist:
raise ValidationError('No %s with pk %d' % (typename, pk))
raise ValidationError('Never heard of type %s' % typename)
编辑(来自注释)-添加节组,像这样更改选择:
choices = (
('Articles', tuple(('article_%d' % obj.pk, str(obj)) for obj in article_list)),
('Tutorials', tuple(('tutorial_%d' % obj.pk, str(obj)) for obj in tutorial_list)),
)