aboutsummaryrefslogtreecommitdiffstatshomepage
path: root/js
diff options
context:
space:
mode:
authorPim van den Berg2014-12-13 13:40:36 +0100
committerPim van den Berg2014-12-13 13:43:26 +0100
commit6c39bf06e43fd97fc821cf1dcd11636532512ff0 (patch)
tree543a69efd7bd399a6e125b7a523fe4cf5fc8caf3 /js
parentAdding option to show time since last update on dashboard. (diff)
downloadapt-panopticon_cgp-6c39bf06e43fd97fc821cf1dcd11636532512ff0.zip
apt-panopticon_cgp-6c39bf06e43fd97fc821cf1dcd11636532512ff0.tar.gz
apt-panopticon_cgp-6c39bf06e43fd97fc821cf1dcd11636532512ff0.tar.bz2
apt-panopticon_cgp-6c39bf06e43fd97fc821cf1dcd11636532512ff0.tar.xz
dashboard: make timeago dynamic using jquery + timeago
Diffstat (limited to 'js')
-rw-r--r--js/jquery.timeago.js214
1 files changed, 214 insertions, 0 deletions
diff --git a/js/jquery.timeago.js b/js/jquery.timeago.js
new file mode 100644
index 0000000..483be57
--- /dev/null
+++ b/js/jquery.timeago.js
@@ -0,0 +1,214 @@
1/**
2 * Timeago is a jQuery plugin that makes it easy to support automatically
3 * updating fuzzy timestamps (e.g. "4 minutes ago" or "about 1 day ago").
4 *
5 * @name timeago
6 * @version 1.4.1
7 * @requires jQuery v1.2.3+
8 * @author Ryan McGeary
9 * @license MIT License - http://www.opensource.org/licenses/mit-license.php
10 *
11 * For usage and examples, visit:
12 * http://timeago.yarp.com/
13 *
14 * Copyright (c) 2008-2013, Ryan McGeary (ryan -[at]- mcgeary [*dot*] org)
15 */
16
17(function (factory) {
18 if (typeof define === 'function' && define.amd) {
19 // AMD. Register as an anonymous module.
20 define(['jquery'], factory);
21 } else {
22 // Browser globals
23 factory(jQuery);
24 }
25}(function ($) {
26 $.timeago = function(timestamp) {
27 if (timestamp instanceof Date) {
28 return inWords(timestamp);
29 } else if (typeof timestamp === "string") {
30 return inWords($.timeago.parse(timestamp));
31 } else if (typeof timestamp === "number") {
32 return inWords(new Date(timestamp));
33 } else {
34 return inWords($.timeago.datetime(timestamp));
35 }
36 };
37 var $t = $.timeago;
38
39 $.extend($.timeago, {
40 settings: {
41 refreshMillis: 60000,
42 allowPast: true,
43 allowFuture: false,
44 localeTitle: false,
45 cutoff: 0,
46 strings: {
47 prefixAgo: null,
48 prefixFromNow: null,
49 suffixAgo: "ago",
50 suffixFromNow: "from now",
51 inPast: 'any moment now',
52 seconds: "less than a minute",
53 minute: "about a minute",
54 minutes: "%d minutes",
55 hour: "about an hour",
56 hours: "about %d hours",
57 day: "a day",
58 days: "%d days",
59 month: "about a month",
60 months: "%d months",
61 year: "about a year",
62 years: "%d years",
63 wordSeparator: " ",
64 numbers: []
65 }
66 },
67
68 inWords: function(distanceMillis) {
69 if(!this.settings.allowPast && ! this.settings.allowFuture) {
70 throw 'timeago allowPast and allowFuture settings can not both be set to false.';
71 }
72
73 var $l = this.settings.strings;
74 var prefix = $l.prefixAgo;
75 var suffix = $l.suffixAgo;
76 if (this.settings.allowFuture) {
77 if (distanceMillis < 0) {
78 prefix = $l.prefixFromNow;
79 suffix = $l.suffixFromNow;
80 }
81 }
82
83 if(!this.settings.allowPast && distanceMillis >= 0) {
84 return this.settings.strings.inPast;
85 }
86
87 var seconds = Math.abs(distanceMillis) / 1000;
88 var minutes = seconds / 60;
89 var hours = minutes / 60;
90 var days = hours / 24;
91 var years = days / 365;
92
93 function substitute(stringOrFunction, number) {
94 var string = $.isFunction(stringOrFunction) ? stringOrFunction(number, distanceMillis) : stringOrFunction;
95 var value = ($l.numbers && $l.numbers[number]) || number;
96 return string.replace(/%d/i, value);
97 }
98
99 var words = seconds < 45 && substitute($l.seconds, Math.round(seconds)) ||
100 seconds < 90 && substitute($l.minute, 1) ||
101 minutes < 45 && substitute($l.minutes, Math.round(minutes)) ||
102 minutes < 90 && substitute($l.hour, 1) ||
103 hours < 24 && substitute($l.hours, Math.round(hours)) ||
104 hours < 42 && substitute($l.day, 1) ||
105 days < 30 && substitute($l.days, Math.round(days)) ||
106 days < 45 && substitute($l.month, 1) ||
107 days < 365 && substitute($l.months, Math.round(days / 30)) ||
108 years < 1.5 && substitute($l.year, 1) ||
109 substitute($l.years, Math.round(years));
110
111 var separator = $l.wordSeparator || "";
112 if ($l.wordSeparator === undefined) { separator = " "; }
113 return $.trim([prefix, words, suffix].join(separator));
114 },
115
116 parse: function(iso8601) {
117 var s = $.trim(iso8601);
118 s = s.replace(/\.\d+/,""); // remove milliseconds
119 s = s.replace(/-/,"/").replace(/-/,"/");
120 s = s.replace(/T/," ").replace(/Z/," UTC");
121 s = s.replace(/([\+\-]\d\d)\:?(\d\d)/," $1$2"); // -04:00 -> -0400
122 s = s.replace(/([\+\-]\d\d)$/," $100"); // +09 -> +0900
123 return new Date(s);
124 },
125 datetime: function(elem) {
126 var iso8601 = $t.isTime(elem) ? $(elem).attr("datetime") : $(elem).attr("title");
127 return $t.parse(iso8601);
128 },
129 isTime: function(elem) {
130 // jQuery's `is()` doesn't play well with HTML5 in IE
131 return $(elem).get(0).tagName.toLowerCase() === "time"; // $(elem).is("time");
132 }
133 });
134
135 // functions that can be called via $(el).timeago('action')
136 // init is default when no action is given
137 // functions are called with context of a single element
138 var functions = {
139 init: function(){
140 var refresh_el = $.proxy(refresh, this);
141 refresh_el();
142 var $s = $t.settings;
143 if ($s.refreshMillis > 0) {
144 this._timeagoInterval = setInterval(refresh_el, $s.refreshMillis);
145 }
146 },
147 update: function(time){
148 var parsedTime = $t.parse(time);
149 $(this).data('timeago', { datetime: parsedTime });
150 if($t.settings.localeTitle) $(this).attr("title", parsedTime.toLocaleString());
151 refresh.apply(this);
152 },
153 updateFromDOM: function(){
154 $(this).data('timeago', { datetime: $t.parse( $t.isTime(this) ? $(this).attr("datetime") : $(this).attr("title") ) });
155 refresh.apply(this);
156 },
157 dispose: function () {
158 if (this._timeagoInterval) {
159 window.clearInterval(this._timeagoInterval);
160 this._timeagoInterval = null;
161 }
162 }
163 };
164
165 $.fn.timeago = function(action, options) {
166 var fn = action ? functions[action] : functions.init;
167 if(!fn){
168 throw new Error("Unknown function name '"+ action +"' for timeago");
169 }
170 // each over objects here and call the requested function
171 this.each(function(){
172 fn.call(this, options);
173 });
174 return this;
175 };
176
177 function refresh() {
178 var data = prepareData(this);
179 var $s = $t.settings;
180
181 if (!isNaN(data.datetime)) {
182 if ( $s.cutoff == 0 || Math.abs(distance(data.datetime)) < $s.cutoff) {
183 $(this).text(inWords(data.datetime));
184 }
185 }
186 return this;
187 }
188
189 function prepareData(element) {
190 element = $(element);
191 if (!element.data("timeago")) {
192 element.data("timeago", { datetime: $t.datetime(element) });
193 var text = $.trim(element.text());
194 if ($t.settings.localeTitle) {
195 element.attr("title", element.data('timeago').datetime.toLocaleString());
196 } else if (text.length > 0 && !($t.isTime(element) && element.attr("title"))) {
197 element.attr("title", text);
198 }
199 }
200 return element.data("timeago");
201 }
202
203 function inWords(date) {
204 return $t.inWords(distance(date));
205 }
206
207 function distance(date) {
208 return (new Date().getTime() - date.getTime());
209 }
210
211 // fix for IE6 suckage
212 document.createElement("abbr");
213 document.createElement("time");
214}));