Let’s see if I can explain myself, I have this models:
class BillHeader(models.Model):
number = models.CharField(_('Bill number'), max_length=10, unique=True, \
default=__number)
client = models.ForeignKey(ClienteYProveedor, verbose_name=_('Client'))
date = models.DateTimeField(_('Date'), default=datetime.now)
def __unicode__(self):
return str(self.number)
class Meta:
abstract = True
class BillFooter(models.Model):
base_import = models.DecimalField(_('Base import'), max_digits=12, \
decimal_places=2)
class Meta:
abstract = True
class BillBody(models.Model):
description = models.CharField(_('Description'), max_length=200)
amount = models.DecimalField(_('Amount'), max_digits=6, decimal_places=2)
discount = models.DecimalField(_('Discount'), max_digits=4, \
decimal_places=2)
price = models.DecimalField(_('Price'), max_digits=12, decimal_places=2)
unitaryprice = models.DecimalField(_('Unitary Price'), max_digits=12, \
decimal_places=2)
def __unicode__(self):
return self.description
class Meta:
abstract = True
class EmittedBill(BillHeader, BillBody, BillFooter):
pass
class ReceivedBill(BillHeader, BillBody, BillFooter):
pass
When the user adds an Emmited or Received bill I need to show BillHeader as a normal fieldset, but BillBody and BillFooter need to be TabularInline.
If I put those as TabularInline in admin.py, an error rises saying that they need a ForeignKey to the related models. Of course, I can’t put those foreign keys, because they are declared at the bottom. I think you guys call this “backwards foreign keys”.
My question is this: how can I do this to show TabularInlines in the admin without making a mess?. I can do it without abstract base classes, but then another problem comes, it shows the other ForeignKey in the TabularInline (if you are on EmmitedBills it shows the FK to ReceivedBills in the TabularInline and viceversa) and I couldn’t been able to exclude them.
Sorry if this is a stupid question, but I’m not a programmer (it’s just a hobby) and I’m really making myself a mess with data models.
I’ll explain better:
I have two types of bills, Emitted and Received and both of them show on the admin home (that’s why I didn’t use a BooleanField to mark them). Both types have the same fields except one, the bill number, which in Emmitted will be autogenerated. Each bill consists on 1 header with number, client and date, 1 or more body inline entries with a description, amount, price, etc. and 1 inline footer, showing the total price without taxes, applied taxes, etc.
Update
I have done everything, but I have a problem, in my new model BillBody has two FK’s (EmmitedBill and ReceivedBill) and they show up in the TabularInline. How can I hide them?field.exclude() gives an error.
I don’t fully understand your question, but you can use
instead of
if the
ModelNamemodel isn’t already declared. Maybe this solves your problem.Inline admins (like
TabularInline) are only used when you have a one-to-many relation, which is created by aForeignKeyon the many side. If you don’t have such a foreign key, then you cannot use a inline admin. Inheritance is definitely different from aForeignKey.However, I think your data model is wrong. It seems like you do want to store bills. There are two types of bills,
emittedandreceivedbills. Bothemittedandreceivedbills do have the same fields. Furthermore, you want that each bill consists of a header with number, client and date, 1 or more body entries, where each entry stores the information you store inBillBodyand 1 or more decimalsbase_number.A probably better data model for you
I have left out the
__unicode__methods.Now you have a foreign key from
BillEntrytoBilland you can use a tabular inline. I didn’t understand your usage ofbase_importso I left this out.Price computation
If your
priceshould always equal something likeamount*unitaryprice - discountoramount*(unitaryprice-discount)then you shouldn’t put this in a field but either compute it when it is needed, either in Python or in the database. If you want to do this in Python you can use a method similar toget_total_price. If you want to compute them when querying the database then it is a little bit more difficult to get it working with Django.In the last case, you can have a look at SQL views, but I think this is a little bit too difficult for a beginner. Another option is to use a custom SQL expression:
This will compute the price for all entries during selection.
Update
If you add two subclasses for emitted and received bills and use multi table inheritance then you can use one foreign key from
BillEntrytoBill.You probably also have to think about the database model generated by Django. Usually, you only want to store elementary data in the database, and not computed data (like you want to do in your footer). So, if prices are computed using some formula and using the
unitaryprice,amountetc. you shouldn’t store the result of this formula but recompute it when necessary (and eventually cache to avoid re-computations). If you don’t do this, chances are that you at some moment update something (for example theamount) and forget to update the computed values (theprice) which leads to inconsistencies in your database (and thus bugs in your application). A good database does have constraints so that it is impossible to store inconsistent database without breaking at least one constraint.I also don’t see why you want a separate header and footer per bill. A model isn’t the real bill, it stores the information for a bill. If you want to have a visible header and footer, then you should do this in your view layer (the template) and not in the model itself.