Django/jQuery 级联 Select 管理框
Django/jQuery Cascading Select Boxes in Admin
我有一份合同 class,其中 contract_mod 允许从之前的合同扩展合同。 contract_mod 应该只显示与我们之前 select 编辑过的人相关的合同。
合同 class returns 字段 人 。因为我从未使用过 AJAX/jQuery 我不知道从哪里开始。
所以,我的目标是字段 contract_mod 取决于 hte 字段 person(使用管理界面)。
class Contract(models.Model):
person = models.ForeignKey(Person) #person hired
contract_mod = models.OneToOneField('self', blank = True, null = True) #allows to extend a contract
...
由于字段 contract_mod 是 OneToOneField 我不能使用 django-smart-selects 或 django-ajax-select
在类似情况下,我做了以下操作(现在已适应您的模型):
型号
class Person(models.Model):
name = models.CharField(max_length=20)
def __unicode__(self):
return self.name
def get_name(self):
return self.name
class Contract(models.Model):
person = models.ForeignKey(Person) #person hired
contract_mod = models.OneToOneField('self', blank = True, null = True)
contract_name = models.CharField(max_length=20) #just for testing
def __unicode__(self):
return self.get_name() + " " +self.contract_name
def get_name(self):
return self.person.get_name() #to make sure you get the person name in the admin
def contract_mod_name(self):
if self.contract_mod:
return self.contract_mod.contract_name
else:
return ""
管理员
class SelectField(forms.ChoiceField):
def clean(self, value):
return value
class CForm(forms.ModelForm):
contracts_from_selected = SelectField()
class Meta:
model = Contract
widgets = { 'contract_mod' : forms.widgets.Select(attrs={'hidden' : 'true'}) }
class CAdmin(admin.ModelAdmin):
form = CForm
list_display = ('contract_name','get_name','contract_mod_name')#what you like
def save_model(self, request, obj, form, change):
if request.POST.get('contracts_from_selected'):
obj.contract_mod=Contract.objects.get(id=int(request.POST.get('contracts_from_selected')))
obj.save()
覆盖 change_form.html 模板(通过将其从 django/contrib/admin/templates/admin 复制到您的 yourapp/templates/admin/yourapp 目录)并添加以下内容 Javascript:
$(function () {
$("#id_person").change(function () {
var options = $("#id_contract_mod option").filter(function () {
return $(this).html().split(" ")[0] === $("#id_person option:selected").html();
}).clone();
$("#id_contracts_from_selected").empty();
$("#id_contracts_from_selected").append(options);
});
});
一个缺点是,它使用可见的 html 条目来存储人-合同关系。所以这个人在下拉列表中也是可见的。为避免这种情况,您可以向选项添加一个属性 - 请参见此处:
Django form field choices, adding an attribute
是的,最好完全隐藏 contract_mod ChoiceField。通过小部件中的 hidden=True 仅隐藏下拉菜单。
我有一份合同 class,其中 contract_mod 允许从之前的合同扩展合同。 contract_mod 应该只显示与我们之前 select 编辑过的人相关的合同。 合同 class returns 字段 人 。因为我从未使用过 AJAX/jQuery 我不知道从哪里开始。
所以,我的目标是字段 contract_mod 取决于 hte 字段 person(使用管理界面)。
class Contract(models.Model):
person = models.ForeignKey(Person) #person hired
contract_mod = models.OneToOneField('self', blank = True, null = True) #allows to extend a contract
...
由于字段 contract_mod 是 OneToOneField 我不能使用 django-smart-selects 或 django-ajax-select
在类似情况下,我做了以下操作(现在已适应您的模型):
型号
class Person(models.Model):
name = models.CharField(max_length=20)
def __unicode__(self):
return self.name
def get_name(self):
return self.name
class Contract(models.Model):
person = models.ForeignKey(Person) #person hired
contract_mod = models.OneToOneField('self', blank = True, null = True)
contract_name = models.CharField(max_length=20) #just for testing
def __unicode__(self):
return self.get_name() + " " +self.contract_name
def get_name(self):
return self.person.get_name() #to make sure you get the person name in the admin
def contract_mod_name(self):
if self.contract_mod:
return self.contract_mod.contract_name
else:
return ""
管理员
class SelectField(forms.ChoiceField):
def clean(self, value):
return value
class CForm(forms.ModelForm):
contracts_from_selected = SelectField()
class Meta:
model = Contract
widgets = { 'contract_mod' : forms.widgets.Select(attrs={'hidden' : 'true'}) }
class CAdmin(admin.ModelAdmin):
form = CForm
list_display = ('contract_name','get_name','contract_mod_name')#what you like
def save_model(self, request, obj, form, change):
if request.POST.get('contracts_from_selected'):
obj.contract_mod=Contract.objects.get(id=int(request.POST.get('contracts_from_selected')))
obj.save()
覆盖 change_form.html 模板(通过将其从 django/contrib/admin/templates/admin 复制到您的 yourapp/templates/admin/yourapp 目录)并添加以下内容 Javascript:
$(function () {
$("#id_person").change(function () {
var options = $("#id_contract_mod option").filter(function () {
return $(this).html().split(" ")[0] === $("#id_person option:selected").html();
}).clone();
$("#id_contracts_from_selected").empty();
$("#id_contracts_from_selected").append(options);
});
});
一个缺点是,它使用可见的 html 条目来存储人-合同关系。所以这个人在下拉列表中也是可见的。为避免这种情况,您可以向选项添加一个属性 - 请参见此处: Django form field choices, adding an attribute
是的,最好完全隐藏 contract_mod ChoiceField。通过小部件中的 hidden=True 仅隐藏下拉菜单。