// Copyright (C) 2007, Fredrik Kuivinen // 2007, Petr Baudis // 2008-2011, Jakub Narebski /** * @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 */