我有一个产品模型,其中包含对Option模型的M2M参考,使用中间模型OptionPrice,它允许我使用特定于产品的值覆盖选项的基本值。
以下是三种模式:
#~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
class Product(models.Model):
productid = UUIDField(db_column='ProductID', editable=False, primary_key=True, auto=True, hyphenate=True)
productname = models.CharField(db_column='ProductName', max_length=100, verbose_name='name', default='')
slug = AutoSlugField(populate_from='productname',unique=True, max_length=100, always_update=True)
productnumber = models.CharField(db_column='ProductNumber', blank=True, max_length=50, null=True, verbose_name='number', default='')
cost = models.DecimalField(decimal_places=4, max_digits=19, db_column='Cost', default=0)
priceperitem = models.DecimalField(decimal_places=4, max_digits=19, db_column='PricePerItem', verbose_name='price per item', default=0)
onspecial = models.BooleanField(db_column='OnSpecial', verbose_name='on special', default=False)
discount = models.FloatField(null=True, db_column='Discount', blank=True, verbose_name='discounted', default=0)
#connections to other models
options = models.ManyToManyField(Option, null=True, through='OptionPrice', verbose_name='product options', blank=True)
optiongroups = models.ManyToManyField(OptionGroup, null=True, verbose_name='product option groups', blank=True)
package = models.ForeignKey(ShippingPackage, verbose_name='shipping container', default=1)
categories = models.ManyToManyField(Category, verbose_name='product categories')
cross_sell = models.ManyToManyField('Product', verbose_name='cross-sell items', blank=True)
#~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
class Option(models.Model):
optionid = UUIDField(db_column='OptionID', editable=False, primary_key=True, auto=True, hyphenate=True)
title = models.CharField(db_column='Title', max_length=500)
price = models.DecimalField(decimal_places=4, max_digits=19, db_column='Price')
pricemodstyle = models.CharField(db_column='PriceModStyle', max_length=50, verbose_name='price modification style', choices=PRICE_MOD_CHOICES)
displayrank = models.IntegerField(db_column='DisplayRank', verbose_name='sort order')
optiongroup = models.ForeignKey('OptionGroup', verbose_name='option group')
#~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
class OptionPrice(models.Model):
optionpriceid = UUIDField(db_column='OptionPriceID', editable=False, primary_key=True, auto=True, hyphenate=True)
option = models.ForeignKey(Option)
product = models.ForeignKey(Product)
price = models.DecimalField(decimal_places=4, max_digits=19, db_column='Price')
pricemodstyle = models.CharField(db_column='PriceModStyle', max_length=50, verbose_name='price modification style', choices=PRICE_MOD_CHOICES)
,对于admin.py,我将产品选项定义为内联
#~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
class ProductAdmin(ImportExportModelAdmin):
resource_class = ProductResource
model=Product
ordering = ['categories__categoryname','productname']
save_on_top = True
save_as = True
inlines = [OptionPriceAdminInline]
#~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
class OptionPriceAdminInline(admin.TabularInline):
model=OptionPrice
extra = 0
save_on_top = True
我的问题是,当我在管理员中加载产品时,内联显示得很漂亮,显示除主键(optionpriceid)之外的每个字段。因此,我第一次向此产品添加选项时,它们会正确保存(自动创建新ID)。如果我然后尝试重新加载产品并编辑已保存的选项,则会出现 MultiValueDictKeyError ,因为这些内联项实际上并未将其主键值包含在管理员发出的行中。
我做错了什么?
答案 0 :(得分:0)
该问题原来是django bug,由this pull request解决。问题是我使用自定义UUIDField作为主键 - 引用票证:
似乎这个问题发生在主键字段不是基于AutoField的模型(例如CharField)上。
当主键字段不是基于AutoField时,'has_auto_field'未设置为True,因此它不会在表单中输出主键。