想在聊天界面想做个和微信一样的时间显示 达到下图这种效果
百度了一下,发现都是有点不全的,把网上的合并了一下组成下方的js文件 记录一下
1.建议新建一个JS文件 common.js
1.第一个方法是把时间戳转成具体时间日期
/**
* 对Date的扩展,将 Date 转化为指定格式的String。
*
* 月(M)、日(d)、小时(h)、分(m)、秒(s)、季度(q) 可以用 1-2 个占位符,
* 年(y)可以用 1-4 个占位符,毫秒(S)只能用 1 个占位符(是 1-3 位的数字)。
*
* 【示例】:
* common.formatDate(new Date(), 'yyyy-MM-dd hh:mm:ss.S') ==> 2006-07-02 08:09:04.423
* common.formatDate(new Date(), 'yyyy-M-d h:m:s.S') ==> 2006-7-2 8:9:4.18
* common.formatDate(new Date(), 'hh:mm:ss.S') ==> 08:09:04.423
*/
var _formatDate = function (date, fmt) {var o = {"M+": date.getMonth() + 1, //月份"d+": date.getDate(), //日"h+": date.getHours(), //小时"m+": date.getMinutes(), //分"s+": date.getSeconds(), //秒"q+": Math.floor((date.getMonth() + 3) / 3), //季度"S": date.getMilliseconds() //毫秒};if (/(y+)/.test(fmt)) fmt = fmt.replace(RegExp.$1, (date.getFullYear() + "").substr(4 - RegExp.$1.length));for (var k in o)if (new RegExp("(" + k + ")").test(fmt)) fmt = fmt.replace(RegExp.$1, (RegExp.$1.length == 1) ? (o[k]) : (("00" + o[k]).substr(("" + o[k]).length)));return fmt;
};
/**
* 仿照微信中的消息时间显示逻辑,将时间戳(单位:毫秒)转换为友好的显示格式.
*
* 1)7天之内的日期显示逻辑是:今天、昨天(-1d)、前天(-2d)、星期?(只显示总计7天之内的星期数,即<=-4d);
* 2)7天之外(即>7天)的逻辑:直接显示完整日期时间。
*
* @param {[long]} timestamp 时间戳(单位:毫秒),形如:1550789954260
* @param {boolean} mustIncludeTime true表示输出的格式里一定会包含“时间:分钟”
* ,否则不包含(参考微信,不包含时分的情况,用于首页“消息”中显示时)
*
* @return {string} 输出格式形如:“刚刚”、“10:30”、“昨天 12:04”、“前天 20:51”、“星期二”、“2019/2/21 12:09”等形式
* @author 即时通讯网([url=http://www.52im.net]http://www.52im.net[/url])
* @since 1.1
*/
export function _getTimeStringAutoShort2 (timestamp, mustIncludeTime){// 当前时间var currentDate = new Date();// 目标判断时间var srcDate = new Date(parseInt(timestamp));var currentYear = currentDate.getFullYear();var currentMonth = (currentDate.getMonth()+1);var currentDateD = currentDate.getDate();var srcYear = srcDate.getFullYear();var srcMonth = (srcDate.getMonth()+1);var srcDateD = srcDate.getDate();var ret = "";// 要额外显示的时间分钟var timeExtraStr = (mustIncludeTime?" "+_formatDate(srcDate, "hh:mm"):"");// 当年if(currentYear == srcYear) {var currentTimestamp = currentDate.getTime();var srcTimestamp = timestamp;// 相差时间(单位:毫秒)var deltaTime = (currentTimestamp-srcTimestamp);// 当天(月份和日期一致才是)if(currentMonth == srcMonth && currentDateD == srcDateD) {// 时间相差60秒以内if(deltaTime < 60 * 1000)ret = "刚刚";// 否则当天其它时间段的,直接显示“时:分”的形式elseret = _formatDate(srcDate, "hh:mm");}// 当年 && 当天之外的时间(即昨天及以前的时间)else {// 昨天(以“现在”的时候为基准-1天)var yesterdayDate = new Date();yesterdayDate.setDate(yesterdayDate.getDate()-1);// 前天(以“现在”的时候为基准-2天)var beforeYesterdayDate = new Date();beforeYesterdayDate.setDate(beforeYesterdayDate.getDate()-2);// 用目标日期的“月”和“天”跟上方计算出来的“昨天”进行比较,是最为准确的(如果用时间戳差值// 的形式,是不准确的,比如:现在时刻是2019年02月22日1:00、而srcDate是2019年02月21日23:00,// 这两者间只相差2小时,直接用“deltaTime/(3600 * 1000)” > 24小时来判断是否昨天,就完全是扯蛋的逻辑了)if(srcMonth == (yesterdayDate.getMonth()+1) && srcDateD == yesterdayDate.getDate())ret = "昨天"+timeExtraStr;// -1d// “前天”判断逻辑同上else if(srcMonth == (beforeYesterdayDate.getMonth()+1) && srcDateD == beforeYesterdayDate.getDate())ret = "前天"+timeExtraStr;// -2delse{// 跟当前时间相差的小时数var deltaHour = (deltaTime/(3600 * 1000));// 如果小于或等 7*24小时就显示星期几if (deltaHour <= 7*24){var weekday=new Array(7);weekday[0]="星期日";weekday[1]="星期一";weekday[2]="星期二";weekday[3]="星期三";weekday[4]="星期四";weekday[5]="星期五";weekday[6]="星期六";// 取出当前是星期几var weedayDesc = weekday[srcDate.getDay()];ret = weedayDesc+timeExtraStr;}// 否则直接显示完整日期时间elseret = _formatDate(srcDate, "yyyy/M/d")+timeExtraStr;}}}// 往年else{ret = _formatDate(srcDate, "yyyy/M/d")+timeExtraStr;}return ret;
};
/*** 聊天时间格式化* 规则:* 1. 每五分钟为一个跨度* 2. 今天显示,小时:分钟,例如:11:12* 3. 昨天显示,昨天 小时:分钟 例如:昨天 11:12* 4. 一周内显示,星期几 小时:分钟 例如:星期四 11:12* 5. 日期差大于一周显示,年月日 小时:分钟 例如:2021年9月30日 11:12* @param currentMessageList 传入的当前会话数组* @param sort 传入数组排序:0-数组时间倒序;1-数组时间正序* @param type 五分钟规则区分:0-永远跟上一个显示的时间对比是否超5分钟 ;1-永远两条消息对比是否超5分钟* @returns {Array|null}*/
export function msgTimeFormat(currentMessageList, sort, type) {// console.log('传入的会话数组', currentMessageList)const newMessageList = []const currentFilterList = currentMessageList.filter((item) => {return item.type !== 'showTime'})currentFilterList.forEach((item, index) => {const date = item.time * 1000let showTimeif (index === 0) {//第一条必显示时间showTime = _getTimeStringAutoShort2(date, true)newMessageList.push({payload: {text: showTime},type: 'showTime',//超五分钟显示时间-标识time: item.time})newMessageList.push(item)} else if (index <= currentFilterList.length - 1) {const current = currentFilterList[index].timelet minutesconst showTimeList = newMessageList.filter((item) => {return item.type === 'showTime'})const lastShowTime = showTimeList[showTimeList.length - 1].time//添加的时间且最后一条,用于对比if (type) {const prev = currentFilterList[index - 1].timeminutes = (current - prev) / 60}if (!sort) {minutes = (current - lastShowTime) / 60} else {minutes = (lastShowTime - current) / 60}//超五分if (minutes > 5) {showTime = _getTimeStringAutoShort2(date, true)newMessageList.push({payload: {text: showTime},type: 'showTime',time: item.time})newMessageList.push(item)} else {newMessageList.push(item)}}})// console.log('最后得到的数组', newMessageList)return newMessageList
}
2.在需要使用的地方引入上述方法
import {msgTimeFormat} from '文件具体位置';//直接把消息列表
messageList=msgTimeFormat(messageList,0,0)
初始列表数组的数据格式
转换后的数据格式
展示的时候判断type类型展示payload里面的text就行了