如何使用动态过滤器制作 Django 查询集?

How to make django querysets with dynamic filters?

所以基本上我在 Django 中有一个网站,它是一个店面,最终用户可以使用三个过滤器。产品类型过滤器(裤子、鞋子、衬衫等)、送货过滤器 (yes/no) 和 location/popularity 过滤器。

目前我的views.py有这个方法

if request.is_ajax():
    if request.GET.get('filter') == 'shirts':
        latest_entries = Entry.objects.filter(entrytype="shirts")
        context = {'latest_entries': latest_entries}
        return render(request, 'storefrontload.html', context)
    if request.GET.get('filter') == 'pants':
        latest_entries = Entry.objects.filter(entrytype="pants")
        context = {'latest_entries': latest_entries}
        return render(request, 'storefrontload.html', context)
    if request.GET.get('filter') == 'shoes':
        latest_entries = Entry.objects.filter(entrytype="shoes")
        context = {'latest_entries': latest_entries}
        return render(request, 'storefrontload.html', context)

如您所见,这处理了第一个过滤器。我遇到的问题是,如果我 select,比方说 'pants',它会按裤子过滤,但会忽略其他两个过滤器中 select 的内容。另一个例子,假设我 select 裤子,页面填充了该过滤器的结果。但是,如果我随后转到交付过滤器和 select "Yes",该页面填充了仅可交付的项目,但忘记了 "pants" 过滤器。

我想知道的是如何在我的视图中创建一个查询集来记住其他两个查询集(如果这有意义的话)。

我能想到的唯一方法是为每个过滤器中的每个值创建 True/False 标志,然后添加大概 100 行 if/then 语句来检查每个标志。必须有更好的方法。

更新:

这就是我从模板传递过滤器值的方式。

function filter(type) {
  $.get("/storefront/?filter="+type, function(data) {
     var $data = data;
     $('.grid').children().remove();
     $('.grid').append( $data ).masonry( 'appended', $data, true ).masonry( 'layout' ); 
  });
}


  //Product Filter    
$("#shirts").unbind().click(function () { 
  filter("shirts");
  return false;
});


$("#pants").unbind().click(function () { 
  filter("pants");
  return false;
});

$("#shoes").unbind().click(function () { 
  filter("shoes");
  return false;
});



  //Delivery Filter
$("#deliveryyes").unbind().click(function () { 
  filter("deliveryyes");
  return false;
});

$("#deliveryno").unbind().click(function () { 
  filter("deliveryno");
  return false;
});

在我的 views.py 中,这行不通:

 entry_types = request.GET.getlist('filter')
 latest_entries = Entry.objects.filter(entrytype__in=entry_types)

因为我需要按条目类型('pants'、'shirts'、鞋子')和交付选项('deliveryyes'、'deliveryno')进行过滤。每个过滤器在我的模型中都有自己的列。

models.py

class Entry(models.Model):
    headline= models.CharField(max_length=200,)
    body_text = models.TextField()
    author=models.ForeignKey(settings.AUTH_USER_MODEL, related_name='entryauthors')
    pub_date=models.DateTimeField(auto_now_add=True)
    zipcode =models.IntegerField(null=True, max_length=10)
    !!! entrytype = models.CharField(null=True, max_length=10)
    !!! deliveryoption=models.CharField(null=True, max_length=5)

一个参数在一个请求中出现多次是完全有效的。使用 QueryDict.getlist() 获取包含所有值的列表。

http://example.com/?foo=12&foo=34

...

print >>sys.stderr, request.GET.getlist('foo')

您可以传递 filter 值的逗号分隔列表:

/mylist/filter=shirts,pants

然后使用 __in 查找获取条目:

entry_types = request.GET.get('filter', '').split(',')
latest_entries = Entry.objects.filter(entrytype__in=entry_types)

或者使用getlist()方法QueryDict:

/mylist/filter=shirts&filter=pants

使用相同的 ORM 调用:

entry_types = request.GET.getlist('filter')
latest_entries = Entry.objects.filter(entrytype__in=entry_types)

UPDATE:将多个类型传递给视图,将它们保存在数组中,并使用join()方法获取它们的逗号分隔字符串:

var types = []; // currently shown entry types

function filter(type) {
  // add or remove the items from the grid
  var index = types.indexOf(type);
  if (index > -1) {
    types.splice(index, 1); // remove the type from the list
  } else {
    types.push(type); // add the type to the filter
  }
  $.get("/storefront/?filter="+types.join(","), function(data) {
    ...
  }
}

更新 2:如果您使用两个字段来过滤查询集,那么您必须在视图中创建两个单独的列表:

filters = request.GET.getlist('filter')
entry_types = [f for f in filters if not f.startswith('delivery')]
delivery_types = [f for f in filters if f.startswith('delivery')]

latest_entries = Entry.objects.all()
if entry_types:
    latest_entries = latest_entries.filter(entrytype__in=entry_types)
if delivery_types:
    latest_entries = latest_entries.filter(deliverytype__in=delivery_types)

JavaScript 代码无需任何修改即可运行。