Django模型中的动态选择字段

Naz*_*han 14 python django django-models django-forms

我的models.py:

SHOP1_CHOICES = (
    ('Food Court', 'Food Court'),
    ('KFC', 'KFC'),

)

SHOP2_CHOICES = (
    ('Sports Arena', 'Sports Arena'),
    ('Disco D', 'Disco D'),

)

SHOP3_CHOICES = (
    ('Bowling Arena', 'Bowling Arena'),
    ('Cinemax', 'Cinemax'),

)

class Feed(models.Model):
  gender = models.CharField(max_length=5, choices=GENDER_CHOICES, default='girl')
  name =models.CharField(max_length=25)
  shop=models.CharField(max_length=20)
  location=models.CharField(max_length=25, choices=SHOP1_CHOICES)
Run Code Online (Sandbox Code Playgroud)

在这里,如果Feed.shop == 'shop1'我想加载SHOP1_CHOICESFeed.location.目前无论什么商店,它只显示SHOP1_CHOICES(毫不奇怪).我如何实现它?我被卡住了,请帮忙.

dan*_*era 17

这是我的方法:

我懒惰加载使用懒惰:

from django.utils.functional import lazy
Run Code Online (Sandbox Code Playgroud)

在这里,帮助选择选项:

def help_SHOP_CHOICES():
    SHOP1_CHOICES = [
        ('Food Court', 'Food Court'),
        ('KFC', 'KFC'),
      ]
    SHOP3_CHOICES = [
        ('Bowling Arena', 'Bowling Arena'),
        ('Cinemax', 'Cinemax'),
      ]
    return random.choice( SHOP1_CHOICES + SHOP3_CHOICES )   # choose one
Run Code Online (Sandbox Code Playgroud)

最后是具有动态选择的模型:

class Feed(models.Model):
  ...
  location=models.CharField(max_length=25, choices=SHOP1_CHOICES)

  def __init__(self, *args, **kwargs):
     super(Feed, self).__init__(*args, **kwargs)
     self._meta.get_field('location').choices = \
                        lazy(help_SHOP_CHOICES,list)()
Run Code Online (Sandbox Code Playgroud)

  • @XavierBrassoud,太棒了。请删除评论以保持帖子干净。此致。 (2认同)

小智 10

来自Django文档:http://docs.djangoproject.com/en/dev/ref/models/fields/#choices

最后,请注意,选择可以是任何可迭代对象 - 不一定是列表或元组.这使您可以动态构造选择.但是如果你发现你自己选择了动态的选择,你可能最好使用一个带有ForeignKey的正确数据库表.选择适用于静态数据,如果有的话,变化不大.


Wil*_*ian 5

我不认为你应该在模型上做这个,形式是一个更好的地方.或者您应该重新考虑您的模型.例如:

class Location(models.Model):
    pass

class Shop(models.Model):
    location = models.ForeignKey(Location)

class Feed(models.Model):
     shop = models.ForeignKey()
Run Code Online (Sandbox Code Playgroud)