| // 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 */ |