почему {% if post.is_liked%} не работает дома. html - PullRequest
1 голос
/ 20 июня 2020

Я передал контекст is_liked из PostListView в домашний шаблон, но почему if post.is_liked оператор не работает? У меня есть функция like_post, которая, когда пользователю понравился пост, is_liked will equal to true и текст меняются с не понравившегося на понравившийся. Но почему операторы if не работали (только показывали не понравившиеся) в шаблоне без сообщений об ошибках? Или я попытался изменить операторы if на {% post.user.is_liked %} и {% user.is_liked %}. Но все равно не вышло, в чем проблема? спасибо

models.py

class Post(models.Model):
    title = models.CharField(max_length=100)
    date_posted = models.DateTimeField(default=timezone.now)
    author = models.ForeignKey(User, on_delete=models.CASCADE)
    likes = models.ManyToManyField(User, related_name='likes', blank=True)

    def __str__(self):
        return self.title

    def get_absolute_url(self):
        return reverse('post-detail', kwargs={'pk': self.pk})

views.py

def home(request):
    context = {
        'posts': Post.objects.all(),
    }
    return render(request, 'blog/home.html', context)


def like_post(request):   # post like
    post = get_object_or_404(Post, id=request.POST.get('post_id'))
    is_liked = False
    if post.likes.filter(id=request.user.id).exists():
        post.likes.remove(request.user)
        is_liked = False
    else:
        post.likes.add(request.user)
        is_liked = True

    return HttpResponseRedirect(post.get_absolute_url())


class PostListView(ListView):
    model = Post
    template_name = 'blog/home.html'  # <app>/<model>_<viewtype>.html
    context_object_name = 'posts'
    ordering = ['-date_posted']
    paginate_by = 10
    is_liked = False

    def get_context_data(self, *, object_list=None, **kwargs):
        context = super(PostListView, self).get_context_data()
        posts = context['posts']
        for post in posts:
            if post.likes.filter(id=self.request.user.id).exists():
                context['is_liked'] = True
            return context


class PostDetailView(DetailView):
    model = Post
    is_liked = False

    def get_context_data(self, **kwargs):
        context = super().get_context_data(**kwargs)
        post = context['post']
        if post.likes.filter(id=self.request.user.id).exists():
            context['is_liked'] = True
        return context

home. html

{% for post in posts %}
  <form action="{% url 'like_post' %}" method="post">
            {% csrf_token %}
            {% if post.is_liked %} #I want to get whether the post is_liked by user
                <h5>liked</h5>
            {% else %}
                <h5>not liked</h5>
            {% endif %}
    </form>
{% endfor %}

1 Ответ

1 голос
/ 20 июня 2020

Вы можете аннотировать набор запросов так, чтобы объекты Post, возникающие из этого, имели дополнительный атрибут .is_liked с подзапросом Exists [Django -doc] :

from django.db.models import <b>Exists</b>, <b>OuterRef</b>

class PostListView(ListView):
    model = Post
    template_name = 'blog/home.html'
    context_object_name = 'posts'
    ordering = ['-date_posted']
    paginate_by = 10

    def get_queryset(self, *args, **kwargs):
        return super().get_queryset(*args, **kwargs).annotate(
            <b>is_liked=Exists(</b>Post.likes.through.objects.filter(
                user_id=self.request.user.id,
                post_id=<b>OuterRef('pk')</b>
            <b>)</b>)
        )
...