1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
|
// Copyright (C) 2011, John 'Warthog9' Hawley <warthog9@eaglescrag.net>
// 2011, Jakub Narebski <jnareb@gmail.com>
/**
* @fileOverview Manipulate dates in gitweb output, adjusting timezone
* @license GPLv2 or later
*/
/**
* Get common timezone, add UI for changing timezones, and adjust
* dates to use requested common timezone.
*
* This function is called during onload event (added to window.onload).
*
* @param {String} tzDefault: default timezone, if there is no cookie
* @param {Object} tzCookieInfo: object literal with info about cookie to store timezone
* @param {String} tzCookieInfo.name: name of cookie to store timezone
* @param {String} tzClassName: denotes elements with date to be adjusted
*/
function onloadTZSetup(tzDefault, tzCookieInfo, tzClassName) {
var tzCookieTZ = getCookie(tzCookieInfo.name, tzCookieInfo);
var tz = tzDefault;
if (tzCookieTZ) {
// set timezone to value saved in a cookie
tz = tzCookieTZ;
// refresh cookie, so its expiration counts from last use of gitweb
setCookie(tzCookieInfo.name, tzCookieTZ, tzCookieInfo);
}
// add UI for changing timezone
addChangeTZ(tz, tzCookieInfo, tzClassName);
// server-side of gitweb produces datetime in UTC,
// so if tz is 'utc' there is no need for changes
var nochange = tz === 'utc';
// adjust dates to use specified common timezone
fixDatetimeTZ(tz, tzClassName, nochange);
}
/* ...................................................................... */
/* Changing dates to use requested timezone */
/**
* Replace RFC-2822 dates contained in SPAN elements with tzClassName
* CSS class with equivalent dates in given timezone.
*
* @param {String} tz: numeric timezone in '(-|+)HHMM' format, or 'utc', or 'local'
* @param {String} tzClassName: specifies elements to be changed
* @param {Boolean} nochange: markup for timezone change, but don't change it
*/
function fixDatetimeTZ(tz, tzClassName, nochange) {
// sanity check, method should be ensured by common-lib.js
if (!document.getElementsByClassName) {
return;
}
// translate to timezone in '(-|+)HHMM' format
tz = normalizeTimezoneInfo(tz);
// NOTE: result of getElementsByClassName should probably be cached
var classesFound = document.getElementsByClassName(tzClassName, "span");
for (var i = 0, len = classesFound.length; i < len; i++) {
var curElement = classesFound[i];
curElement.title = 'Click to change timezone';
if (!nochange) {
// we use *.firstChild.data (W3C DOM) instead of *.innerHTML
// as the latter doesn't always work everywhere in every browser
var epoch = parseRFC2822Date(curElement.firstChild.data);
var adjusted = formatDateRFC2882(epoch, tz);
curElement.firstChild.data = adjusted;
}
}
}
/* ...................................................................... */
/* Adding triggers, generating timezone menu, displaying and hiding */
/**
* Adds triggers for UI to change common timezone used for dates in
* gitweb output: it marks up and/or creates item to click to invoke
* timezone change UI, creates timezone UI fragment to be attached,
* and installs appropriate onclick trigger (via event delegation).
*
* @param {String} tzSelected: pre-selected timezone,
* 'utc' or 'local' or '(-|+)HHMM'
* @param {Object} tzCookieInfo: object literal with info about cookie to store timezone
* @param {String} tzClassName: specifies elements to install trigger
*/
function addChangeTZ(tzSelected, tzCookieInfo, tzClassName) {
// make link to timezone UI discoverable
addCssRule('.'+tzClassName + ':hover',
'text-decoration: underline; cursor: help;');
// create form for selecting timezone (to be saved in a cookie)
var tzSelectFragment = document.createDocumentFragment();
tzSelectFragment = createChangeTZForm(tzSelectFragment,
tzSelected, tzCookieInfo, tzClassName);
// event delegation handler for timezone selection UI (clicking on entry)
// see http://www.nczonline.net/blog/2009/06/30/event-delegation-in-javascript/
// assumes that there is no existing document.onclick handler
document.onclick = function onclickHandler(event) {
//IE doesn't pass in the event object
event = event || window.event;
//IE uses srcElement as the target
var target = event.target || event.srcElement;
switch (target.className) {
case tzClassName:
// don't display timezone menu if it is already displayed
if (tzSelectFragment.childNodes.length > 0) {
displayChangeTZForm(target, tzSelectFragment);
}
break;
} // end switch
};
}
/**
* Create DocumentFragment with UI for changing common timezone in
* which dates are shown in.
*
* @param {DocumentFragment} documentFragment: where attach UI
* @param {String} tzSelected: default (pre-selected) timezone
* @param {Object} tzCookieInfo: object literal with info about cookie to store timezone
* @returns {DocumentFragment}
*/
function createChangeTZForm(documentFragment, tzSelected, tzCookieInfo, tzClassName) {
var div = document.createElement("div");
div.className = 'popup';
/* '<div class="close-button" title="(click on this box to close)">X</div>' */
var closeButton = document.createElement('div');
closeButton.className = 'close-button';
closeButton.title = '(click on this box to close)';
closeButton.appendChild(document.createTextNode('X'));
closeButton.onclick = closeTZFormHandler(documentFragment, tzClassName);
div.appendChild(closeButton);
/* 'Select timezone: <br clear="all">' */
div.appendChild(document.createTextNode('Select timezone: '));
var br = document.createElement('br');
br.clear = 'all';
div.appendChild(br);
/* '<select name="tzoffset">
* ...
* <option value="-0700">UTC-07:00</option>
* <option value="-0600">UTC-06:00</option>
* ...
* </select>' */
var select = document.createElement("select");
select.name = "tzoffset";
//select.style.clear = 'all';
select.appendChild(generateTZOptions(tzSelected));
select.onchange = selectTZHandler(documentFragment, tzCookieInfo, tzClassName);
div.appendChild(select);
documentFragment.appendChild(div);
return documentFragment;
}
/**
* Hide (remove from DOM) timezone change UI, ensuring that it is not
* garbage collected and that it can be re-enabled later.
*
* @param {DocumentFragment} documentFragment: contains detached UI
* @param {HTMLSelectElement} target: select element inside of UI
* @param {String} tzClassName: specifies element where UI was installed
* @returns {DocumentFragment} documentFragment
*/
function removeChangeTZForm(documentFragment, target, tzClassName) {
// find containing element, where we appended timezone selection UI
// `target' is somewhere inside timezone menu
var container = target.parentNode, popup = target;
while (container &&
container.className !== tzClassName) {
popup = container;
container = container.parentNode;
}
// safety check if we found correct container,
// and if it isn't deleted already
if (!container || !popup ||
container.className !== tzClassName ||
popup.className !== 'popup') {
return documentFragment;
}
// timezone selection UI was appended as last child
// see also displayChangeTZForm function
var removed = popup.parentNode.removeChild(popup);
if (documentFragment.firstChild !== removed) { // the only child
// re-append it so it would be available for next time
documentFragment.appendChild(removed);
}
// all of inline style was added by this script
// it is not really needed to remove it, but it is a good practice
container.removeAttribute('style');
return documentFragment;
}
/**
* Display UI for changing common timezone for dates in gitweb output.
* To be used from 'onclick' event handler.
*
* @param {HTMLElement} target: where to install/display UI
* @param {DocumentFragment} tzSelectFragment: timezone selection UI
*/
function displayChangeTZForm(target, tzSelectFragment) {
// for absolute positioning to be related to target element
target.style.position = 'relative';
target.style.display = 'inline-block';
// show/display UI for changing timezone
target.appendChild(tzSelectFragment);
}
/* ...................................................................... */
/* List of timezones for timezone selection menu */
/**
* Generate list of timezones for creating timezone select UI
*
* @returns {Object[]} list of e.g. { value: '+0100', descr: 'GMT+01:00' }
*/
function generateTZList() {
var timezones = [
{ value: "utc", descr: "UTC/GMT"},
{ value: "local", descr: "Local (per browser)"}
];
// generate all full hour timezones (no fractional timezones)
for (var x = -12, idx = timezones.length; x <= +14; x++, idx++) {
var hours = (x >= 0 ? '+' : '-') + padLeft(x >=0 ? x : -x, 2);
timezones[idx] = { value: hours + '00', descr: 'UTC' + hours + ':00'};
if (x === 0) {
timezones[idx].descr = 'UTC\u00B100:00'; // 'UTC±00:00'
}
}
return timezones;
}
/**
* Generate <options> elements for timezone select UI
*
* @param {String} tzSelected: default timezone
* @returns {DocumentFragment} list of options elements to appendChild
*/
function generateTZOptions(tzSelected) {
var elems = document.createDocumentFragment();
var timezones = generateTZList();
for (var i = 0, len = timezones.length; i < len; i++) {
var tzone = timezones[i];
var option = document.createElement("option");
if (tzone.value === tzSelected) {
option.defaultSelected = true;
}
option.value = tzone.value;
option.appendChild(document.createTextNode(tzone.descr));
elems.appendChild(option);
}
return elems;
}
/* ...................................................................... */
/* Event handlers and/or their generators */
/**
* Create event handler that select timezone and closes timezone select UI.
* To be used as $('select[name="tzselect"]').onchange handler.
*
* @param {DocumentFragment} tzSelectFragment: timezone selection UI
* @param {Object} tzCookieInfo: object literal with info about cookie to store timezone
* @param {String} tzCookieInfo.name: name of cookie to save result of selection
* @param {String} tzClassName: specifies element where UI was installed
* @returns {Function} event handler
*/
function selectTZHandler(tzSelectFragment, tzCookieInfo, tzClassName) {
//return function selectTZ(event) {
return function (event) {
event = event || window.event;
var target = event.target || event.srcElement;
var selected = target.options.item(target.selectedIndex);
removeChangeTZForm(tzSelectFragment, target, tzClassName);
if (selected) {
selected.defaultSelected = true;
setCookie(tzCookieInfo.name, selected.value, tzCookieInfo);
fixDatetimeTZ(selected.value, tzClassName);
}
};
}
/**
* Create event handler that closes timezone select UI.
* To be used e.g. as $('.closebutton').onclick handler.
*
* @param {DocumentFragment} tzSelectFragment: timezone selection UI
* @param {String} tzClassName: specifies element where UI was installed
* @returns {Function} event handler
*/
function closeTZFormHandler(tzSelectFragment, tzClassName) {
//return function closeTZForm(event) {
return function (event) {
event = event || window.event;
var target = event.target || event.srcElement;
removeChangeTZForm(tzSelectFragment, target, tzClassName);
};
}
/* end of adjust-timezone.js */
|