* Revision 0.9.44 fixes some more charset related problems. The user description is now properly decoded for /query views, and the channel name for mode messages. * Revision 0.9.43 fixes the ``unknown CTCP'' handler, so that it displays the correct diagnostic, instead of ``my_unkctcpMsg''. * Revision 0.9.42 fixes charset problems with part reasons, and another charset problem with kick messages. It also deals with spaces in nicknames and usernames by replacing them with underscores, instead of just failing to log in. This revision also makes it so that you no longer need to restart ChatZilla after changing your description or username. * Revision 0.9.41 adds an ``autoperform'' pref to networks that can be used to execute arbitrary ChatZilla commands when you connect to a network. These autoperform commands will be executed before any of your ``Open At Startup'' channels are processed. This release also makes ChatZilla recognize the optional ``reason'' parameter for PART messages. Many IRC servers do not yet support PART reasons, so don't be surprised if you don't see any. git-svn-id: svn://10.0.0.236/trunk@148240 18797224-902f-48f8-a5cc-f745e15eee43
923 lines
22 KiB
JavaScript
923 lines
22 KiB
JavaScript
/* -*- Mode: C++; tab-width: 8; indent-tabs-mode: nil; c-basic-offset: 4 -*-
|
|
*
|
|
* The contents of this file are subject to the Mozilla Public
|
|
* License Version 1.1 (the "License"); you may not use this file
|
|
* except in compliance with the License. You may obtain a copy of
|
|
* the License at http://www.mozilla.org/MPL/
|
|
*
|
|
* Software distributed under the License is distributed on an "AS
|
|
* IS" basis, WITHOUT WARRANTY OF ANY KIND, either express or
|
|
* implied. See the License for the specific language governing
|
|
* rights and limitations under the License.
|
|
*
|
|
* The Original Code is JSIRC Library
|
|
*
|
|
* The Initial Developer of the Original Code is New Dimensions Consulting,
|
|
* Inc. Portions created by New Dimensions Consulting, Inc. are
|
|
* Copyright (C) 1999 New Dimenstions Consulting, Inc. All
|
|
* Rights Reserved.
|
|
*
|
|
* Contributor(s):
|
|
* Robert Ginda, rginda@ndcico.com, original author
|
|
*
|
|
*
|
|
* JavaScript utility functions.
|
|
*
|
|
* 1999-08-15 rginda@ndcico.com v1.0
|
|
*/
|
|
|
|
var utils = new Object();
|
|
|
|
var DEBUG = true;
|
|
var dd;
|
|
|
|
if (DEBUG) {
|
|
var _dd_pfx = "";
|
|
var _dd_singleIndent = " ";
|
|
var _dd_indentLength = _dd_singleIndent.length;
|
|
var _dd_currentIndent = "";
|
|
var _dd_lastDumpWasOpen = false;
|
|
var _dd_timeStack = new Array();
|
|
var _dd_disableDepth = Number.MAX_VALUE;
|
|
var _dd_currentDepth = 0;
|
|
dd = function _dd(str) {
|
|
if (typeof str != "string") {
|
|
dump(str + "\n");
|
|
} else if (str == "") {
|
|
dump("<empty-string>\n");
|
|
} else if (str[str.length - 1] == "{") {
|
|
++_dd_currentDepth;
|
|
if (_dd_currentDepth >= _dd_disableDepth)
|
|
return;
|
|
if (str.indexOf("OFF") == 0)
|
|
_dd_disableDepth = _dd_currentDepth;
|
|
_dd_timeStack.push (new Date());
|
|
if (_dd_lastDumpWasOpen)
|
|
dump("\n");
|
|
dump (_dd_pfx + _dd_currentIndent + str);
|
|
_dd_currentIndent += _dd_singleIndent;
|
|
_dd_lastDumpWasOpen = true;
|
|
} else if (str[0] == "}") {
|
|
if (--_dd_currentDepth >= _dd_disableDepth)
|
|
return;
|
|
_dd_disableDepth = Number.MAX_VALUE;
|
|
var sufx = (new Date() - _dd_timeStack.pop()) / 1000 + " sec";
|
|
_dd_currentIndent =
|
|
_dd_currentIndent.substr(0, _dd_currentIndent.length -
|
|
_dd_indentLength);
|
|
if (_dd_lastDumpWasOpen)
|
|
dump(str + " " + sufx + "\n");
|
|
else
|
|
dump(_dd_pfx + _dd_currentIndent + str + " " +
|
|
sufx + "\n");
|
|
_dd_lastDumpWasOpen = false;
|
|
} else {
|
|
if (_dd_currentDepth >= _dd_disableDepth)
|
|
return;
|
|
if (_dd_lastDumpWasOpen)
|
|
dump("\n");
|
|
dump(_dd_pfx + _dd_currentIndent + str + "\n");
|
|
_dd_lastDumpWasOpen = false;
|
|
}
|
|
}
|
|
} else {
|
|
dd = function (){};
|
|
}
|
|
|
|
var jsenv = new Object();
|
|
jsenv.HAS_SECURITYMANAGER = ((typeof netscape == "object") &&
|
|
(typeof netscape.security == "object"));
|
|
jsenv.HAS_XPCOM = ((typeof Components == "object") &&
|
|
(typeof Components.classes == "object"));
|
|
jsenv.HAS_JAVA = (typeof java == "object");
|
|
jsenv.HAS_RHINO = (typeof defineClass == "function");
|
|
jsenv.HAS_DOCUMENT = (typeof document == "object");
|
|
jsenv.HAS_NSPR_EVENTQ = jsenv.HAS_DOCUMENT;
|
|
jsenv.HAS_STREAM_PROVIDER = ("nsIStreamProvider" in Components.interfaces);
|
|
|
|
function dumpObject (o, pfx, sep)
|
|
{
|
|
var p;
|
|
var s = "";
|
|
|
|
sep = (typeof sep == "undefined") ? " = " : sep;
|
|
pfx = (typeof pfx == "undefined") ? "" : pfx;
|
|
|
|
for (p in o)
|
|
{
|
|
if (typeof (o[p]) != "function")
|
|
s += pfx + p + sep + o[p] + "\n";
|
|
else
|
|
s += pfx + p + sep + "function\n";
|
|
}
|
|
|
|
return s;
|
|
|
|
}
|
|
|
|
/* Dumps an object in tree format, recurse specifiec the the number of objects
|
|
* to recurse, compress is a boolean that can uncompress (true) the output
|
|
* format, and level is the number of levels to intitialy indent (only useful
|
|
* internally.) A sample dumpObjectTree (o, 1) is shown below.
|
|
*
|
|
* + parent (object)
|
|
* + users (object)
|
|
* | + jsbot (object)
|
|
* | + mrjs (object)
|
|
* | + nakkezzzz (object)
|
|
* | *
|
|
* + bans (object)
|
|
* | *
|
|
* + topic (string) 'ircclient.js:59: nothing is not defined'
|
|
* + getUsersLength (function) 9 lines
|
|
* *
|
|
*/
|
|
function dumpObjectTree (o, recurse, compress, level)
|
|
{
|
|
var s = "";
|
|
var pfx = "";
|
|
|
|
if (typeof recurse == "undefined")
|
|
recurse = 0;
|
|
if (typeof level == "undefined")
|
|
level = 0;
|
|
if (typeof compress == "undefined")
|
|
compress = true;
|
|
|
|
for (var i = 0; i < level; i++)
|
|
pfx += (compress) ? "| " : "| ";
|
|
|
|
var tee = (compress) ? "+ " : "+- ";
|
|
|
|
for (i in o)
|
|
{
|
|
var t, ex;
|
|
|
|
try
|
|
{
|
|
t = typeof o[i];
|
|
}
|
|
catch (ex)
|
|
{
|
|
t = "ERROR";
|
|
}
|
|
|
|
switch (t)
|
|
{
|
|
case "function":
|
|
var sfunc = String(o[i]).split("\n");
|
|
if (sfunc[2] == " [native code]")
|
|
sfunc = "[native code]";
|
|
else
|
|
if (sfunc.length == 1)
|
|
sfunc = String(sfunc);
|
|
else
|
|
sfunc = sfunc.length + " lines";
|
|
s += pfx + tee + i + " (function) " + sfunc + "\n";
|
|
break;
|
|
|
|
case "object":
|
|
s += pfx + tee + i + " (object)";
|
|
if (o[i] == null)
|
|
{
|
|
s += " null\n";
|
|
break;
|
|
}
|
|
|
|
s += "\n";
|
|
|
|
if (!compress)
|
|
s += pfx + "|\n";
|
|
if ((i != "parent") && (recurse))
|
|
s += dumpObjectTree (o[i], recurse - 1,
|
|
compress, level + 1);
|
|
break;
|
|
|
|
case "string":
|
|
if (o[i].length > 200)
|
|
s += pfx + tee + i + " (" + t + ") " +
|
|
o[i].length + " chars\n";
|
|
else
|
|
s += pfx + tee + i + " (" + t + ") '" + o[i] + "'\n";
|
|
break;
|
|
|
|
case "ERROR":
|
|
s += pfx + tee + i + " (" + t + ") ?\n";
|
|
break;
|
|
|
|
default:
|
|
s += pfx + tee + i + " (" + t + ") " + o[i] + "\n";
|
|
|
|
}
|
|
|
|
if (!compress)
|
|
s += pfx + "|\n";
|
|
|
|
}
|
|
|
|
s += pfx + "*\n";
|
|
|
|
return s;
|
|
|
|
}
|
|
|
|
function ecmaEscape(str)
|
|
{
|
|
function replaceNonPrintables(ch)
|
|
{
|
|
rv = ch.charCodeAt().toString(16);
|
|
if (rv.length == 1)
|
|
rv = "0" + rv;
|
|
else if (rv.length == 3)
|
|
rv = "u0" + rv;
|
|
else if (rv.length == 4)
|
|
rv = "u" + rv;
|
|
|
|
return "%" + rv;
|
|
};
|
|
|
|
// Replace any character that is not in the 69 character set
|
|
// [ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789@*_+-./]
|
|
// with an escape sequence. Two digit sequences in the form %XX are used
|
|
// for characters whose codepoint is less than 255, %uXXXX for all others.
|
|
// See section B.2.1 of ECMA-262 rev3 for more information.
|
|
return str.replace(/[^A-Za-z0-9@*_+.\-\/]/g, replaceNonPrintables);
|
|
}
|
|
|
|
function ecmaUnescape(str)
|
|
{
|
|
function replaceEscapes(seq)
|
|
{
|
|
var ary = seq.match(/([\da-f]{1,2})(.*)|u([\da-f]{1,4})/);
|
|
if (!ary)
|
|
return "<ERROR>";
|
|
|
|
if (ary[1])
|
|
{
|
|
// two digit escape, possibly with cruft after
|
|
rv = String.fromCharCode(parseInt(ary[1], 16)) + ary[2];
|
|
}
|
|
else
|
|
{
|
|
// four digits, no cruft
|
|
rv = String.fromCharCode(parseInt(ary[3], 16));
|
|
}
|
|
|
|
return rv;
|
|
};
|
|
|
|
// Replace the escape sequences %X, %XX, %uX, %uXX, %uXXX, and %uXXXX with
|
|
// the characters they represent, where X is a hexadecimal digit.
|
|
// See section B.2.2 of ECMA-262 rev3 for more information.
|
|
return str.replace(/%u?([\da-f]{1,4})/ig, replaceEscapes);
|
|
}
|
|
|
|
function replaceVars(str, vars)
|
|
{
|
|
// replace "string $with a $variable", with
|
|
// "string " + vars["with"] + " with a " + vars["variable"]
|
|
|
|
function doReplace(symbol)
|
|
{
|
|
var name = symbol.substr(1);
|
|
if (name in vars)
|
|
return vars[name];
|
|
|
|
return "$" + name;
|
|
};
|
|
|
|
return str.replace(/(\$\w[\w\d\-]+)/g, doReplace);
|
|
}
|
|
|
|
function formatException (ex)
|
|
{
|
|
if (ex instanceof Error)
|
|
return getMsg (MSG_FMT_JSEXCEPTION, [ex.name, ex.message, ex.fileName,
|
|
ex.lineNumber]);
|
|
|
|
return String(ex);
|
|
}
|
|
|
|
/*
|
|
* Clones an existing object (Only the enumerable properties
|
|
* of course.) use as a function..
|
|
* var c = Clone (obj);
|
|
* or a constructor...
|
|
* var c = new Clone (obj);
|
|
*/
|
|
function Clone (obj)
|
|
{
|
|
var robj = new Object();
|
|
|
|
for (var p in obj)
|
|
robj[p] = obj[p];
|
|
|
|
return robj;
|
|
|
|
}
|
|
|
|
function Copy(source, dest, overwrite)
|
|
{
|
|
if (!dest)
|
|
dest = new Object();
|
|
|
|
for (var p in source)
|
|
{
|
|
if (overwrite || !(p in dest))
|
|
dest[p] = source[p];
|
|
}
|
|
|
|
return dest;
|
|
}
|
|
|
|
/*
|
|
* matches a real object against one or more pattern objects.
|
|
* if you pass an array of pattern objects, |negate| controls wether to check
|
|
* if the object matches ANY of the patterns, or NONE of the patterns.
|
|
*/
|
|
function matchObject (o, pattern, negate)
|
|
{
|
|
negate = Boolean(negate);
|
|
|
|
function _match (o, pattern)
|
|
{
|
|
if (pattern instanceof Function)
|
|
return pattern(o);
|
|
|
|
for (p in pattern)
|
|
{
|
|
var val;
|
|
/* nice to have, but slow as molases, allows you to match
|
|
* properties of objects with obj$prop: "foo" syntax */
|
|
/*
|
|
if (p[0] == "$")
|
|
val = eval ("o." +
|
|
p.substr(1,p.length).replace (/\$/g, "."));
|
|
else
|
|
*/
|
|
val = o[p];
|
|
|
|
if (pattern[p] instanceof Function)
|
|
{
|
|
if (!pattern[p](val))
|
|
return false;
|
|
}
|
|
else
|
|
{
|
|
var ary = (new String(val)).match(pattern[p]);
|
|
if (ary == null)
|
|
return false;
|
|
else
|
|
o.matchresult = ary;
|
|
}
|
|
}
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
if (!(pattern instanceof Array))
|
|
return Boolean (negate ^ _match(o, pattern));
|
|
|
|
for (var i in pattern)
|
|
if (_match (o, pattern[i]))
|
|
return !negate;
|
|
|
|
return negate;
|
|
|
|
}
|
|
|
|
function matchEntry (partialName, list)
|
|
{
|
|
|
|
if ((typeof partialName == "undefined") ||
|
|
(String(partialName) == ""))
|
|
return list;
|
|
|
|
var ary = new Array();
|
|
|
|
for (var i in list)
|
|
{
|
|
if (list[i].indexOf(partialName) == 0)
|
|
ary.push (list[i]);
|
|
}
|
|
|
|
return ary;
|
|
|
|
}
|
|
|
|
function encodeChar(ch)
|
|
{
|
|
return "%" + ch.charCodeAt(0).toString(16);
|
|
}
|
|
|
|
function escapeFileName(fileName)
|
|
{
|
|
return fileName.replace(/[^\w\d.,#-_]/g, encodeChar);
|
|
}
|
|
|
|
function getCommonPfx (list)
|
|
{
|
|
var pfx = list[0];
|
|
var l = list.length;
|
|
|
|
for (var i = 0; i < l; i++)
|
|
{
|
|
for (var c = 0; c < pfx.length; ++c)
|
|
{
|
|
if (c >= list[i].length)
|
|
{
|
|
pfx = pfx.substr (0, c);
|
|
break;
|
|
}
|
|
else
|
|
{
|
|
if (pfx[c] != list[i][c])
|
|
pfx = pfx.substr (0, c);
|
|
}
|
|
}
|
|
}
|
|
|
|
return pfx;
|
|
|
|
}
|
|
|
|
function openTopWin (url)
|
|
{
|
|
return openDialog (getBrowserURL(), "_blank", "chrome,all,dialog=no", url);
|
|
}
|
|
|
|
function getWindowByType (windowType)
|
|
{
|
|
const MEDIATOR_CONTRACTID =
|
|
"@mozilla.org/appshell/window-mediator;1";
|
|
const nsIWindowMediator = Components.interfaces.nsIWindowMediator;
|
|
|
|
var windowManager =
|
|
Components.classes[MEDIATOR_CONTRACTID].getService(nsIWindowMediator);
|
|
|
|
return windowManager.getMostRecentWindow(windowType);
|
|
}
|
|
|
|
function renameProperty (obj, oldname, newname)
|
|
{
|
|
|
|
if (oldname == newname)
|
|
return;
|
|
|
|
obj[newname] = obj[oldname];
|
|
delete obj[oldname];
|
|
|
|
}
|
|
|
|
function newObject(contractID, iface)
|
|
{
|
|
if (!jsenv.HAS_XPCOM)
|
|
return null;
|
|
|
|
var obj = Components.classes[contractID].createInstance();
|
|
var rv;
|
|
|
|
switch (typeof iface)
|
|
{
|
|
case "string":
|
|
rv = obj.QueryInterface(Components.interfaces[iface]);
|
|
break;
|
|
|
|
case "object":
|
|
rv = obj.QueryInterface[iface];
|
|
break;
|
|
|
|
default:
|
|
rv = null;
|
|
break;
|
|
}
|
|
|
|
return rv;
|
|
|
|
}
|
|
|
|
function getContentWindow(frame)
|
|
{
|
|
try
|
|
{
|
|
if (!frame || !("contentWindow" in frame))
|
|
return false;
|
|
|
|
return frame.contentWindow;
|
|
}
|
|
catch (ex)
|
|
{
|
|
// throws exception is contentWindow is gone
|
|
return null;
|
|
}
|
|
}
|
|
|
|
function getPriv (priv)
|
|
{
|
|
if (!jsenv.HAS_SECURITYMANAGER)
|
|
return true;
|
|
|
|
var rv = true;
|
|
|
|
try
|
|
{
|
|
netscape.security.PrivilegeManager.enablePrivilege(priv);
|
|
}
|
|
catch (e)
|
|
{
|
|
dd ("getPriv: unable to get privlege '" + priv + "': " + e);
|
|
rv = false;
|
|
}
|
|
|
|
return rv;
|
|
|
|
}
|
|
|
|
function len(o)
|
|
{
|
|
var l = 0;
|
|
|
|
for (var p in o)
|
|
++l;
|
|
|
|
return l;
|
|
}
|
|
|
|
function keys (o)
|
|
{
|
|
var rv = new Array();
|
|
|
|
for (var p in o)
|
|
rv.push(p);
|
|
|
|
return rv;
|
|
|
|
}
|
|
|
|
function stringTrim (s)
|
|
{
|
|
if (!s)
|
|
return "";
|
|
s = s.replace (/^\s+/, "");
|
|
return s.replace (/\s+$/, "");
|
|
|
|
}
|
|
|
|
/* the offset should be in seconds, it will be rounded to 2 decimal places */
|
|
function formatDateOffset (offset, format)
|
|
{
|
|
var seconds = roundTo(offset % 60, 2);
|
|
var minutes = Math.floor(offset / 60);
|
|
var hours = Math.floor(minutes / 60);
|
|
minutes = minutes % 60;
|
|
var days = Math.floor(hours / 24);
|
|
hours = hours % 24;
|
|
|
|
if (!format)
|
|
{
|
|
var ary = new Array();
|
|
if (days > 0)
|
|
ary.push (getMsg(MSG_DAYS, days));
|
|
if (hours > 0)
|
|
ary.push (getMsg(MSG_HOURS, hours));
|
|
if (minutes > 0)
|
|
ary.push (getMsg(MSG_MINUTES, minutes));
|
|
if (seconds > 0 || offset == 0)
|
|
ary.push (getMsg(MSG_SECONDS, seconds));
|
|
|
|
format = ary.join(", ");
|
|
}
|
|
else
|
|
{
|
|
format = format.replace ("%d", days);
|
|
format = format.replace ("%h", hours);
|
|
format = format.replace ("%m", minutes);
|
|
format = format.replace ("%s", seconds);
|
|
}
|
|
|
|
return format;
|
|
}
|
|
|
|
function arrayContains (ary, elem)
|
|
{
|
|
return (arrayIndexOf (ary, elem) != -1);
|
|
}
|
|
|
|
function arrayIndexOf (ary, elem)
|
|
{
|
|
for (var i in ary)
|
|
if (ary[i] == elem)
|
|
return i;
|
|
|
|
return -1;
|
|
}
|
|
|
|
function arrayInsertAt (ary, i, o)
|
|
{
|
|
|
|
ary.splice (i, 0, o);
|
|
|
|
/* doh, forgot about that 'splice' thing
|
|
if (ary.length < i)
|
|
{
|
|
this[i] = o;
|
|
return;
|
|
}
|
|
|
|
for (var j = ary.length; j > i; j--)
|
|
ary[j] = ary[j - 1];
|
|
|
|
ary[i] = o;
|
|
*/
|
|
}
|
|
|
|
function arrayRemoveAt (ary, i)
|
|
{
|
|
|
|
ary.splice (i, 1);
|
|
|
|
/* doh, forgot about that 'splice' thing
|
|
if (ary.length < i)
|
|
return false;
|
|
|
|
for (var j = i; j < ary.length; j++)
|
|
ary[j] = ary[j + 1];
|
|
|
|
ary.length--;
|
|
*/
|
|
|
|
}
|
|
|
|
/* length should be an even number >= 6 */
|
|
function abbreviateWord (str, length)
|
|
{
|
|
if (str.length <= length || length < 6)
|
|
return str;
|
|
|
|
var left = str.substr (0, (length / 2) - 1);
|
|
var right = str.substr (str.length - (length / 2) + 1);
|
|
|
|
return left + "..." + right;
|
|
}
|
|
|
|
/*
|
|
* Inserts the string |hyphen| into string |str| every |pos| characters.
|
|
* If there are any wordbreaking characters in |str| within -/+5 characters of
|
|
* of a |pos| then the hyphen is inserted there instead, in order to produce a
|
|
* "cleaner" break.
|
|
*/
|
|
function hyphenateWord (str, pos, hyphen)
|
|
{
|
|
if (str.length <= pos)
|
|
return str;
|
|
if (typeof hyphen == "undefined")
|
|
hyphen = " ";
|
|
|
|
/* search for a nice place to break the word, fuzzfactor of +/-5, centered
|
|
* around |pos| */
|
|
var splitPos =
|
|
str.substring(pos - 5, pos + 5).search(/[^A-Za-z0-9]/);
|
|
|
|
splitPos = (splitPos != -1) ? pos - 4 + splitPos : pos;
|
|
var left = str.substr (0, splitPos);
|
|
var right = hyphenateWord(str.substr (splitPos), pos, hyphen);
|
|
|
|
return left + hyphen + right;
|
|
}
|
|
|
|
/*
|
|
* Like hyphenateWord, except individual chunks of the word are returned as
|
|
* elements of an array.
|
|
*/
|
|
function splitLongWord (str, pos)
|
|
{
|
|
if (str.length <= pos)
|
|
return [str];
|
|
|
|
var ary = new Array();
|
|
var right = str;
|
|
|
|
while (right.length > pos)
|
|
{
|
|
/* search for a nice place to break the word, fuzzfactor of +/-5,
|
|
* centered around |pos| */
|
|
var splitPos =
|
|
right.substring(pos - 5, pos + 5).search(/[^A-Za-z0-9]/);
|
|
|
|
splitPos = (splitPos != -1) ? pos - 4 + splitPos : pos;
|
|
ary.push(right.substr (0, splitPos));
|
|
right = right.substr (splitPos);
|
|
}
|
|
|
|
ary.push (right);
|
|
|
|
return ary;
|
|
}
|
|
|
|
function getRandomElement (ary)
|
|
{
|
|
|
|
return ary[Math.floor(Math.random() * ary.length)];
|
|
|
|
}
|
|
|
|
function roundTo (num, prec)
|
|
{
|
|
|
|
return Math.round(num * Math.pow (10, prec)) / Math.pow (10, prec);
|
|
|
|
}
|
|
|
|
function randomRange (min, max)
|
|
{
|
|
|
|
if (typeof min == "undefined")
|
|
min = 0;
|
|
|
|
if (typeof max == "undefined")
|
|
max = 1;
|
|
|
|
return Math.floor(Math.random() * (max - min + 1)) + min;
|
|
|
|
}
|
|
|
|
function getStackTrace ()
|
|
{
|
|
|
|
if (!jsenv.HAS_XPCOM)
|
|
return "No stack trace available.";
|
|
|
|
var frame = Components.stack.caller;
|
|
var str = "<top>";
|
|
|
|
while (frame)
|
|
{
|
|
var name = frame.name ? frame.name : "[anonymous]";
|
|
str += "\n" + name + "@" + frame.lineNumber;
|
|
frame = frame.caller;
|
|
}
|
|
|
|
return str;
|
|
|
|
}
|
|
|
|
function getInterfaces (cls)
|
|
{
|
|
if (!jsenv.HAS_XPCOM)
|
|
return null;
|
|
|
|
var rv = new Object();
|
|
var e;
|
|
|
|
for (var i in Components.interfaces)
|
|
{
|
|
try
|
|
{
|
|
var ifc = Components.interfaces[i];
|
|
cls.QueryInterface(ifc);
|
|
rv[i] = ifc;
|
|
}
|
|
catch (e)
|
|
{
|
|
/* nada */
|
|
}
|
|
}
|
|
|
|
return rv;
|
|
|
|
}
|
|
|
|
/**
|
|
* Calls a named function for each element in an array, sending
|
|
* the same parameter each call.
|
|
*
|
|
* @param ary an array of objects
|
|
* @param func_name string name of function to call.
|
|
* @param data data object to pass to each object.
|
|
*/
|
|
function mapObjFunc(ary, func_name, data)
|
|
{
|
|
/*
|
|
* WARNING: Caller assumes resonsibility to verify ary
|
|
* and func_name
|
|
*/
|
|
|
|
for (var i in ary)
|
|
ary[i][func_name](data);
|
|
}
|
|
|
|
/**
|
|
* Passes each element of an array to a given function object.
|
|
*
|
|
* @param func a function object.
|
|
* @param ary an array of values.
|
|
*/
|
|
function map(func, ary) {
|
|
|
|
/*
|
|
* WARNING: Caller assumnes responsibility to verify
|
|
* func and ary.
|
|
*/
|
|
|
|
for (var i in ary)
|
|
func(ary[i]);
|
|
|
|
}
|
|
|
|
function getSpecialDirectory(name)
|
|
{
|
|
if (!("directoryService" in utils))
|
|
{
|
|
const DS_CTR = "@mozilla.org/file/directory_service;1";
|
|
const nsIProperties = Components.interfaces.nsIProperties;
|
|
|
|
utils.directoryService =
|
|
Components.classes[DS_CTR].getService(nsIProperties);
|
|
}
|
|
|
|
return utils.directoryService.get(name, Components.interfaces.nsIFile);
|
|
}
|
|
|
|
function getFileFromURLSpec(url)
|
|
{
|
|
const FILE_CTRID = "@mozilla.org/network/protocol;1?name=file";
|
|
const nsIFileProtocolHandler = Components.interfaces.nsIFileProtocolHandler;
|
|
|
|
var handler = Components.classes[FILE_CTRID].createInstance();
|
|
handler = handler.QueryInterface(nsIFileProtocolHandler);
|
|
return handler.getFileFromURLSpec(url);
|
|
}
|
|
|
|
function getURLSpecFromFile (file)
|
|
{
|
|
if (!file)
|
|
return null;
|
|
|
|
const IOS_CTRID = "@mozilla.org/network/io-service;1";
|
|
const LOCALFILE_CTRID = "@mozilla.org/file/local;1";
|
|
|
|
const nsIIOService = Components.interfaces.nsIIOService;
|
|
const nsILocalFile = Components.interfaces.nsILocalFile;
|
|
|
|
if (typeof file == "string")
|
|
{
|
|
var fileObj =
|
|
Components.classes[LOCALFILE_CTRID].createInstance(nsILocalFile);
|
|
fileObj.initWithPath(file);
|
|
file = fileObj;
|
|
}
|
|
|
|
var service = Components.classes[IOS_CTRID].getService(nsIIOService);
|
|
/* In sept 2002, bug 166792 moved this method to the nsIFileProtocolHandler
|
|
* interface, but we need to support older versions too. */
|
|
if ("getURLSpecFromFile" in service)
|
|
return service.getURLSpecFromFile(file);
|
|
|
|
var nsIFileProtocolHandler = Components.interfaces.nsIFileProtocolHandler;
|
|
var fileHandler = service.getProtocolHandler("file");
|
|
fileHandler = fileHandler.QueryInterface(nsIFileProtocolHandler);
|
|
return fileHandler.getURLSpecFromFile(file);
|
|
}
|
|
|
|
function alert(msg, parent, title)
|
|
{
|
|
var PROMPT_CTRID = "@mozilla.org/embedcomp/prompt-service;1";
|
|
var nsIPromptService = Components.interfaces.nsIPromptService;
|
|
var ps = Components.classes[PROMPT_CTRID].createInstance(nsIPromptService);
|
|
if (!parent)
|
|
parent = window;
|
|
if (!title)
|
|
title = MSG_ALERT;
|
|
ps.alert (parent, title, msg);
|
|
}
|
|
|
|
function confirm(msg, parent, title)
|
|
{
|
|
var PROMPT_CTRID = "@mozilla.org/embedcomp/prompt-service;1";
|
|
var nsIPromptService = Components.interfaces.nsIPromptService;
|
|
var ps = Components.classes[PROMPT_CTRID].createInstance(nsIPromptService);
|
|
if (!parent)
|
|
parent = window;
|
|
if (!title)
|
|
title = MSG_CONFIRM;
|
|
return ps.confirm (parent, title, msg);
|
|
}
|
|
|
|
function prompt(msg, initial, parent, title)
|
|
{
|
|
var PROMPT_CTRID = "@mozilla.org/embedcomp/prompt-service;1";
|
|
var nsIPromptService = Components.interfaces.nsIPromptService;
|
|
var ps = Components.classes[PROMPT_CTRID].createInstance(nsIPromptService);
|
|
if (!parent)
|
|
parent = window;
|
|
if (!title)
|
|
title = MSG_PROMPT;
|
|
rv = { value: initial };
|
|
|
|
if (!ps.prompt (parent, title, msg, rv, null, {value: null}))
|
|
return null;
|
|
|
|
return rv.value
|
|
}
|