30👍
I just came through this.
class Profile(models.Model):
accounts = models.ManyToManyField('project.Account',
through='project.ProfileAccount')
def get_accounts(self):
return self.accounts.order_by('link_to_profile')
class Account(models.Model):
name = models.CharField(max_length=32)
class ProfileAccount(models.Model):
profile = models.ForeignKey('project.Profile')
account = models.ForeignKey('project.Account', related_name='link_to_profile')
number = models.PositiveIntegerField()
class Meta:
ordering = ('number',)
I removed the fields which were off-topic except of Account.name
.
That’s the shortest solution I’ve found, no idea if it was possible to use in 2010, but it certainly is now.
- [Django]-How to convert JSON data into a Python object?
- [Django]-Better ArrayField admin widget?
- [Django]-Django – How to get admin url from model instance
5👍
a ManyToManyField manager allows you to select/filter data from the related model directly, without using any model connection of the through model on your django level…
Likewise,
if you try:
pr = Profile.objects.get(pk=1)
pr.account.all()
returns you all account related to that profile. As you see, there exists no direct relation to the through model ProfileAccount, so you can not use the M2M relation at this point… You must use a reverse relation to the through model and filter the results…
pr = Profile.objects.get(pk=1)
pr.profileaccount_set.order_by('number')
will give you an ordered queryset, but, in this case, what you have in queryset is profileaccount objects, not account objects… So you have to use another django level relation to go to each related account with:
pr = Profile.objects.get(pk=1)
for pacc in pr.profileaccount_set.order_by('number'):
pacc.account
- [Django]-403 Forbidden error with Django and mod_wsgi
- [Django]-Dropdown in Django Model
- [Django]-How do I get the object if it exists, or None if it does not exist in Django?
5👍
Add the related name to ProfileAccounts and then change the ordering in Accounts with that ‘related_name__number’. Note two underscores between related_name and number. See below:
class Accounts(models.Model):
.
.
.
class Meta:
ordering = ('profile_accounts__number',)
class ProfileAccounts(models.Model):
.
.
.
account = models.ForeignKey('project.Accounts', related_name='profile_accounts')
number = models.PositiveIntegerField()
class Meta:
ordering = ('number',)
- [Django]-Virtualenv and source version control
- [Django]-Get name of primary field of Django model
- [Django]-How to remove white space from html source code
2👍
There’s a typo in the Profile (it’s “accout” when I think you mean “account”), but more importantly you have your singular/plural forms mixed up in the model.
In Django the practice is generally to name your classes singular, and your ManyToManyField names plural. So:
class Profile(models.Model):
name = models.CharField(max_length=32)
accounts = models.ManyToManyField(
'Account',
through='ProfileAccount'
)
def __unicode__(self)
return self.name
class Account(models.Model):
name = models.CharField(max_length=32)
type = models.CharField(max_length=32)
class Meta:
ordering = ('name',)
def __unicode__(self)
return self.name
class ProfileAccount(models.Model):
profile = models.ForeignKey(Profile)
account = models.ForeignKey(Account)
number = models.PositiveIntegerField()
class Meta:
ordering = ('number',)
I’m a little confused at what you’re trying to do with this model, but if you make those changes, then for acct_number in self.profile.accounts.all().order_by('number'):
should work. Assuming no other issues.
- [Django]-How to force Django Admin to use select_related?
- [Django]-Django Calendar Widget?
- [Django]-What does 'many = True' do in Django Rest FrameWork?
0👍
The easiest solution for this particular problem seems to be
for acct in self.profile.accounts.order_by('profileaccounts'):
pass
- [Django]-How to force migrations to a DB if some tables already exist in Django?
- [Django]-How to update fields in a model without creating a new record in django?
- [Django]-How to rename items in values() in Django?
0👍
I have this on a number of my models, but in my opinion (and unlike all the other answers) you shouldn’t need to specify the order_by
again because, well, it’s already specified in the through
model. Specifying it again breaks the DRY (don’t repeat yourself) principle.
I would use:
qs = profile.profileaccounts_set.all()
This gives the set of ProfileAccounts associated with a profile using your configured ordering. Then:
for pa in qs:
print(pa.account.name)
For bonus points, you can also speed up the overall process by using select_related
in the query.
- [Django]-Django – How to do tuple unpacking in a template 'for' loop
- [Django]-Apache not serving django admin static files
- [Django]-Expected view to be called with a URL keyword argument named "pk"