在社区中,发布的动态信息,经常会有一个相对余实际发布时间的相对时间。比如这里的微博:
服务端存储的时间格式,一般为 Unix 时间戳,比如 2019/1/6 13:40:1 的Unix 时间戳为 1546753201651。前端在获取到这个时间戳之后,会转换为可读格式的时间。在社交类产品中,一般会将时间戳转换为 x 分钟前,x 小时前或者 x 天前,因为这样的显示方式用户体验更好。
我们可以自定义一个 v-relative-time 指令来实现上述功能。
html:
<!DOCTYPE html> <html lang="en"> <head> <meta charset="UTF-8"> <title>Title</title> <style type="text/css"> </style> </head> <body> <div id="app" v-cloak> 现在时间:<div v-relative-time="now"></div><p></p> 2019/1/6 13:45:02:<div v-relative-time="1546753502000"></div><p></p> 2019/1/6 8:02:02:<div v-relative-time="1546732922000"></div><p></p> 2019/1/5 22:02:02:<div v-relative-time="before"></div><p></p> 2019/1/1 22:02:02:<div v-relative-time="1546351322000"></div><p></p> 2018/1/6 8:02:02:<div v-relative-time="1515196922000"></div> </div> <script src="https://cdn.bootcss.com/vue/2.2.2/vue.min.js"></script> <script src="index.js"></script> </body> </html>
注意:div v-relative-time 指令的入参为精确到毫秒的 Unix 时间戳,如果入参单位为秒,那么可以乘以 1000 后,再传入。
js:
/** * 时间对象 * @type {{getCurrentUnix: Time.getCurrentUnix, getTodayUnix: Time.getTodayUnix, getThisYearUnix: Time.getThisYearUnix, format: Time.format, compensateZero: Time.compensateZero, transform: Time.transform}} */ var Time = { //获取当前 Unix 时间戳 getCurrentUnix: function () { return new Date().getTime(); }, //获取今日 0 点 0 分 0 秒的 Unix 时间戳 getTodayUnix: function () { var date = new Date(); date.setHours(0); date.setMinutes(0); date.setSeconds(0); date.setMilliseconds(0); return date.getTime(); }, //获取今年 1 月 1 日 0 点 0 分 0 秒的 Unix 时间戳 getThisYearUnix: function () { var date = new Date(); date.setMonth(0); date.setDate(1); date.setHours(0); date.setMinutes(0); date.setSeconds(0); date.setMilliseconds(0); return date.getTime(); }, //格式化日期;
时间转换逻辑为:
我们专门设计了一个 Time 对象,用于定义与时间相关的函数:
以下是与时间相关的小知识:
Math.floor() Math.ceil()
/** * 相对时间指令 */ Vue.directive('relative-time', { bind: function (el, binding) { el.innerHTML = Time.transform(binding.value); el._relativeTime = setInterval(function () { el.innerHTML = Time.transform(binding.value); }, 60000);//每分钟,刷新一次 }, unbind: function (el) { clearInterval(el._relativeTime); delete el._relativeTime; } }); var app = new Vue({ el: '#app', data: { now: (new Date()).getTime(), //2019/1/5 22:02:02 before: 1546696922000 } });
在相对时间指令中,我们在 bind() 中,把指令中的入参转换为相对时间,然后写入指令所在的元素中,接着还定义了一个每分钟更新元素内容的定时器。在 unbind() 中,执行清除定时器操作。
渲染结果:
编写自定义指令,建议如下:
本文示例代码