0

我在几天内都在与这个问题作斗争,但找不到适合我的情况的解决方案。我正在尝试在不刷新页面的情况下制作喜欢系统。在喜欢和不喜欢的同步模式下,系统工作正常,但是当我尝试添加 AJAX 时,我得到 405,并且只有最后一条评论可以单击,我理解 Ajax 不理解 django url 的问题或 pk 喜欢我的变体 {% url 'store:like' comment.pk %} ,但如何解决?

模板中有这一部分:

{% for comment in comments %}
       <h6 class="card-header">
       {{ comment.author }}<small> добавлен {{ comment.created_at|date:'M d, Y H:i' }} </small>
       </h6>
        <div class="card-body">
        <h4>{{ comment }}</h4>
        <form id="like-{{comment.pk}}" method="POST" action="{% url 'store:add_like' comment.pk %}">
                    {% csrf_token %}

                    <button style="background-color: transparent; border: none; box-shadow: none;" type="submit">
                        <a class="btn btn-success" id="like-count-{{comment.pk}}"> Likes {{ comment.likes.all.count }}</a>
                    </button>
                </form>
        

        </div>
        {% empty %}
        <p>Для данного товара  ещё нет комментариев.</p>

        {% endfor %}

我在同一个模板中调用 ajax:

<script type="text/javascript">
    $(document).ready(function(){

        

  $('[id^="like-"]').submit(function(e){

            e.preventDefault();
            var endpoint = $(this).attr('action');
            var serializedData = $(this).serializeArray();
      $.ajax({
               type: 'POST',
               url: endpoint,
               data: serializedData,
               success: function(response) {
                     $( "#like-count-"+response["id"].toString()).text("Likes "+response["like_count"]);

                },
                error: function(rs, e) {
                       alert(rs.responseText);
                }
          });
    })

这部分来自网址:

path('products/<int:pk>/like/', addlike, name='like'),

查看喜欢:

@api_view(['POST'])
def addlike(request, pk, *args, **kwargs):

        is_ajax = request.META.get('HTTP_X_REQUESTED_WITH') == 'XMLHttpRequest'
        if request.method == 'POST' and is_ajax:
            
            post = Comment.objects.get(pk=pk)


            is_dislike = False

            for dislike in post.dislikes.all():
                if dislike == request.user:
                    is_dislike = True
                    break

            if is_dislike:
                post.dislikes.remove(request.user)

            is_like = False

            for like in post.likes.all():
                if like == request.user:
                    is_like = True
                    break

            if not is_like:
                post.likes.add(request.user)

            if is_like:
                post.likes.remove(request.user)

            all_post_likes = post.total_likes() (function from models)
            return JsonResponse({"success": True, "like_count": all_post_likes, "id": pk}, status=200)
        else:

            return JsonResponse({"success": False}, status=400)

如何强制 AJAX 调用我需要的 pk?(最后我找到了解决方案,更新了代码的最终版本)

4

2 回答 2

1

去掉 ‍<code>data-url‍</code> 并设置action属性,因为当你点击提交按钮时,默认情况下这个POST请求会被发送到当前 URL 并且你会收到 405 状态码,但是如果你设置action了这个POST请求将被发送到likeurl:

<form id="like" method="POST" action="{% url 'store:like' comment.pk %}">
    {% csrf_token %}
     <input type="hidden" value="{{comment.pk}}" name="id">
     <input type="hidden" name="next" value="{{ request.path }}">
     <button style="background-color: transparent; border: none; box-shadow: none;" type="submit">
     <a class="btn btn-success" id="like"> Likes {{ comment.likes.all.count }}</a>
     </button>
</form>
        

在js中你可以得到这样的URL

var endpoint = $(this).attr('action');

更新

所有表单都具有相同的“ID”,因此只执行第一个表单,其余表单不由 Ajax 管理,要解决此问题,您可以ID对表单进行不同的处理(此条件也适用于a标签):

<form id="like-{{comment.pk}}" method="POST" action="{% url 'store:like' comment.pk %}">
     {% csrf_token %}
     ........
     .....
     <a class="btn btn-success" id="likes-count-{{comment.pk}}"> Likes {{ comment.likes.all.count }}</a>

您可以通过这种方式接收事件并更新喜欢的数量。

$(document).ready(function(){
  $('[id^="like-"]').submit(function(e){
    e.preventDefault();
    var endpoint = $(this).attr('action');
    var serializedData = $(this).serializeArray();
     
    $.ajax({
       url: endpoint,
       method: "POST",
       data: serializedData,
       success: function(response){
         $("#likes-count-" + serializedData[1].value).text("Likes "+response["like_count"]);
       }
    });
  })
});

在视图中,您应该返回新的点赞数:

return JsonResponse({"success": True, "like_count": new_like_count}, status=200)

在成功函数中,您可以通过 获取新数字,response现在我们更改标签的文本值a以更改点赞数。

^=意思是:选择具有指定属性的元素,其值恰好以给定字符串开头。属性StartsWith1

于 2022-01-08T10:24:24.383 回答
0

我认为您应该将 csrf_token 与 ajax 请求一起发送 headers: { "X-CSRFToken": token }到您的 ajax 请求中,“token 是 csrf_token”或将 @csrf_exempt 装饰器添加到您的函数中,但它会使您的视图对 CSRF 攻击不安全。

你可以在这里找到更多信息https://docs.djangoproject.com/en/4.0/ref/csrf/

于 2022-01-08T09:41:16.960 回答