Javascript timestamp к относительному времени (например, 2 секунды назад, неделю назад и т. Д.), Лучшие методы? - PullRequest
51 голосов
/ 24 мая 2011

Я ищу хороший фрагмент JS для преобразования временной метки (например, из Twitter API) в удобное для пользователя относительное время (например, 2 секунды назад, неделю назад и т. Д.).

Любой желающийподелитесь некоторыми из своих любимых методов (желательно без использования плагинов)?

Ответы [ 11 ]

94 голосов
/ 24 мая 2011

Ну, это довольно просто, если вы не слишком озабочены точностью. Что не так с тривиальным методом?

function timeDifference(current, previous) {

    var msPerMinute = 60 * 1000;
    var msPerHour = msPerMinute * 60;
    var msPerDay = msPerHour * 24;
    var msPerMonth = msPerDay * 30;
    var msPerYear = msPerDay * 365;

    var elapsed = current - previous;

    if (elapsed < msPerMinute) {
         return Math.round(elapsed/1000) + ' seconds ago';   
    }

    else if (elapsed < msPerHour) {
         return Math.round(elapsed/msPerMinute) + ' minutes ago';   
    }

    else if (elapsed < msPerDay ) {
         return Math.round(elapsed/msPerHour ) + ' hours ago';   
    }

    else if (elapsed < msPerMonth) {
        return 'approximately ' + Math.round(elapsed/msPerDay) + ' days ago';   
    }

    else if (elapsed < msPerYear) {
        return 'approximately ' + Math.round(elapsed/msPerMonth) + ' months ago';   
    }

    else {
        return 'approximately ' + Math.round(elapsed/msPerYear ) + ' years ago';   
    }
}

Рабочий пример здесь .

Возможно, вы захотите настроить его так, чтобы лучше обрабатывать единичные значения (например, 1 day вместо 1 days), если это вас беспокоит.

19 голосов
/ 29 апреля 2014

Вот точное подражание времени твиттера назад без плагинов:

  function timeSince(timeStamp) {
    var now = new Date(),
      secondsPast = (now.getTime() - timeStamp.getTime()) / 1000;
    if(secondsPast < 60){
      return parseInt(secondsPast) + 's';
    }
    if(secondsPast < 3600){
      return parseInt(secondsPast/60) + 'm';
    }
    if(secondsPast <= 86400){
      return parseInt(secondsPast/3600) + 'h';
    }
    if(secondsPast > 86400){
        day = timeStamp.getDate();
        month = timeStamp.toDateString().match(/ [a-zA-Z]*/)[0].replace(" ","");
        year = timeStamp.getFullYear() == now.getFullYear() ? "" :  " "+timeStamp.getFullYear();
        return day + " " + month + year;
    }
  }

Суть https://gist.github.com/timuric/11386129

Скрипка http://jsfiddle.net/qE8Lu/1/

Надеюсь, это поможет.

5 голосов
/ 24 мая 2011

Тад!Timeago: http://timeago.yarp.com/

Ох, подожди - без плагинов?Почему это тогда?Я полагаю, вы могли бы открыть файл плагина и извлечь из него смелость.

4 голосов
/ 16 декабря 2018

Intl.RelativeTimeFormat - собственный API

В настоящее время (18 декабря) a Стадия 3 предложение и уже реализовано в Chrome 71

const rtf = new Intl.RelativeTimeFormat('en', { numeric: 'auto' });

const millisecondsPerDay = 24 * 60 * 60 * 1000;

[
  [3.14 , 'second' ],
  [-15  , 'minute' ],
  [8    , 'hour'   ],
  [-1   , 'day'    ],
  [3    , 'week'   ],
  [-5   , 'month'  ],
  [2    , 'quarter'],
  [-42  , 'year'   ],
  [(new Date('9/22/2018') - new Date())/millisecondsPerDay,'day']
].forEach(d => console.log(   rtf.format(d[0], d[1])  ));

Intl.RelativeTimeFormat доступно по умолчанию в V8 v7.1.179 и Хром 71 . По мере того как этот API станет более доступным, вы найдете библиотеки, такие как Moment.js , Globalize и date-fns , отбрасывающие их зависимость от жестко закодированных баз данных CLDR в пользу нативного родственника функциональность форматирования времени, тем самым улучшая время загрузки производительность, разбор и производительность во время компиляции, время выполнения производительность и использование памяти.

4 голосов
/ 14 июня 2016

Вдохновленный на Diego Castillo awnser's и в timeago.js , я написал для этого свой собственный ванильный плагин.

var timeElement = document.querySelector('time'),
    time = new Date(timeElement.getAttribute('datetime'));

timeElement.innerText = TimeAgo.inWords(time.getTime());

var TimeAgo = (function() {
  var self = {};
  
  // Public Methods
  self.locales = {
    prefix: '',
    sufix:  'ago',
    
    seconds: 'less than a minute',
    minute:  'about a minute',
    minutes: '%d minutes',
    hour:    'about an hour',
    hours:   'about %d hours',
    day:     'a day',
    days:    '%d days',
    month:   'about a month',
    months:  '%d months',
    year:    'about a year',
    years:   '%d years'
  };
  
  self.inWords = function(timeAgo) {
    var seconds = Math.floor((new Date() - parseInt(timeAgo)) / 1000),
        separator = this.locales.separator || ' ',
        words = this.locales.prefix + separator,
        interval = 0,
        intervals = {
          year:   seconds / 31536000,
          month:  seconds / 2592000,
          day:    seconds / 86400,
          hour:   seconds / 3600,
          minute: seconds / 60
        };
    
    var distance = this.locales.seconds;
    
    for (var key in intervals) {
      interval = Math.floor(intervals[key]);
      
      if (interval > 1) {
        distance = this.locales[key + 's'];
        break;
      } else if (interval === 1) {
        distance = this.locales[key];
        break;
      }
    }
    
    distance = distance.replace(/%d/i, interval);
    words += distance + separator + this.locales.sufix;

    return words.trim();
  };
  
  return self;
}());


// USAGE
var timeElement = document.querySelector('time'),
    time = new Date(timeElement.getAttribute('datetime'));

timeElement.innerText = TimeAgo.inWords(time.getTime());
<time datetime="2016-06-13"></time>
1 голос
/ 31 января 2018

Если вам нужна многоязычность и вы не хотите добавлять большую библиотеку, как момент intl -lativeformat от Yahoo это хорошее решение.

var rf = new IntlRelativeFormat('en-US');

var posts = [
    {
        id   : 1,
        title: 'Some Blog Post',
        date : new Date(1426271670524)
    },
    {
        id   : 2,
        title: 'Another Blog Post',
        date : new Date(1426278870524)
    }
];

posts.forEach(function (post) {
    console.log(rf.format(post.date));
});
// => "3 hours ago"
// => "1 hour ago"
1 голос
/ 11 октября 2017

Для этой цели есть также sugar.js и относительная функция.

относительная - выводит строку в единицах относительно текущей даты («назад» или «отныне»).

1 голос
/ 04 декабря 2016

Плагины Datetime существуют, потому что очень трудно сделать это правильно.Это видео, объясняющее несоответствия даты и времени , проливает некоторый свет на проблему.

Все вышеперечисленные решения без плагинов некорректны.

Для работы сДаты и время с использованием плагина предпочтительнее .Из сотен плагинов, которые имеют к нему отношение, мы используем Moment.js , и он делает свою работу.

Из Twitter API dcumentation мы видим их метку времениформат:

"created_at":"Wed Aug 27 13:08:45 +0000 2008"

Мы можем проанализировать его с помощью Moment.js

const postDatetime = moment(
  "Wed Aug 27 13:08:45 +0000 2008",
  "dddd, MMMM Do, h:mm:ss a, YYYY"
);
const now = moment();
const timeAgo = now.diff(postDatetime, 'seconds');

Чтобы указать предпочтительную единицу времени для diff, мы можем использоватьметод isSame.Например:

if (now.isSame(postDatetime, 'day')) {
  const timeUnit = 'days';
}

В целом, построение чего-то вроде:

`Posted ${timeAgo} ${timeUnit} ago`;

Обратитесь к документации вашего плагина для обработки расчетов относительного времени (то есть: "Как давно?").

1 голос
/ 21 октября 2014

Для всех, кто заинтересовался, я создал помощника Handlebars для этого.Использование:

    {{#beautify_date}}
        {{timestamp_ms}}
    {{/beautify_date}}

Помощник:

    Handlebars.registerHelper('beautify_date', function(options) {
        var timeAgo = new Date(parseInt(options.fn(this)));

        if (Object.prototype.toString.call(timeAgo) === "[object Date]") {
            if (isNaN(timeAgo.getTime())) {
                return 'Not Valid';
            } else {
                var seconds = Math.floor((new Date() - timeAgo) / 1000),
                intervals = [
                    Math.floor(seconds / 31536000),
                    Math.floor(seconds / 2592000),
                    Math.floor(seconds / 86400),
                    Math.floor(seconds / 3600),
                    Math.floor(seconds / 60)
                ],
                times = [
                    'year',
                    'month',
                    'day',
                    'hour',
                    'minute'
                ];

                var key;
                for(key in intervals) {
                    if (intervals[key] > 1)  
                        return intervals[key] + ' ' + times[key] + 's ago';
                    else if (intervals[key] === 1) 
                        return intervals[key] + ' ' + times[key] + ' ago';
                }

                return Math.floor(seconds) + ' seconds ago';
            }
        } else {
            return 'Not Valid';
        }
    });
0 голосов
/ 01 июля 2019

Для пользователей Moment.js он имеет функцию fromNow (), которая возвращает «x дней» или «x часов назад» из текущей даты / времени.

moment([2007, 0, 29]).fromNow();     // 4 years ago
moment([2007, 0, 29]).fromNow(true); // 4 years
Добро пожаловать на сайт PullRequest, где вы можете задавать вопросы и получать ответы от других членов сообщества.
...