You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
224 lines
6.4 KiB
224 lines
6.4 KiB
// Copyright (C) 2007, Fredrik Kuivinen <frekui@gmail.com> |
|
// 2007, Petr Baudis <pasky@suse.cz> |
|
// 2008-2011, Jakub Narebski <jnareb@gmail.com> |
|
|
|
/** |
|
* @fileOverview Generic JavaScript code (helper functions) |
|
* @license GPLv2 or later |
|
*/ |
|
|
|
|
|
/* ============================================================ */ |
|
/* ............................................................ */ |
|
/* Padding */ |
|
|
|
/** |
|
* pad INPUT on the left with STR that is assumed to have visible |
|
* width of single character (for example nonbreakable spaces), |
|
* to WIDTH characters |
|
* |
|
* example: padLeftStr(12, 3, '\u00A0') == '\u00A012' |
|
* ('\u00A0' is nonbreakable space) |
|
* |
|
* @param {Number|String} input: number to pad |
|
* @param {Number} width: visible width of output |
|
* @param {String} str: string to prefix to string, defaults to '\u00A0' |
|
* @returns {String} INPUT prefixed with STR x (WIDTH - INPUT.length) |
|
*/ |
|
function padLeftStr(input, width, str) { |
|
var prefix = ''; |
|
if (typeof str === 'undefined') { |
|
ch = '\u00A0'; // using ' ' doesn't work in all browsers |
|
} |
|
|
|
width -= input.toString().length; |
|
while (width > 0) { |
|
prefix += str; |
|
width--; |
|
} |
|
return prefix + input; |
|
} |
|
|
|
/** |
|
* Pad INPUT on the left to WIDTH, using given padding character CH, |
|
* for example padLeft('a', 3, '_') is '__a' |
|
* padLeft(4, 2) is '04' (same as padLeft(4, 2, '0')) |
|
* |
|
* @param {String} input: input value converted to string. |
|
* @param {Number} width: desired length of output. |
|
* @param {String} ch: single character to prefix to string, defaults to '0'. |
|
* |
|
* @returns {String} Modified string, at least SIZE length. |
|
*/ |
|
function padLeft(input, width, ch) { |
|
var s = input + ""; |
|
if (typeof ch === 'undefined') { |
|
ch = '0'; |
|
} |
|
|
|
while (s.length < width) { |
|
s = ch + s; |
|
} |
|
return s; |
|
} |
|
|
|
|
|
/* ............................................................ */ |
|
/* Handling browser incompatibilities */ |
|
|
|
/** |
|
* Create XMLHttpRequest object in cross-browser way |
|
* @returns XMLHttpRequest object, or null |
|
*/ |
|
function createRequestObject() { |
|
try { |
|
return new XMLHttpRequest(); |
|
} catch (e) {} |
|
try { |
|
return window.createRequest(); |
|
} catch (e) {} |
|
try { |
|
return new ActiveXObject("Msxml2.XMLHTTP"); |
|
} catch (e) {} |
|
try { |
|
return new ActiveXObject("Microsoft.XMLHTTP"); |
|
} catch (e) {} |
|
|
|
return null; |
|
} |
|
|
|
|
|
/** |
|
* Insert rule giving specified STYLE to given SELECTOR at the end of |
|
* first CSS stylesheet. |
|
* |
|
* @param {String} selector: CSS selector, e.g. '.class' |
|
* @param {String} style: rule contents, e.g. 'background-color: red;' |
|
*/ |
|
function addCssRule(selector, style) { |
|
var stylesheet = document.styleSheets[0]; |
|
|
|
var theRules = []; |
|
if (stylesheet.cssRules) { // W3C way |
|
theRules = stylesheet.cssRules; |
|
} else if (stylesheet.rules) { // IE way |
|
theRules = stylesheet.rules; |
|
} |
|
|
|
if (stylesheet.insertRule) { // W3C way |
|
stylesheet.insertRule(selector + ' { ' + style + ' }', theRules.length); |
|
} else if (stylesheet.addRule) { // IE way |
|
stylesheet.addRule(selector, style); |
|
} |
|
} |
|
|
|
|
|
/* ............................................................ */ |
|
/* Support for legacy browsers */ |
|
|
|
/** |
|
* Provides getElementsByClassName method, if there is no native |
|
* implementation of this method. |
|
* |
|
* NOTE that there are limits and differences compared to native |
|
* getElementsByClassName as defined by e.g.: |
|
* https://developer.mozilla.org/en/DOM/document.getElementsByClassName |
|
* http://www.whatwg.org/specs/web-apps/current-work/multipage/dom.html#dom-getelementsbyclassname |
|
* http://www.whatwg.org/specs/web-apps/current-work/multipage/dom.html#dom-document-getelementsbyclassname |
|
* |
|
* Namely, this implementation supports only single class name as |
|
* argument and not set of space-separated tokens representing classes, |
|
* it returns Array of nodes rather than live NodeList, and has |
|
* additional optional argument where you can limit search to given tags |
|
* (via getElementsByTagName). |
|
* |
|
* Based on |
|
* http://code.google.com/p/getelementsbyclassname/ |
|
* http://www.dustindiaz.com/getelementsbyclass/ |
|
* http://stackoverflow.com/questions/1818865/do-we-have-getelementsbyclassname-in-javascript |
|
* |
|
* See also http://ejohn.org/blog/getelementsbyclassname-speed-comparison/ |
|
* |
|
* @param {String} class: name of _single_ class to find |
|
* @param {String} [taghint] limit search to given tags |
|
* @returns {Node[]} array of matching elements |
|
*/ |
|
if (!('getElementsByClassName' in document)) { |
|
document.getElementsByClassName = function (classname, taghint) { |
|
taghint = taghint || "*"; |
|
var elements = (taghint === "*" && document.all) ? |
|
document.all : |
|
document.getElementsByTagName(taghint); |
|
var pattern = new RegExp("(^|\\s)" + classname + "(\\s|$)"); |
|
var matches= []; |
|
for (var i = 0, j = 0, n = elements.length; i < n; i++) { |
|
var el= elements[i]; |
|
if (el.className && pattern.test(el.className)) { |
|
// matches.push(el); |
|
matches[j] = el; |
|
j++; |
|
} |
|
} |
|
return matches; |
|
}; |
|
} // end if |
|
|
|
|
|
/* ............................................................ */ |
|
/* unquoting/unescaping filenames */ |
|
|
|
/**#@+ |
|
* @constant |
|
*/ |
|
var escCodeRe = /\\([^0-7]|[0-7]{1,3})/g; |
|
var octEscRe = /^[0-7]{1,3}$/; |
|
var maybeQuotedRe = /^\"(.*)\"$/; |
|
/**#@-*/ |
|
|
|
/** |
|
* unquote maybe C-quoted filename (as used by git, i.e. it is |
|
* in double quotes '"' if there is any escape character used) |
|
* e.g. 'aa' -> 'aa', '"a\ta"' -> 'a a' |
|
* |
|
* @param {String} str: git-quoted string |
|
* @returns {String} Unquoted and unescaped string |
|
* |
|
* @globals escCodeRe, octEscRe, maybeQuotedRe |
|
*/ |
|
function unquote(str) { |
|
function unq(seq) { |
|
var es = { |
|
// character escape codes, aka escape sequences (from C) |
|
// replacements are to some extent JavaScript specific |
|
t: "\t", // tab (HT, TAB) |
|
n: "\n", // newline (NL) |
|
r: "\r", // return (CR) |
|
f: "\f", // form feed (FF) |
|
b: "\b", // backspace (BS) |
|
a: "\x07", // alarm (bell) (BEL) |
|
e: "\x1B", // escape (ESC) |
|
v: "\v" // vertical tab (VT) |
|
}; |
|
|
|
if (seq.search(octEscRe) !== -1) { |
|
// octal char sequence |
|
return String.fromCharCode(parseInt(seq, 8)); |
|
} else if (seq in es) { |
|
// C escape sequence, aka character escape code |
|
return es[seq]; |
|
} |
|
// quoted ordinary character |
|
return seq; |
|
} |
|
|
|
var match = str.match(maybeQuotedRe); |
|
if (match) { |
|
str = match[1]; |
|
// perhaps str = eval('"'+str+'"'); would be enough? |
|
str = str.replace(escCodeRe, |
|
function (substr, p1, offset, s) { return unq(p1); }); |
|
} |
|
return str; |
|
} |
|
|
|
/* end of common-lib.js */
|
|
|