--- layout: post title: "Disqus API 评论嵌套问题" date: 2017-08-06 16:15:57+0800 thumb: IMG_PATH/disqus.svg category: tech tags: ["Disqus", "Disqus API"] --- 实现评论嵌套逻辑很简单:若评论列表排序为时间升序,只要评论逐条加载,无父评论就是根评论,有父评论就插到父评论下即可;排序若为时间降序,评论嵌套也不难,只要反序(reverse)之,就跟升序的无区别了。 ## 限制评论获取数量出现的问题 单篇文章的评论数量过多时,一次性全部加载就会比较慢,一次性加载太多内容体验上也不大好,所以得限制一次性加载的评论数量。 当每次加载限制数量时,按时间升序不会出问题,但在时间降序时,问题就来了,下面以一个例子来说下: ```bash thread │ ├─post 7 │ ├─post 4 │ └─post 5 │  └─post 8 │ ├─post 2 │ └─post 1 ├─post 6 └─post 3 ``` 当每次获取评论的数量限制在 3 时: 1. 第一次获取的是 6、7、8 楼,其中阿 7 正常加载,而阿 6、阿 8 找不到父评。 2. 第二次获取的是 3、4、5 楼,其中阿 4、阿 5 正常加载,阿 3 找不到父评。 3. 第三次获取的是 1、2 楼,都是根评论,正常加载。 以上就只有 1、2、4、5、7 是正常加载的,丢了 3 条。 ## 前端解决 在后端不改的前提下,我选择在前端设置一个未加载的数组变量,当找不到父评时便把评论 `push` 进去,每次加载评论列表时 `concat` 它。上面例子使用这个方法解决便如下: 1. `loadlist = [6, 7, 8], unload = [6, 8];` 2. `loadlist = [3, 4, 5, 6, 8], unload = [3, 6];` 3. `loadlist = [1, 2, 3, 6], unload = [];` 注:每次待加载评论数组 `loadlist`,加载后的未加载数组 `unload`。 这样的话,评论就都能正常加载完成了,但有个问题,初衷限制每次加载评论数量为 3,这个 3 在此并没法体现出来,因为第一次加载显示 1 条,第二次显示 3 条,第三次显示 4 条。原本应该的 332 变成 134 了。 ## 后端解决 在前端不改的前提下,后端最好的解决方法就是每次返回都能正常加载的评论数据,按上面的例子,应该返回的数据是: 1. 第一次 4、5、7 2. 第二次 1、2、8 3. 第三次 3、6 后端要处理的感觉就挺多,可以先获取所有的评论[[1]][1],再根据有没父评等,并不简单的排序,还要处理需要传的参数。感觉比较麻烦且不大合理,我就没去折腾。 ## 前端 + 后端解决 另一种妥协的解决方法是结合前后端,后端一次性返回所有评论数据,再由前端去处理每次的显示。前端变成一次获取,多次加载,前端代码会多一些。 综合以上的解决方法,最终选择比较简单的第一种方法解决,虽然例子看起来出入很大,但实际上应用起来。若限制每次加载的评论数为 50,每次加载多几条少几条,在观感上没啥区别。 ## Disqus 评论框的解决方法 直至前几天,和 @W_Z_C 讨论这个问题(看不到[点此](/pre-moderation-on-disqus.html#comment-3446041177)): {% include media.html height="300" width="100%" src="https://embed.disqus.com/p/1kzq75w?p=1" %} 之后,我就真去看看,到底 Disqus 原生评论框是怎么解决这个问题的。 原生评论框每次获取的评论数量是正常的,限制多少就是多少。再看请求,发现评论框用的 API 与我自己用的(`threads/listPosts` 或 `posts/list`)竟然不一样,它是使用 `threads/listPostsThreaded`,Disqus API 文档并没有这个接口的有关说明。看情况,使用这个就应该可以很完美地解决上面所遇到的问题。 经测试,`threads/listPostsThreaded`的权限跟在 API 文档里有的,并无区别。 不过这里有个问题,Disqus 原生评论框,无论如何都是时间降序排序。我想要的是:无父评的按时间倒序,有父评的就是按时间正序,即上面例子中,3 和 6 位置倒过来,比较符合阅读习惯。 ## 参考资料 [1]: https://stackoverflow.com/questions/15080258/how-to-get-all-the-comments-from-disqus "How to get all the comments from Disqus?" **本文历史** * 2017 年 08 月 06 日 完成初稿