23

models.py

class SinglePoint(models.Model):
    attributes = models.TextField(blank=True)
    name = models.CharField(max_length=100)
    geom = models.PointField() #Kartenposition
    objects = models.GeoManager()

class Connection(models.Model):
    name = models.CharField(max_length=100)
    #points = models.ManyToManyField(SinglePoint) #OLD
    p1 = models.ForeignKey(SinglePoint, related_name='p1_set') #NEW
    p2 = models.ForeignKey(SinglePoint, related_name='p2_set') #NEW
    obs = models.ManyToManyField(Observation, blank=True)
    conds = models.ManyToManyField(Condition, blank=True)
    objects = models.GeoManager()

class Meta:
    order_with_respect_to = 'p1'

In my view.py:

...
p1_points = SinglePoint.objects.filter(p1_set__vektordata__order__project__slug=slug)
p2_points = SinglePoint.objects.filter(p2_set__vektordata__order__project__slug=slug)
...

Before I switched to ForeignKey, it worked with:

points = SinglePoint.objects.filter(connection__vektordata__order__project__slug=slug)

How to 'join' these two QuerySets to one QuerySet and make a distinct()?

Thanks!

4 Answers4

72

It took me a while to find this

all_points = p1_points | p2_points
Krystian Cybulski
  • 10,325
  • 11
  • 66
  • 97
9

I am not familiar with geodjango, but combining QuerySets into one QuerySet is possible via the Q-Object and Boolean Operators. See http://docs.djangoproject.com/en/dev/topics/db/queries/#complex-lookups-with-q-objects

Example:

Q(p1_points) | Q(p2_points)

I can't help you further, because I am not really sure what you are trying to accomplish.

stefanw
  • 10,208
  • 3
  • 34
  • 34
  • I want to get all SinglePoint instances in my Project without double SinglePoint instances (distinct). It was no problem with a ManyToMany-Field, because then I accessed them with .filter(connection__vektordata__ ...).distinct() and now I have two Filds: .filter(p1_set__vektordata__ ... and .filter(p2_set__vektordata__ ...) –  Jul 14 '09 at 16:03
  • Sorry, I meant "Fields" not "Filds" –  Jul 14 '09 at 16:04
  • results = Q(p1_points) | Q(p2_points) results.children gives me [[, ], [, ]] but I wish to have [, , , ] to call distinct() –  Jul 14 '09 at 16:07
6

I think Q queries can achieve what you need like this:

points = SinglePoint.objects.filter(
    Q(p1_set__vektordata__order__project__slug=slug) |
    Q(p2_set__vektordata__order__project__slug=slug)
).distinct()
JoshB
  • 701
  • 1
  • 8
  • 12
3

p1_points.union(p2_points)

See Docs

techkuz
  • 3,028
  • 4
  • 27
  • 58
  • The problem with `union()` is that it restricts the actions to the resulting queryset. For instance, you can't `.filter()` or `.annotate()` which is a pretty big drawback. Read more about it [in the docs](https://docs.djangoproject.com/en/4.0/ref/models/querysets/#union). – Austin A Jan 20 '22 at 00:20