Как исключить текущий объект в запросе ManyToMany? - PullRequest
1 голос
/ 04 марта 2011

У меня есть две базовые модели: история и категория:

class Category(models.Model):
    title = models.CharField(max_length=50)
    slug = models.SlugField()
    ...

class Story(models.Model):
    headline = models.CharField(max_length=50)
    slug = models.SlugField()
    categories = models.ManyToManyField(Category)
    ...

И мой взгляд на детали истории:

from django.views.generic import date_based, list_detail
from solo.apps.news.models import Story

def story_detail(request, slug, year, month, day):
"""
Displays story detail. If user is superuser, view will display 
unpublished story detail for previewing purposes.
"""
stories = None
if request.user.is_superuser:
    stories = Story.objects.all()
else:
    stories = Story.objects.live()
return date_based.object_detail(
    request,
    year=year,
    month=month,
    day=day,
    date_field='pub_date',
    slug=slug,
    queryset=stories,
    template_object_name = 'story',
)

На представлении для данного объекта истории - я 'используя общий подробный вид - я хотел бы отобразить список историй, связанных с текущей историей, через категории, примененные к текущей истории.

Вот как я делаю это в настоящее время в деталях историиtemplate:

{% for category in story.category.all %}
    <ul id="related_stories">
    {% for story in category.story_set.all|slice:"5" %}
        <li><a href="{{ story.get_absolute_url }}" title="{{ story.headline }}">{{ story.headline }}</a></li>
    {% endfor %}
    </ul>
{% endfor %}

Это дает мне то, что мне нужно, за исключением того, что я не хочу отображать связанный заголовок для истории, которую я сейчас просматриваю.

Я полагаю, что это делается с помощьюФильтр «исключить», но я не уверен, относится ли он к модели «Категория» или «История» как метод, или как ее создать.

Любая помощь будет принята с благодарностью!

Ответы [ 2 ]

0 голосов
/ 04 марта 2011

Сделайте это:

class Story(models.Model):
...
   @property
   def related_story_set(self):
       category_id_list = self.category.values_list("id", flat=True)
       return Story.objects.filter(category__id__in=category_id_list).exclude(id=self.id)

Тогда вы можете сделать это в шаблоне:

<ul id="related_stories">
{% for related_story in story.related_story_set.all|slice:"5" %}
    <li><a href="{{ related_story.get_absolute_url }}" title="{{ related_story.headline }}">{{ related_story.headline }}</a></li>
{% endfor %}
</ul>
0 голосов
/ 04 марта 2011

Вы можете просто проверить в шаблоне, является ли текущая повторяемая история оригинальной:

{% for category in story.category.all %}
    <ul id="related_stories">
        {% for substory in category.story_set.all|slice:"5" %}
            {% if substory != story %}
                <li><a href="{{ substory.get_absolute_url }}" title="{{ substory.headline }}">{{ story.headline }}</a></li>
            {% endif %}
        {% endfor %}
    </ul>
{% endfor %}

Вы просили поместить его в модельный метод:

class Story(models.Model):
    ...

    def get_categories_with_stories(self):
        categories = self.category.all()
        for category in categories:
            category.stories = category.story_set.exclude(id=self.id)[:5]
        return categories

Это не решает проблему ваших дорогих запросов, но это не было частью вопроса.

{% for category in story.get_categories_with_stories %}
    <ul id="related_stories">
        {% for substory in category.stories %}
               <li><a href="{{ substory.get_absolute_url }}" title="{{ substory.headline }}">{{ story.headline }}</a></li>
        {% endfor %}
    </ul>
{% endfor %}
...