wordpress文章浏览历史(非插件版)


wordpress如果添加文章浏览记录区域,可以很好的提高网站的用户体验 比如: 某人经常通过搜索引擎进来我的网站, 可能发现以前在这里也看过一些文章, 比较合他口味, 久而久之就能发展成常客.
如果懒得看可以直接使用插件:wordpress文章浏览历史插件

【实现原理】

这个功能实现非常简单, 平白一点来说就是找个地方将页面一些信息保存起来, 如果页面两次被访问, 新的内容覆盖掉旧的内容; 在页面加载的时候, 将保存的信息取出显示.

实现方法也很多, 最土的是将内容保存在 cookie 中, 但用 cookie 储存数据存在一些问题. 比如: cookie 是随 HTTP 响应一起被发送的, 会对服务器端响应时间产生一定程度的影响, 尤其是在使用 XMLHttpRequest 对象向服务器端发送或请求数据的时候.

利用 cookie 虽然比较土, 但最实用, 因为几乎可以兼容所有浏览器. 我这里使用的 localStorage, 数据完全保存在浏览器中, 不会有影响服务器响应, 但 IE6/7 中不能使用.

这里要完成以下几个功能点:

  • 将存储内容转成数组对象, 因为 localStorage 只能保存字符串.
  • 去重, 如果保存的内容已经存在, 则删除旧数据, 以最新的为准.
  • 限制保存数量, 没有必要无限保存历史导致列表超长.

【实现步骤】

我已经写好了一个名为 View History 的脚本, 不到 50 行代码, 就是为了实现上述的三个功能.
1. 引用脚本并初始化对象

  1. <script src="view-history.js"></script>
  2. <script>
  3. /* <![CDATA[ */
  4. if(typeof localStorage !== 'undefined' && typeof JSON !== 'undefined') {
  5. var viewHistory = new ViewHistory();
  6. viewHistory.init({
  7. limit: 5,
  8. storageKey: 'viewHistory',
  9. primaryKey: 'url'
  10. });
  11. }
  12. /* ]]> */
  13. </script>

2. 在文章页面保存浏览信息

  1. <script>
  2. /* <![CDATA[ */
  3. // 如果 ViewHistory 的实例存在,则可以将页面信息写入。
  4. if(viewHistory) {
  5. var page = {
  6. "title": document.getElementsByTagName('title')[0].innerHTML,
  7. "url": location.href // 这是 primaryKey
  8. // "time": new Date()
  9. // "author": ...
  10. // 这里可以写入更多相关内容作为浏览记录中的信息
  11. };
  12. viewHistory.addHistory(page);
  13. }
  14. /* ]]> */
  15. </script>

3. 显示历史浏览记录

  1. <script>
  2. /* <![CDATA[ */
  3. var wrap = document.getElementById('view-history');
  4. // 如果 ViewHistory 的实例存在,并且外层节点存在,则可显示历史浏览记录
  5. if(viewHistory && wrap) {
  6. // 获取浏览记录
  7. var histories = viewHistory.getHistories();
  8. // 组装列表
  9. var list = document.createElement('ul');
  10. if(histories && histories.length > 0) {
  11. for(var i=histories.length-1; i>=0; i--) {
  12. var history = histories[i];
  13. var item = document.createElement('li');
  14. var link = document.createElement('a');
  15. link.href = history.url;
  16. link.innerHTML = history.title;
  17. item.appendChild(link);
  18. list.appendChild(item);
  19. }
  20. // 插入页面特定位置
  21. wrap.appendChild(list);
  22. }
  23. }
  24. /* ]]> */
  25. </script>

【说明】

只用 localStorage 做历史浏览记录不能解决时效性问题, 也就是不能获得最新的文章信息. 如果我将文章评论数加到浏览记录中, 那么用户看到的仅是他访问当时的评论数量. 解决办法是有的, 有些网站只保存文章的 ID, 在页面加载时再动态加载文章信息, 比如: 评论数和平均评价.

附完整代码:

  1. ViewHistory = function() {
  2.  
  3. this.config = {
  4. limit: 10,
  5. storageKey: 'viewHistory',
  6. primaryKey: 'url'
  7. };
  8.  
  9. this.cache = {
  10. localStorage: null,
  11. userData: null,
  12. attr: null
  13. };
  14. };
  15.  
  16. ViewHistory.prototype = {
  17.  
  18. init: function(config) {
  19. this.config = config || this.config;
  20. var _self = this;
  21.  
  22. // define localStorage
  23. if (!window.localStorage && (this.cache.userData = document.body) && this.cache.userData.addBehavior && this.cache.userData.addBehavior('#default#userdata')) {
  24. this.cache.userData.load((this.cache.attr = 'localStorage'));
  25.  
  26. this.cache.localStorage = {
  27. 'getItem': function(key) {
  28. return _self.cache.userData.getAttribute(key);
  29. },
  30. 'setItem': function(key, value) {
  31. _self.cache.userData.setAttribute(key, value);
  32. _self.cache.userData.save(_self.cache.attr);
  33. }
  34. };
  35.  
  36. } else {
  37. this.cache.localStorage = window.localStorage;
  38. }
  39. },
  40.  
  41. addHistory: function(item) {
  42. var items = this.getHistories();
  43. for(var i=0, len=items.length; i<len; i++) {
  44. if(item[this.config.primaryKey] && items[i][this.config.primaryKey] && item[this.config.primaryKey] === items[i][this.config.primaryKey]) {
  45. items.splice(i, 1);
  46. break;
  47. }
  48. }
  49.  
  50. items.push(item);
  51.  
  52. if(this.config.limit > 0 && items.length > this.config.limit) {
  53. items.splice(0, 1);
  54. }
  55.  
  56. var json = JSON.stringify(items);
  57. this.cache.localStorage.setItem(this.config.storageKey, json);
  58. },
  59.  
  60. getHistories: function() {
  61. var history = this.cache.localStorage.getItem(this.config.storageKey);
  62. if(history) {
  63. return JSON.parse(history);
  64. }
  65. return [];
  66. }
  67. };