new Javascript libraries from all over the web.

This commit is contained in:
Matthew Wilson 2005-12-01 01:18:45 +00:00
parent 48d69dfd28
commit 8cc0abc682
17 changed files with 7103 additions and 0 deletions

View file

@ -0,0 +1,128 @@
// ===================================================================
// Author: Matt Kruse <matt@ajaxtoolbox.com>
// WWW: http://www.AjaxToolbox.com/
//
// NOTICE: You may use this code for any purpose, commercial or
// private, without any further permission from the author. You may
// remove this notice from your final code if you wish, however it is
// appreciated by the author if at least my web site address is kept.
//
// You may *NOT* re-distribute this code in any way except through its
// use. That means, you can include it in your product, or your web
// site, or any other form where the code is actually being used. You
// may not put the plain javascript up on your site for download or
// include it in your javascript libraries for download.
// If you wish to share this code with others, please just point them
// to the URL instead.
// Please DO NOT link directly to my .js files from your site. Copy
// the files to your server and use them there. Thank you.
// ===================================================================
function AjaxRequest(){var req =new Object();
req.timeout =null;
req.generateUniqueUrl =true;
req.url =window.location.href;
req.method ="GET";
req.async =true;
req.username =null;
req.password =null;
req.parameters =new Object();
req.requestIndex =AjaxRequest.numAjaxRequests++;
req.responseReceived =false;
req.groupName =null;
req.queryString ="";
req.responseText =null;
req.responseXML =null;
req.status =null;
req.statusText =null;
req.aborted =false;
req.xmlHttpRequest =null;
req.onTimeout=null;
req.onLoading=null;
req.onLoaded=null;
req.onInteractive=null;
req.onComplete=null;
req.onSuccess=null;
req.onError=null;
req.onGroupBegin=null;
req.onGroupEnd=null;
req.xmlHttpRequest =AjaxRequest.getXmlHttpRequest();
if(req.xmlHttpRequest==null){return null;}req.xmlHttpRequest.onreadystatechange =
function(){if(req==null || req.xmlHttpRequest==null){return;}if(req.xmlHttpRequest.readyState==1){req.onLoadingInternal(req);}if(req.xmlHttpRequest.readyState==2){req.onLoadedInternal(req);}if(req.xmlHttpRequest.readyState==3){req.onInteractiveInternal(req);}if(req.xmlHttpRequest.readyState==4){req.onCompleteInternal(req);}};
req.onLoadingInternalHandled=false;
req.onLoadedInternalHandled=false;
req.onInteractiveInternalHandled=false;
req.onCompleteInternalHandled=false;
req.onLoadingInternal=
function(){if(req.onLoadingInternalHandled){return;}AjaxRequest.numActiveAjaxRequests++;
if(AjaxRequest.numActiveAjaxRequests==1 && typeof(window['AjaxRequestBegin'])=="function"){AjaxRequestBegin();}if(req.groupName!=null){if(typeof(AjaxRequest.numActiveAjaxGroupRequests[req.groupName])=="undefined"){AjaxRequest.numActiveAjaxGroupRequests[req.groupName] =0;}AjaxRequest.numActiveAjaxGroupRequests[req.groupName]++;
if(AjaxRequest.numActiveAjaxGroupRequests[req.groupName]==1 && typeof(req.onGroupBegin)=="function"){req.onGroupBegin(req.groupName);}}if(typeof(req.onLoading)=="function"){req.onLoading(req);}req.onLoadingInternalHandled=true;};
req.onLoadedInternal=
function(){if(req.onLoadedInternalHandled){return;}if(typeof(req.onLoaded)=="function"){req.onLoaded(req);}req.onLoadedInternalHandled=true;};
req.onInteractiveInternal=
function(){if(req.onInteractiveInternalHandled){return;}if(typeof(req.onInteractive)=="function"){req.onInteractive(req);}req.onInteractiveInternalHandled=true;};
req.onCompleteInternal=
function(){if(req.onCompleteInternalHandled || req.aborted){return;}req.onCompleteInternalHandled=true;
AjaxRequest.numActiveAjaxRequests--;
if(AjaxRequest.numActiveAjaxRequests==0 && typeof(window['AjaxRequestEnd'])=="function"){AjaxRequestEnd(req.groupName);}if(req.groupName!=null){AjaxRequest.numActiveAjaxGroupRequests[req.groupName]--;
if(AjaxRequest.numActiveAjaxGroupRequests[req.groupName]==0 && typeof(req.onGroupEnd)=="function"){req.onGroupEnd(req.groupName);}}req.responseReceived =true;
req.status =req.xmlHttpRequest.status;
req.statusText =req.xmlHttpRequest.statusText;
req.responseText =req.xmlHttpRequest.responseText;
req.responseXML =req.xmlHttpRequest.responseXML;
if(typeof(req.onComplete)=="function"){req.onComplete(req);}if(req.xmlHttpRequest.status==200 && typeof(req.onSuccess)=="function"){req.onSuccess(req);}else if(typeof(req.onError)=="function"){req.onError(req);}delete req.xmlHttpRequest['onreadystatechange'];
req.xmlHttpRequest =null;};
req.onTimeoutInternal=
function(){if(req!=null && req.xmlHttpRequest!=null && !req.onCompleteInternalHandled){req.aborted =true;
req.xmlHttpRequest.abort();
AjaxRequest.numActiveAjaxRequests--;
if(AjaxRequest.numActiveAjaxRequests==0 && typeof(window['AjaxRequestEnd'])=="function"){AjaxRequestEnd(req.groupName);}if(req.groupName!=null){AjaxRequest.numActiveAjaxGroupRequests[req.groupName]--;
if(AjaxRequest.numActiveAjaxGroupRequests[req.groupName]==0 && typeof(req.onGroupEnd)=="function"){req.onGroupEnd(req.groupName);}}if(typeof(req.onTimeout)=="function"){req.onTimeout(req);}delete req.xmlHttpRequest['onreadystatechange'];
req.xmlHttpRequest =null;}};
req.process =
function(){if(req.xmlHttpRequest!=null){if(req.generateUniqueUrl && req.method=="GET"){req.parameters["AjaxRequestUniqueId"] =new Date().getTime() + "" + req.requestIndex;}var content =null;
for(var i in req.parameters){if(req.queryString.length>0){req.queryString +="&";}req.queryString +=encodeURIComponent(i) + "=" + encodeURIComponent(req.parameters[i]);}if(req.method=="GET"){if(req.queryString.length>0){req.url +=((req.url.indexOf("?")>-1)?"&":"?") + req.queryString;}}req.xmlHttpRequest.open(req.method,req.url,req.async,req.username,req.password);
if(req.method=="POST"){if(typeof(req.xmlHttpRequest.setRequestHeader)!="undefined"){req.xmlHttpRequest.setRequestHeader('Content-type', 'application/x-www-form-urlencoded');}content =req.queryString;}if(req.timeout>0){setTimeout(req.onTimeoutInternal,req.timeout);}req.xmlHttpRequest.send(content);}};
req.handleArguments =
function(args){for(var i in args){if(typeof(req[i])=="undefined"){req.parameters[i] =args[i];}else{req[i] =args[i];}}};
req.getAllResponseHeaders =
function(){if(req.xmlHttpRequest!=null){if(req.responseReceived){return req.xmlHttpRequest.getAllResponseHeaders();}alert("Cannot getAllResponseHeaders because a response has not yet been received");}};
req.getResponseHeader =
function(headerName){if(req.xmlHttpRequest!=null){if(req.responseReceived){return req.xmlHttpRequest.getResponseHeader(headerName);}alert("Cannot getResponseHeader because a response has not yet been received");}};
return req;}AjaxRequest.getXmlHttpRequest =function(){if(window.XMLHttpRequest){return new XMLHttpRequest();}else if(window.ActiveXObject){/*@cc_on @*/
/*@if(@_jscript_version >=5)
try{return new ActiveXObject("Msxml2.XMLHTTP");}catch(e){try{return new ActiveXObject("Microsoft.XMLHTTP");}catch(E){return null;}}@end @*/}else{return null;}};
AjaxRequest.isActive =function(){return(AjaxRequest.numActiveAjaxRequests>0);};
AjaxRequest.get =function(args){AjaxRequest.doRequest("GET",args);};
AjaxRequest.post =function(args){AjaxRequest.doRequest("POST",args);};
AjaxRequest.doRequest =function(method,args){if(typeof(args)!="undefined" && args!=null){var myRequest =new AjaxRequest();
myRequest.method =method;
myRequest.handleArguments(args);
myRequest.process();}};
AjaxRequest.submit =function(theform, args){var myRequest =new AjaxRequest();
if(myRequest==null){return false;}var serializedForm =AjaxRequest.serializeForm(theform);
myRequest.method =theform.method.toUpperCase();
myRequest.url =theform.action;
myRequest.handleArguments(args);
myRequest.queryString =serializedForm;
myRequest.process();
return true;};
AjaxRequest.serializeForm =function(theform){var els =theform.elements;
var len =els.length;
var queryString ="";
this.addField =
function(name,value){if(queryString.length>0){queryString +="&";}queryString +=encodeURIComponent(name) + "=" + encodeURIComponent(value);};
for(var i=0;i<len;i++){var el =els[i];
if(!el.disabled){switch(el.type){case 'text': case 'password': case 'hidden': case 'textarea':
this.addField(el.name,el.value);
break;
case 'select-one':
if(el.selectedIndex>=0){this.addField(el.name,el.options[el.selectedIndex].value);}break;
case 'select-multiple':
for(var j=0;j<el.options.length;j++){if(el.options[j].selected){this.addField(el.name,el.options[j].value);}}break;
case 'checkbox': case 'radio':
if(el.checked){this.addField(el.name,el.value);}break;}}}return queryString;};
AjaxRequest.numActiveAjaxRequests =0;
AjaxRequest.numActiveAjaxGroupRequests =new Object();
AjaxRequest.numAjaxRequests =0;

863
www/extras/js/b/beyond.js Normal file
View file

@ -0,0 +1,863 @@
/*
beyond.js
by Dan Shappir and Sjoerd Visscher
For more information see http://w3future.com/html/beyondJS
*/
var beyondVer = 1.00;
var _FP = Function.prototype;
if ( typeof(parseInt.apply) != "function" ) {
_FP.apply = function(obj, args) {
var s, tmp;
if ( obj ) {
tmp = "__tmp__"; // + Math.random();
switch ( typeof(obj) ) {
case "string":
s = "'" + obj.replace(_FP.apply.re, "\\'") + "'";
_SP[tmp] = this;
break;
case "number":
s = "(" + obj + ")";
_NP[tmp] = this;
break;
default:
s = "obj";
obj[tmp] = this;
break;
}
s = "var r = " + s + "." + tmp + "(";
}
else
s = "var r = this(";
for ( var i = 0 ; i < args.length ; ++i )
s += "args[" + i + "],";
s = ( args.length ? s.slice(0, -1) : s ) + ");";
eval(s);
return r;
};
_FP.apply.re = /\'/g;
_FP.call = function(obj) {
return this.apply(obj, Array.from(arguments).slice(1));
};
}
function isUndefined(x) {
var t = typeof(x);
return t == "undefined" || t == "unknown";
}
function isDefined(x) {
return !isUndefined(x);
}
Object.propertyNames = function(x, a) {
if ( !a ) a = [];
for ( var i in x )
a.append(i);
return a;
};
Object.propertyValues = function(x, a) {
return Object.propertyNames(x).collect(a, Function.from(x, null));
};
Object.toArray = function(x) {
return Object.propertyNames(x).zip(Object.propertyValues(x));
};
Object.from = function(n, v, x) {
if ( !x ) x = {};
var length = Math.min(n.length, v.length);
for ( var i = 0 ; i < length ; ++i )
x[n[i]] = v[i];
return x;
};
_FP.force = function() {
return "returnValue" in this ? this.returnValue : this.returnValue = this.call();
};
_FP.delayed = function() {
return Function.from(this.curry(arguments), "force");
};
_FP.caching = function() {
var self = this;
var map = {};
return function() {
var s = Array.from(arguments).toString();
return isDefined(map[s]) ? map[s] : map[s] = self.apply(this, arguments);
}.withArgString(this);
};
_FP.withArgString = function(x) {
if ( !x ) return this;
if ( typeof(x) != "string" )
x = Function.argString(x);
var self = this;
eval("function __withArgString__(" + x + ") { return self.apply(this, arguments); }");
return __withArgString__;
};
_FP.curry = function(m) {
var self = this;
var map = arguments.length != 1 || !m || typeof(m) != "object" ? arguments : m;
function createApplyArg(x) {
var val = map[x];
if ( val && val.asArgNameFlag ) {
args.append(val.toString());
return val.toString();
}
var arg = "__curry__.map['" + x + "']";
if ( typeof(val) == "function" && val.asValueFlag ) {
var list = addArgs(val);
if ( list.length ) list = "," + list;
arg += ".call(this" + list + ")";
}
return arg;
}
function addArgs(f) {
var args = Function.argString(f);
if ( !args )
return "";
args.split(_FP.curry.re).foreach(function(arg) {
if ( argSet.addToSet(arg) )
extraArgs.append(arg);
});
return args;
}
var origargs = Function.argString(this);
origargs = origargs ? origargs.split(_FP.curry.re) : [];
if ( map !== m && origargs.length < arguments.length ) origargs.length = arguments.length;
var args = [], extraArgs = [], argSet = new stringSet;
var applyArgs = origargs.collect(function(a, i) {
if ( isUndefined(a) ) {
if ( isDefined(map[i]) )
return createApplyArg(i);
a = "arg" + i;
args.append(a);
return a;
}
if ( isDefined(map[a]) )
return createApplyArg(a);
if ( isDefined(map[i]) )
return createApplyArg(i);
if ( isDefined(map[i - self.length]) )
return createApplyArg(i - self.length);
args.append(a);
return a;
});
eval("function __curry__(" + args.concat(extraArgs).join(",") + ") {" +
"return self.apply(this, " +
(applyArgs.length == 1 ? "[].append(" + applyArgs[0] + ")" : "[" + applyArgs.join(",") + "]") +
"); }");
__curry__.map = map;
return __curry__;
};
_FP.curry.re = /\s*,\s*/;
_FP.using = function() {
return this.curry(Array.from(arguments).collect(select.curry("typeof", {
"function" : function(x) { return x.asValue(); },
"string" : function(x) { return x.asArgName(); },
"undefined" : function(x) { return x; }
})));
};
_FP.andThen = function(f) {
eval("function first(" + Function.argString(this) + ") { return g.first.apply(this, arguments); }");
function second(x) {
return g.second.apply ? g.second.apply(this, arguments) : g.second(x);
}
var g = second.using(first);
g.first = this;
g.second = typeof(f) != "string" ? f : Function.from(null, f);
return g;
};
_FP.andReturn = function(v) {
return this.andThen(Function.from(v));
};
_FP.replace = function(orig, replace) {
if ( !replace )
replace = Function.NOP;
if ( this.second == orig )
this.second = replace;
else for ( var f = this ; f.second ; f = f.second )
if ( f.first == orig )
f.first = replace;
return this;
};
_FP.strict = function(args) {
var self = this, type = typeof(args), length;
if ( type == "number" )
length = args;
if ( type != "string" )
args = Function.argString(this);
if ( type != "number" )
length = args.split(",").length;
return function() {
return self.apply(this, Array.from(arguments).head(length));
}.withArgString(args);
};
_FP.delay = function(iMilliSeconds) {
var self = this;
if ( !iMilliSeconds )
iMilliSeconds = 0;
return function() {
var args = arguments;
return window.setTimeout(function() { self.apply(this, args); }, iMilliSeconds);
}.withArgString(this);
};
_FP.asValue = function() {
var f = this.curry({});
f.asValueFlag = true;
return f;
};
_FP.toMethod = function() {
return this.using(Function.This);
};
_FP.gate = function(cond, els) {
var gate = function() {
return ( typeof(gate.cond) == "function" ? gate.cond.apply(this, arguments) : gate.cond ) ?
gate.then.apply(this, arguments) :
typeof(gate.els) == "function" ? gate.els.apply(this, arguments) : gate.els;
}.withArgString(this);
gate.cond = cond;
gate.then = this;
gate.els = els;
return gate;
};
_FP.loop = function(cond) {
var loop = function() {
while ( select("typeof", {
"function" : function(x) { return x.apply(this, arguments); },
"number" : function() { return loop.cond--; },
"undefined" : isDefined(loop.cond) ? loop.cond : true
}, loop.cond) ) {
var r = loop.action.apply(this, arguments);
if ( isDefined(r) )
return r;
}
}.withArgString(this);
loop.cond = cond;
loop.action = this;
return loop;
};
_FP.factory = function() {
var self = this;
return function() {
return eval("new self(" +
(0).upTo(arguments.length-1).join2("arguments[", ",", "]") +
")");
}.withArgString(this);
};
Function.argString = function(f) {
var result = ("" + f).match(Function.argString.re);
return result ? result[1] : null;
};
Function.argString.re = /\(([^)]*)/;
Function.NOP = function() {
if ( arguments.length > 0 )
return arguments[0];
};
Function.This = function() {
return this;
};
Function.args = function() {
return Array.from(arguments);
};
Function.invoke = function(f) {
return f();
};
Function.event = function(e) {
return e ? e : event;
};
Function.from = function(obj, field, args) {
var f;
if ( isUndefined(obj) || obj === null ) {
f = function(obj) {
return Function.from(obj, f.field, args).apply(obj, Array.from(arguments).slice(1));
};
if ( args && args.length )
f = f.withArgString("obj," + args);
}
else if ( isUndefined(field) ) {
if ( !args )
args = Function.argString(obj);
if ( typeof(args) != "string" )
f = function() {
return f.obj;
};
else
f = function() {
return f.obj.apply(this, arguments);
}.withArgString(args);
}
else if ( field === null )
f = function(x) {
return f.obj[x];
};
else if ( typeof(field) == "function" )
f = function() {
return f.field.apply(f.obj, arguments);
}.withArgString(args ? args : Function.argString(field));
else {
if ( !args )
args = Function.argString(obj[field]);
if ( typeof(args) != "string" )
f = function() {
if ( f.field.search(/\s/) > -1 )
return f.obj[f.field];
return eval("f.obj." + f.field);
};
else
f = function() {
if ( f.field.search(/\s/) > -1 )
return f.obj[f.field].apply(f.obj, arguments);
var s = "f.obj." + f.field + "(";
Array.from(arguments).foreach(function(v, i) { s += "arguments[" + i + "],"; });
return eval(( arguments.length ? s.slice(0, -1) : s ) + ");");
}.withArgString(args);
}
f.obj = obj;
f.field = field;
return f;
}
Function.set = function(obj, property) {
var f;
if ( isUndefined(obj) || obj === null )
f = function(obj, value) {
return Function.set(obj, f.property).call(this, value);
};
else if ( isUndefined(property) )
f = function(property, value) {
return Function.set(f.obj, property).call(this, value);
};
else
f = function(value) {
if ( f.property.search(/\s/) > -1 )
f.obj[f.property] = value;
else
eval("f.obj." + f.property + " = value");
return value;
}
f.obj = obj;
f.property = property;
return f;
}
var _AP = Array.prototype;
if ( typeof(_AP.push) != "function" )
_AP.push = function() {
for ( var i = 0 ; i < arguments.length ; ++i )
this[this.length] = arguments[i];
return this.length;
}
if ( typeof(_AP.pop) != "function" )
_AP.pop = function() {
if ( this.length ) {
var item = this[this.length-1];
--this.length;
return item;
}
}
if ( typeof(_AP.shift) != "function" )
_AP.shift = function() {
this.foreach(function(v, i, self) { self[i] = self[i+1]; });
--this.length;
return this;
}
if ( typeof(_AP.unshift) != "function" )
_AP.unshift = function() {
var self = this;
Array.from(arguments).concat(this).foreach(function(v, i) { self[i] = v; });
return this;
}
if ( typeof(_AP.splice) != "function" )
_AP.splice = function(start, deleteCount) {
var a = start > 0 ? this.slice(0, start) : [];
a = a.concat(Array.from(arguments).slice(2), this.slice(start+deleteCount));
var self = this, deleted = this.slice(start, start+deleteCount);
a.foreach(function(v, i) { self[i] = v; });
this.length = a.length;
return deleted;
}
_AP.head = function(length) {
return isUndefined(length) ? this[0] : this.slice(0, length);
};
_AP.tail = function(start) {
return this.slice(start ? start : 1);
};
_AP.itemAt = function(i) {
return this[i];
};
_AP.isEmpty = function() {
return this.length === 0;
};
_AP.empty = function() {
this.length = 0;
return this;
};
_AP.order = function(f) {
if ( !f )
f = function(a, b) {
return a < b ? -1 : a > b ? 1 : 0;
};
return this.sort(function(a, b) {
var r = 0;
a.zip(b).foreach(function(pair) {
if ( r = f(pair[0], pair[1]) )
return false;
});
return r;
});
};
_AP.join2 = function(prefix, infix, postfix) {
if ( isUndefined(prefix) || prefix === null ) prefix = "";
if ( isUndefined(infix) || infix === null ) infix = "";
if ( isUndefined(postfix) || postfix === null ) postfix = "";
var r = this.join(postfix + infix + prefix);
return r.length ? prefix + r + postfix : r;
};
_AP.append = function() {
for ( var i = 0 ; i < arguments.length ; ++i )
if ( isDefined(arguments[i]) )
this[this.length] = arguments[i];
return this;
};
_AP.extend = function(x) {
if ( isDefined(x) ) {
if ( x.constructor == Array || typeof(x.foreach) != "function")
this.concat(x);
else {
var self = this;
x.foreach(function(v) { self.append(v); });
}
}
return this;
};
_AP.feed = function(x) {
x.extend(this);
return this;
};
_AP.foreach = function(f) {
for ( var i = 0 ; i < this.length ; ++i )
if ( f(this[i], i, this) === false )
return i;
};
_AP.coalesce = function(r, f) {
if ( isUndefined(f) ) {
f = arguments[0];
r = arguments[1];
}
if ( typeof(f) == "string" )
f = f.asMethod();
this.foreach(function(v, i, self) {
var t = f(r, v, i, self);
if ( isDefined(t) )
r = t;
});
return r;
};
_AP.fold = function(r, f) {
if ( isUndefined(f) ) {
f = arguments[0];
r = arguments[1];
}
if ( typeof(f) == "string" )
f = f.toFunction();
return this.coalesce(r, function(r, v) {
return isDefined(r) ? f(r, v) : v;
});
};
_AP.foldr = function(r, f) {
if ( isUndefined(f) ) {
f = arguments[0];
r = arguments[1];
}
if ( typeof(f) == "string" )
f = f.toFunction();
return this.fold(r, function(r, v) { return f(v, r); });
};
_AP.collect = function(a, f) {
if ( isUndefined(f) ) {
f = arguments[0];
a = arguments[1];
}
if ( typeof(f) == "string" )
f = f.asMethod();
if ( !a )
a = this.constructor ? new this.constructor : [];
return this.coalesce(a, function(r, v, i, self) { return r.append(f(v, i, self)); });
};
_AP.call = function(f) {
this.foreach(f);
return this;
};
_AP.asLongAs = function(a, f) {
if ( isUndefined(f) ) {
f = arguments[0];
a = arguments[1];
}
if ( f.constructor === RegExp )
f = Function.from(f, "test");
if ( !a )
a = this.constructor ? new this.constructor : [];
this.foreach(function(v, i, self) {
if ( !f(v, i, self) ) return false;
a.append(v);
});
return a;
};
_AP.filter = function(f, other) {
if ( f.constructor === RegExp )
f = Function.from(f, "test");
return this.collect(other ?
function(v, i, a) {
if ( f(v, i, a) )
return v;
other.append(v);
} :
function(v, i, a) {
if ( f(v, i, a) )
return v;
}
);
};
_AP.search = function(f) {
if ( f.constructor === RegExp )
f = Function.from(f, "test");
else if ( typeof(f) != "function" ) {
var x = f;
f = function(y) { return y === x; };
}
var result = this.foreach(function(v, i, self) { return !f(v, i, self); });
return typeof(result) == "number" ? result : -1;
};
_AP.indexOf = function(pattern) {
if ( typeof(pattern.foreach) != "function" )
pattern = [].append(pattern);
var result = this.foreach(function(v, i, self) {
return -1 !== pattern.foreach(function(v) {
return v === self[i++];
});
});
return typeof(result) == "number" ? result : -1;
};
_AP.inverse = function() {
return this.collect(function(v) {
return typeof(v.reverse) == "function" ? v.reverse() : v;
}).reverse();
}
_AP.flush = function(f) {
for ( ; this.length ; this.shift() )
if ( f(this[0], this) === false )
break;
return this;
};
_AP.split = function(f) {
if ( f.constructor === RegExp )
f = Function.from(f, "test");
else if ( typeof(f) != "function" )
f = "===".curry({ 0 : f });
var a = [[]], index = 0;
this.foreach(function(v) {
if ( f(v) )
a[++index] = [];
else
a[index].append(v);
});
return a;
};
_AP.zip = function() {
var result = this.collect(function(v) { return Function.args(v); });
Array.from(arguments).foreach(function(a) {
if ( !a || isUndefined(a.foreach) )
a = Function.args(v);
var last = 0;
if ( IsUndefined(a.foreach(function(v, i) {
var x = result.itemAt(i);
if ( isUndefined(x) ) return false;
x.push(v);
last = i;
})) ) {
if ( isDefined(result.length) )
result.length = last+1;
else
result = result.slice(0, last+1);
}
});
return result;
};
_AP.zipWith = function(f) {
if ( typeof(f) == "string" ) f = f.toFunction();
return this.zip.apply(this, Array.from(arguments).tail()).collect(function(v) {
return f.apply(null, v);
});
};
_AP.spread = function() {
return this.coalesce([], "extend");
};
Array.from = function(x) {
if ( typeof(x.toArray) == "function" ) {
x = x.toArray();
if ( this === Array )
return x;
}
if ( typeof(x.foreach) == "function" ) {
var a = new this;
x.foreach(function(v) { a.push(v); });
return a;
}
if ( typeof(x.length) == "number" ) {
var a = new this;
if ( typeof(x) != "string" )
for ( i = 0 ; i < x.length ; ++i )
a.push(x[i]);
else
for ( i = 0 ; i < x.length ; ++i )
a.push(x.charAt(i));
return a;
}
return (new this).push(x);
}
Array.fill = function(f, a) {
if ( !a ) a = new this;
var i = a.length;
if ( !i ) i = 0;
for ( ; ; ++i ) {
var v = f(i, a);
if ( isUndefined(v) )
return a;
a.append(v);
}
};
Array.recurse = function(f, resultIfEmpty) {
if ( isUndefined(resultIfEmpty) ) resultIfEmpty = new this;
return function(x) {
return this.isEmpty() ?
typeof(resultIfEmpty) == "function" ?
resultIfEmpty(x) : resultIfEmpty :
f(this.head(), this.tail(), x);
};
};
_AP.equals = Array.recurse(function(h, t, x) {
return typeof(x.head) == "function" && typeof(x.tail) == "function" &&
( typeof(h.equals) == "function" ? h.equals(x.head()) : h == x.head() ) &&
t.equals(x.tail());
},
Function.from(null, "isEmpty")
);
function iteratable(x) {
var p = x.prototype;
if ( !p || p == Object )
p = x;
else {
if ( typeof(p.push) == "function" ) {
if ( !x.from )
x.from = Array.from;
}
if ( typeof(p.append) == "function" ) {
if ( !x.fill )
x.fill = Array.fill;
}
}
if ( typeof(p.foreach) == "function" ) {
if ( !p.coalesce )
p.coalesce = _AP.coalesce;
if ( !p.fold )
p.fold = _AP.fold;
if ( !p.foldr )
p.foldr = _AP.foldr;
if ( !p.collect )
p.collect = _AP.collect;
if ( !p.filter )
p.filter = _AP.filter;
if ( !p.search )
p.search = _AP.search;
}
}
var _NP = Number.prototype;
_NP.times = function(f, counter) {
counter = counter || 0;
for ( var i = 0 ; i < this ; ++i )
f(counter++);
};
_NP.upTo = function(target, step) {
if ( !step ) step = 1;
var a = [];
for ( var i = this.valueOf() ; i <= target ; i += step )
a.append(i);
return a;
};
_NP.downTo = function(target, step) {
if ( !step ) step = 1;
var a = [];
for ( var i = this.valueOf() ; i >= target ; i -= step )
a.append(i);
return a;
};
_NP.to = function(target, step) {
return this < target ? this.upTo(target, step) : this.downTo(target, step);
};
_NP.chr = function() {
return String.fromCharCode(this);
};
var _SP = String.prototype;
_SP.itemAt = String.charAt;
_SP.asc = function() {
return this.charCodeAt(0);
};
_SP.to = function(target, step) {
return this.asc().to(target.toString().asc(), step).collect(Function.from(null, "chr"));
};
_SP.toFunctionUnary = function() {
eval("function __unary__(op) { return " + this + " op; }");
__unary__.op = this;
return __unary__;
};
_SP.toFunctionBinary = function() {
eval("function __binary__(op1, op2) { return op1 " + this + " op2; }");
__binary__.op = this;
return __binary__;
};
_SP.toFunction = function() {
return ",!,~,++,--,new,delete,typeof,void,".indexOf("," + this + ",") > -1 ?
this.toFunctionUnary() : this.toFunctionBinary();
};
_SP.toMethod = function() {
return this.toFunction().toMethod();
};
_SP.apply = function(obj, args) {
return this.toFunction().apply(obj, args);
};
_SP.call = function(obj) {
return this.toFunction().apply(obj, Array.from(arguments).slice(1));
};
_SP.curry = function() {
return _FP.curry.apply(this.toFunction(), arguments);
};
_SP.using = function() {
return _FP.using.apply(this.toFunction(), arguments);
};
_SP.asArgName = function() {
var self = this;
return { toString : function() { return self; }, asArgNameFlag : true };
};
_SP.asMethod = function(args) {
eval("function __method__(obj) { return obj." + this + ".apply(obj, Array.from(arguments).slice(1)); }");
return args ? __method__.withArgString("obj," + args) : __method__;
};
_SP.asMethodUsing = function() {
var args = (1).upTo(arguments.length).collect(function(v) { return "arg" + v; }).join(",");
return _FP.curry.apply(this.asMethod(args), (new Array(1)).concat(Array.from(arguments)));
};
var _RP = RegExp.prototype;
_RP.iterate = function(s, f) {
var sre = this.toString(), sep = sre.lastIndexOf("/");
var re = new RegExp(sre.slice(1, sep), sre.slice(sep+1) + "g");
for ( var a ; a = re.exec(s) ; )
if ( f(a, s, re) === false )
return a;
return null;
};
_RP.not = function() {
return "!".using(Function.from(this, "test", "str"));
};
if ( typeof(Enumerator) == "function" && typeof(Enumerator.prototype) == "object" ) {
var _EP = Enumerator.prototype;
_EP.foreach = function(f) {
for ( this.moveFirst() ; !this.atEnd() ; this.moveNext() )
if ( f(this.item(), this) === false )
return this.item();
return null;
};
iteratable(Enumerator);
}
function select(prep, map, selector) {
if ( arguments.length < 3 ) {
selector = map;
map = prep;
prep = arguments[2];
}
var r;
switch ( typeof(prep) ) {
case "string":
prep = prep.toFunction();
case "function":
case "object":
if ( prep ) {
r = map[prep(selector, this)];
break;
}
default:
r = map[selector];
}
if ( isUndefined(r) ) r = map[r];
return typeof(r) == "function" ? r.call(this, selector, prep) : r;
}
function stringSet(initItems) {
if ( initItems )
this.addToSet.apply(this, initItems);
}
stringSet.prototype = {
addToSet : function(item) {
var set = this;
return Array.from(arguments).coalesce(false, function(r, item) {
return !set[item] ? set[item] = true : r;
});
},
foreach : function(f) {
for ( var item in this )
if ( this[item] !== stringSet.prototype[item] && f(item, this) === false )
return item;
return null;
},
mergeSet : function(set) {
return Array.from(this).concat(Array.from(set)).toStringSet();
},
include : function() {
return "&&".using(
Function.from(this, null),
Function.from(stringSet.prototype, null).andThen(isUndefined));
},
exclude : function() {
return this.include().andThen("!".toFunctionUnary());
}
};
_AP.toStringSet = function() {
return new stringSet(this);
};
Math.even = function(n) {
return n % 2 == 0;
};
Math.odd = function(n) {
return n % 2 != 0;
};
var ff = isUndefined(ff) ? Function.from : ff;
function debugAlert(x) {
if ( debugAlert.on )
Array.from(arguments).foreach(alert);
if ( arguments.length > 0 )
return x;
}
debugAlert.on = true;

View file

@ -0,0 +1,93 @@
/*
beyondBrowser.js
by Dan Shappir and Sjoerd Visscher
For more information see http://w3future.com/html/beyondJS
*/
if ( typeof(beyondVer) !== "number" || beyondVer < 0.98 )
alert("beyondBrowser requires beyond JS library ver 0.98 or higher");
var beyondBrowserVer = 0.96;
Array.as = function(c) {
if ( typeof(c.length) != "number" )
return [].append(c);
if ( !c.foreach )
c.foreach = _AP.foreach;
if ( !c.head )
c.head = _AP.head;
if ( !c.tail )
c.tail = _AP.tail;
if ( !c.itemAt )
c.tail = _AP.itemAt;
if ( !c.isEmpty )
c.isEmpty = _AP.isEmpty;
if ( !c.slice )
c.slice = function(start, end) {
var a = Array.from(this);
return isDefined(end) ? a.slice(start, end) : a.slice(start);
};
iteratable(c);
return c;
};
_SP.element = function() {
return document.getElementById(this);
};
_SP.children = function(index, subIndex) {
var e = this.element();
var c = isDefined(subIndex) ? e.children(index, subIndex) :
isDefined(index) ? e.children(index) : e.children;
return isDefined(c.length) ? Array.as(c) : c;
};
_SP.foreach = function(f, r) {
return ( isUndefined(document.all) ? [ this.element() ] : Array.as(document.all[this]) ).foreach(f, r);
};
iteratable("");
_SP.setTo = function(html) {
this.foreach(function(e) { e.innerHTML = html; });
return this;
};
_SP.into = function(id) {
id.toString().setTo(this);
return this;
};
_SP.add = function(where, html) {
this.foreach(function(e) { e.insertAdjacentHTML(where, html); });
return this;
};
_SP.insert = _SP.add.curry("afterBegin");
_SP.append = _SP.add.curry("beforeEnd");
_SP.showHide = function(show) {
var display = show ? "" : "none";
this.foreach(function(e) { e.style.display = display; });
return this;
};
_SP.show = _SP.showHide.curry(true);
_SP.hide = _SP.showHide.curry(false);
_SP.toggleShow = function() {
this.foreach(function(e) { e.style.display = e.style.display.length ? "" : "none"; });
return this;
};
_SP.moveTo = function(clientX, clientY) {
this.foreach(function(e) { e.style.left = clientX + "px"; e.style.top = clientY + "px"; });
return this;
};
_SP.w = function(text) {
return "<" + this + ">" + text + "</" + this.replace(_SP.w.re, "") + ">";
};
_SP.w.re = /\s.*/;
_SP.tag = function(tag) {
return tag.toString().w(this);
};
_SP.write = function(s) {
return document.write(this.w(s));
};
if ( isUndefined(document.getElementById) )
document.getElementById = function(id) {
return Array.as(document.all[id])[0];
};
if ( isUndefined(document.getElementsByTagName) )
document.getElementsByTagName = function(tagName) {
tagName = tagName.toUpperCase()
return Array.as(document.all).filter(function(e) { return e.tagName.toUpperCase() == tagName; });
};

View file

@ -0,0 +1,126 @@
/*
BeyondDispatch.JS
by Dan Shappir and Sjoerd Visscher
For more information see http://w3future.com/html/beyondJS
*/
if ( typeof(beyondVer) !== "number" || beyondVer < 0.98 )
alert("beyondDispatch requires Beyond JS library ver 0.98 or higher");
var beyondDispatchVer = 0.90;
function dispatch(otherwise, argString) {
var options = [];
var isFunction = typeof(otherwise) == "function";
if ( isFunction && isUndefined(argString) )
argString = Function.argString(otherwise);
var dispatcher = function() {
var self = this, args = arguments, result;
return 0 <= options.foreach(function(v) {
if ( v != null && v[0].apply(self, args) ) {
result = typeof(v[1]) == "function" ? v[1].apply(self, args) : v[1];
return false;
}
}) ? result : isFunction ? otherwise.apply(self, args) : otherwise;
};
dispatcher.add = function(test, target) {
if ( isUndefined(target) ) {
target = test;
test = target.length;
}
if ( typeof(test) == "string" ) {
var args = typeof(target) == "function" ? Function.argString(target) : "";
eval("test = function(" + args + ") { return " + test + "; }");
return options.push([test, target]) - 1;
}
test = dispatch.fix(test);
return isUndefined(test) ? -1 : options.push([test, target]) - 1;
};
dispatcher.addList = function() {
var result = -1;
for ( var i = 0 ; i < arguments.length ; i += 2 )
result = this.add(arguments[i], arguments[i + 1]);
return result;
};
dispatcher.remove = function(index) {
options[index] = null;
};
dispatcher.removeAll = function() {
options = [];
};
dispatcher.clone = function() {
var clone = dispatch(otherwise, argString);
options.foreach(function(e) {
if ( e != null )
clone.add(e[0], e[1]);
});
return clone;
};
return dispatcher.withArgString(argString);
}
dispatch.fix = select.curry({
prep: "typeof",
map : {
number : function(x) {
if ( x >= 0 )
return function() { return arguments.length == x; };
x = -x;
return function() { return arguments.length >= x; }
},
object : function(x) {
if ( typeof(x.foreach) == "function" )
return function() {
var args = arguments;
return -1 == x.foreach(function(v, i) {
var t = typeof(v);
if ( isUndefined(t) )
return true;
if ( i >= args.length )
return false;
var a = args[i];
if ( t == "function" )
return v.call(this, a);
if ( v instanceof RegExp )
return v.test(a);
if ( t == "string" ) {
t = v.slice(1);
switch ( v.charAt(0) ) {
case ".":
switch ( typeof(a) ) {
case "number":
a = new Number(a);
break;
case "boolean":
a = new Boolean(a);
break;
case "string":
a = new String(a);
}
var i = t.indexOf(":");
if ( i > -1 ) {
v = t.slice(i + 1);
t = t.slice(0, i);
}
if ( !(t in a) )
return false;
if ( i == -1 )
return true;
a = a[t];
t = v;
case ":":
return typeof(a) == t ||
( t != "function" && eval("typeof(" + t + ") == 'function' && a instanceof " + t) );
case "^":
return eval(t + ".isPrototypeOf(a)");
case "?":
return eval("a " + t);
case "\\":
v = t;
}
}
return typeof(v.equals) == "function" ? v.equals(a) : v == a;
});
};
},
"function" : function(x) { return x; }
}
});

View file

@ -0,0 +1,367 @@
/*
BeyondLazy.JS
by Dan Shappir and Sjoerd Visscher
For more information see http://w3future.com/html/beyondJS
*/
if ( typeof(beyondVer) !== "number" || beyondVer < 0.98 )
alert("beyondLazy requires Beyond JS library ver 0.98 or higher");
var beyondLazyVer = 0.96;
function Lazy(generator, length) {
this.generator = generator ? generator : function() { return []; };
this.length = length;
}
var _LP = Lazy.prototype;
Lazy.cache = true;
_LP.foreach = function(f) {
var item = [];
for ( var i = 0 ; (item = this.generator(item)).length ; ++i )
if ( f(item[0], i, this) === false )
return i;
};
_LP.collect = function(f) {
var self = this;
if ( typeof(f) == "string" )
f = f.asMethod();
return function(prev) {
var current = self.generator(isDefined(prev[1]) ? prev[1] : []);
return current.length ? [ f(current[0], prev[0]), current ] : current;
}.lazy(this.length);
};
_LP.call = function(f) {
var self = this;
return function(prev) {
var current = self.generator(isDefined(prev[1]) ? prev[1] : []);
f(current[0]);
return current;
}.lazy();
};
_LP.asLongAs = function(f) {
var self = this;
if ( f.constructor === RegExp )
f = Function.from(f, "test");
return function(prev) {
var current = self.generator(prev);
return current.length && f(current[0]) ? current : [];
}.lazy();
};
_LP.filter = function(f, other) {
var self = this;
if ( f.constructor === RegExp )
f = Function.from(f, "test");
return function(item) {
while ( (item = self.generator(item)).length && !f(item[0]) )
if ( other )
other.append(item[0]);
return item;
}.lazy();
};
_LP.zip = function() {
var collections = [ this ].concat(Array.from(arguments).collect(Function.from(null, "lazy")));
return function(prev) {
var length = Number.POSITIVE_INFINITY;
collections.foreach(function(c) {
if ( c.length && c.length < length )
length = c.length;
});
var current_length = prev[1] ? prev[1] : 0;
if ( current_length >= length )
return [];
if ( !prev.length )
(collections.length + 2).times(function() { prev.push([]); });
var current = [ [] , current_length >= 0 ? current_length + 1 : -1 ];
collections.foreach(function(c, i) {
var x = c.generator(prev[i + 2]);
if ( !x.length ) {
current = [];
return false;
}
current[0].push(x[0]);
current.push(x);
});
return current;
}.lazy();
};
_LP.zipWith = _AP.zipWith;
_LP.spread = function() {
var self = this;
return function(prev) {
var current;
for (;;) {
switch ( prev.length ) {
case 0:
current = self.generator([]);
break;
case 2:
current = self.generator(prev[1]);
break;
default:
current = prev[2].generator(prev[3]);
if ( current.length )
return [ current[0], prev[1], prev[2], current ];
prev = [, prev[1]];
continue;
}
if ( !current.length )
return [];
var value = current[0];
if ( !value || typeof(value.foreach) != "function" || typeof(value.lazy) != "function" )
return [ value, current ];
prev = [, current, value.lazy(), []];
}
}.lazy();
};
_LP.head = function(length) {
if ( !length ) return this.generator([])[0];
var self = this;
return function(prev) {
var index;
if ( prev.length ) {
index = prev[2] + 1;
prev = prev[1];
}
else
index = 1;
if ( index > length )
return [];
var current = self.generator(prev);
return current.length ? [ current[0], current, index ] : current;
}.lazy(length);
};
_LP.tail = function(start) {
if ( !start ) start = 1;
return this.collect(function(current, prev) { return [ current, isUndefined(prev) ? 0 : prev[1]+1 ]; }).
filter(function(current) { return current[1] >= start; }).
collect(function(current) { return current[0]; });
};
_LP.slice = function(start, end) {
var result = start ? this.tail(start) : this;
if ( isDefined(end) ) end -= start;
return end >= 1 ? result.head(end) : result;
};
_LP.isEmpty = function() {
return !this.generator([]).length;
};
_LP.itemAt = function(index) {
return this.tail(start).head();
};
_LP.empty = function() {
this.generator = function() {
return [];
};
this.length = 0;
return this;
};
_LP.toString = function() {
return Array.from(this).toString();
};
_LP.toLocaleString = function() {
return Array.from(this).toLocaleString();
};
_LP.valueOf = function() {
return Array.from(this).valueOf();
};
_LP.join = function(separator) {
var a = Array.from(this);
return isDefined(separator) ? a.join(separator) : a.join();
};
_LP.reverse = function() {
return Array.from(this).reverse();
};
_LP.concat = function() {
var a = [ this ].concat(Array.from(arguments)).filter(isDefined).collect(function(v) {
return v != null && v.constructor == Array ? v.lazy() : v;
});
return a.length == 1 ? this : function(prev) {
if ( !prev.length ) prev = [ null, [], 0 ];
for (;;) {
var current = a[prev[2]];
if ( isUndefined(current) )
return [];
if ( current === null || typeof(current.constructor) !== "function" || current.constructor !== Lazy )
return [ current, [], prev[2]+1 ];
current = current.generator(prev[1]);
if ( current.length )
return [ current[0], current, prev[2] ];
prev = [ null, [], prev[2]+1 ];
}
}.lazy();
};
_LP.push = _LP.append = function() {
var a = Array.from(arguments).filter(isDefined);
if ( a.length ) {
var generator = this.generator;
this.generator = function(prev) {
if ( isUndefined(prev[2]) ) {
var current = generator(prev);
if ( current.length )
return current;
prev = [ null, null, 0 ];
}
return isDefined(a[prev[2]]) ? [ a[prev[2]], null, prev[2]+1 ] : [];
};
}
return this;
};
_LP.shift = function() {
var result = this.head();
if ( isDefined(result) )
this.generator = this.generator.lazy().tail().generator;
return result;
};
_LP.unshift = function() {
var a = Array.from(arguments).filter(isDefined);
if ( a.length )
this.generator = a.lazy().concat(this.generator.lazy()).generator;
return this;
};
_LP.splice = function(start, deleteCount) {
var a = Array.from(arguments).slice(2).filter(isDefined);
var clone = this.generator.lazy();
var result = clone.tail(start+deleteCount);
if ( a.length ) result = a.lazy().concat(result);
if ( start > 0 ) result = clone.head(start).concat(result);
this.generator = result.generator;
return clone.slice(start, start+deleteCount);
};
_LP.extend = function(x) {
if ( isDefined(x) ) {
if ( typeof(x) == "function" )
x = x().lazy();
var generator = this.generator;
this.generator = function(prev) {
if ( !prev.length )
prev = [ null, prev, generator ];
var current = prev[2](prev[1]);
if ( current.length )
return [ current[0], current, prev[2] ];
if ( prev[2] === x.generator )
return current;
current = x.generator(current);
return current.length ? [ current[0], current, x.generator ] : current;
};
this.length = this.u;
if ( Lazy.cache )
return this.cached();
}
return this;
};
_LP.feed = _AP.feed;
_LP.cached = function() {
if ( isUndefined(this.isCached) ) {
this.isCached = true;
var cache = [], generator = this.generator;
this.generator = function(prev) {
if ( !prev.length )
prev = [ null, [], 0 ];
if ( prev[2] < cache.length )
return cache[prev[2]];
var current = generator(prev[1]);
return cache[cache.length] = current.length ? [ current[0], current, cache.length+1 ] : [];
}
}
return this;
};
_LP.equals = _AP.equals;
_LP.lazy = Function.This;
_FP.lazy = function(length) {
return new Lazy(this, length);
};
_NP.lazyUp = function(end, step) {
var undef, start = this.valueOf();
if ( !(step > 0) ) step = 1;
var length;
if ( isUndefined(end) || end === null )
end = Number.POSITIVE_INFINITY;
else
length = Math.floor((end - start + 1)/step);
return function(item) {
if ( !item.length ) return [ start, undef ];
var value = item[0] + step;
return value <= end ? [ value, undef ] : [];
}.lazy(length);
};
_NP.lazyDown = function(end, step) {
var undef, start = this.valueOf();
if ( !(step > 0) ) step = 1;
var length;
if ( isUndefined(end) || end === null )
end = Number.NEGATIVE_INFINITY;
else
length = Math.floor((start - end + 1)/step);
return function(item) {
if ( !item.length ) return [ start, undef ];
var value = item[0] - step;
return value >= end ? [ value, undef ] : [];
}.lazy(length);
};
_NP.lazy = function(end, step) {
if ( isUndefined(end) || end === null ) end = Number.POSITIVE_INFINITY;
return this < end ? this.lazyUp(end, step) : this.lazyDown(end, step);
};
_SP.lazy = function(target, step) {
return this.asc().lazy(target.toString().asc(), step).collect(Function.from(null, "chr"));
};
_AP.lazy = function() {
return (0).lazy(this.length-1).collect(Function.from(this, "itemAt"));
};
_AP.cycle = function(n) {
var self = this;
var cycle = isUndefined(n) ?
function() {
return self.lazy().extend(cycle);
} :
1 <= n ?
function(n) {
return 1 == n ? self.lazy() : self.lazy().extend(cycle.curry(n-1));
} :
function() {
return [].lazy();
};
return cycle(n);
};
if ( _EP )
_EP.lazy = function() {
var self = this;
return function(item) {
if ( !item.length )
self.moveFirst();
else
self.moveNext();
return !self.atEnd() ? [ self.item(), self ] : [];
}.lazy();
};
Lazy.from = function(x) {
if ( typeof(x.lazy) == "function" )
return x.lazy();
if ( typeof(x.length) == "number" )
return this.fill(function(index, item) { return x[index]; });
return Array.from(x).lazy();
};
Lazy.fill = function(f) {
var lazy = function(item) {
var index = item[1] ? item[1] : 0;
var value = f(index, item[0]);
return isDefined(value) ? [ value, index+1 ] : [];
}.lazy();
return this.cache ? lazy.cached() : lazy;
};
Lazy.recurse = Array.recurse;
iteratable(Lazy);

View file

@ -0,0 +1,467 @@
/*
BeyondRhino.JS
by Dan Shappir and Sjoerd Visscher
For more information see http://w3future.com/html/beyondJS
*/
load("beyond.js", "beyondLazy.js", "beyondStreams.js");
var beyondRhinoVer = 0.9;
var Beyond = {};
importPackage(java.util);
function javaArray(type, length) {
var alloc = new Function("length", "return java.lang.reflect.Array.newInstance(" + type + ", length)");
return alloc(length);
}
function alert(s) {
print(s);
}
function toJava(v, asObjects) {
return typeof(v.toJava) == "function" ? v.toJava(asObjects) : v;
}
_NP.toJava = function() {
var v = this.valueOf();
return v == Math.round(v) ? new java.lang.Integer(v) : new java.lang.Double(v);
};
_SP.toJava = function() {
var v = this.valueOf();
return new java.lang.String(v);
};
Boolean.prototype.toJava = function() {
var v = this.valueOf();
return new java.lang.Boolean(v);
};
function javaType(v) {
switch ( typeof(v) ) {
case "boolean":
return java.lang.Boolean.TYPE;
case "number":
return v == Math.round(v) ? java.lang.Integer.TYPE : java.lang.Double.TYPE;
case "string":
return java.lang.String;
}
return java.lang.Object;
}
_AP.toJava = function(asObjects) {
if ( !this.length )
return java.lang.reflect.Array.newInstance(java.lang.Object, 0);
var r;
if ( asObjects )
r = java.lang.reflect.Array.newInstance(java.lang.Object, this.length);
else {
var type = this.coalesce(function(type, v) {
var jt = javaType(v);
return isUndefined(type) || type == jt || ( type == java.lang.Integer.TYPE && jt == java.lang.Double.TYPE ) ?
jt : java.lang.Object;
});
r = java.lang.reflect.Array.newInstance(type, this.length);
}
this.foreach(function(v, i) {
r[i] = toJava(v);
});
return r;
};
function toJavaScript(v) {
if ( typeof(v) != "object" || v == null )
return v;
if ( typeof(v.getClass) != "function" ) {
if ( typeof(v.valueOf) == "function" )
v = v.valueOf();
return "undefined" != v ? v : undefined;
}
var cls = v.getClass();
switch ( String(cls.getName()) ) {
case "java.lang.Boolean":
return "true" == v;
case "java.lang.Byte":
case "java.lang.Short":
case "java.lang.Integer":
case "java.lang.Long":
return parseInt(v);
case "java.lang.Float":
case "java.lang.Double":
return parseFloat(v);
default:
var s = String(v);
return s == v ? s : v;
}
}
// Hack to enable use of Array.from() on Java arrays
_AP.toArray = null;
_AP.enumeration = function() {
var self = this;
var index = 0;
return new Enumeration() {
hasMoreElements : function() {
return index < self.length;
},
nextElement : function() {
return toJava(self[index++]);
}
};
};
_LP.enumeration = function() {
var self = this;
var item = [];
var cached = false;
var reuse = false;
return new Enumeration() {
hasMoreElements : function() {
if ( !cached ) {
cached = reuse = true;
item = self.generator(item);
}
return item.length;
},
nextElement : function() {
if ( !reuse )
item = self.generator(item);
cached = !cached;
reuse = false;
return toJava(item[0]);
}
};
};
Beyond.enumeration = function(e) {
return function() {
return e.hasMoreElements() ? [ toJavaScript(e.nextElement()) ] : [];
}.lazy();
};
_AP.iterator = function() {
var self = this;
var index = 0;
return new Iterator() {
hasNext : function() {
return index < self.length;
},
next : function() {
return toJava(self[index++]);
},
remove : function() {
if ( index > 0 )
self.splice(--index, 1);
}
};
};
_LP.iterator = function() {
var self = this;
var item = [];
var cached = false;
var reuse = false;
return new Iterator() {
hasNext : function() {
if ( !cached ) {
cached = reuse = true;
item = self.generator(item);
}
return item.length;
},
next : function() {
if ( !reuse )
item = self.generator(item);
cached = !cached;
reuse = false;
return toJava(item[0]);
}
};
};
Beyond.iterator = function(i) {
return function() {
return i.hasNext() ? [ toJavaScript(i.next()) ] : [];
}.lazy();
};
function isTypeOrDerived(v, type) {
for ( var c = v.getClass() ; c ; c = c.getSuperclass() )
if ( c.getName() == type )
return true;
return false;
}
_AP.listIterator = function() {
var self = this;
var index, prev;
return new ListIterator() {
hasNext : function() {
return isDefined(index) ? index < self.length : self.length > 0;
},
hasPrevious : function() {
return isDefined(index) ? index > 0 : self.length > 0;
},
next : function() {
if ( isUndefined(index) )
index = 0;
prev = index;
return toJava(self[index++]);
},
previous : function() {
if ( isUndefined(index) )
index = self.length;
prev = index;
return toJava(self[--index]);
},
nextIndex : function() {
return isUndefined(index) ? 0 : index;
},
previousIndex : function() {
return ( isUndefiend(index) ? self.length : index ) - 1;
},
set : function(element) {
if ( isDefined(prev) )
self.splice(prev, 1, toJavaScript(element));
},
add : function(element) {
prev = index;
self.splice(index++, 0, toJavaScript(element));
},
remove : function() {
if ( isDefined(prev) )
self.splice(prev, 1);
}
};
};
Beyond.reverseIterator = function(i) {
return function() {
return i.hasPrevious() ? [ toJavaScript(i.previous()) ] : [];
}.lazy();
};
_AP.collection = function() {
var self = this;
return new Collection() {
add : function(element) {
element = toJavaScript(element)
if ( self.search(element) != -1 )
return false;
self.push(element);
return true;
},
addAll : function(collection) {
var me = this;
return Beyond.iterator(collection.iterator()).coalesce(function(r, v) {
return me.add(v) || r;
});
},
clear : function() {
self.length = 0;
},
contains : function(element) {
return self.search(toJavaScript(element)) != -1;
},
containsAll : function(collection) {
var me = this;
return Beyond.iterator(collection.iterator()).coalesce(true, function(r, v) {
return r && me.contains(v);
});
},
isEmpty : function() {
return self.length == 0;
},
iterator : function() {
return self.iterator();
},
remove : function(element) {
element = toJavaScript(element)
var index = self.search(element);
if ( index == -1 )
return false;
self.splice(index, 1);
return true;
},
removeAll : function(collection) {
var me = this;
return Beyond.iterator(collection.iterator()).coalesce(function(r, v) { return me.remove(v) || r; });
},
retainAll : function(collection) {
var length = self.length;
var filtered = self.filter(function(v) { return collection.contains(toJava(v)); });
filtered.foreach(function(v, i) { self[i] = v; });
self.length = filtered.length;
return self.length != length;
},
size : function() {
return self.length;
},
toArray : function(a) {
if ( typeof(a) != "object" )
return self.toJava(true);
var type = a.getClass().getName().slice(2, -1);
var java = self.collect(function(v) { return toJava(v, true); }).
filter(function(v) { return isTypeOrDerived(v, type); });
if ( a.length < java.length )
a = javaArray(type, java.length);
java.foreach(function(v, i) { a[i] = v; });
if ( a.length > java.length )
Arrays.fill(a, java.length, a.length, null);
return a;
}
};
};
Beyond.collection = function(collection) {
return function(item) {
var i = item.length ? item[1] : collection.iterator();
return i.hasNext() ? [ toJavaScript(i.next()) , i ] : [];
}.lazy();
};
_AP.list = function() {
var self = this;
return new List() {
add : function(index, element) {
if ( isUndefined(element) )
self.push(toJavaScript(index));
else
self.splice(index, 0, toJavaScript(element));
return true;
},
addAll : function(index, collection) {
var me = this;
if ( isUndefined(collection) )
Beyond.iterator(index.iterator()).foreach(function(v) { me.add(v); });
else
Beyond.iterator(collection.iterator()).foreach(function(v) { me.add(index++, v); });
return true;
},
clear : function() {
self.length = 0;
},
contains : function(element) {
return self.search(toJavaScript(element)) != -1;
},
containsAll : function(collection) {
var me = this;
return Beyond.iterator(collection.iterator()).coalesce(true, function(r, v) { return r && me.contains(v); });
},
get : function(index) {
var x = toJava(self[index]);
return toJava(self[index]);
},
indexOf : function(element) {
return self.search(toJavaScript(element));
},
lastIndexOf : function(element) {
var index = self.reverese().search(toJavaScript(element));
return index == -1 ? -1 : self.length - index - 1;
},
isEmpty : function() {
return self.length == 0;
},
iterator : function() {
return self.iterator();
},
listIterator : function(index) {
var i = self.listIterator();
if ( isDefined(index) )
for ( ; i.hasNext() && i.nextIndex() < index ; i.next() );
return i;
},
remove : function(index) {
return self.splice(index, 1)[0];
},
removeAll : function(collection) {
var me = this;
return Beyond.iterator(collection.iterator()).coalesce(function(r, v) { return me.remove(v) || r; });
},
retainAll : function(collection) {
var length = self.length;
var filtered = self.filter(function(v) { return collection.contains(toJava(v)); });
filtered.foreach(function(v, i) { self[i] = v; });
self.length = filtered.length;
return self.length != length;
},
set : function(index, element) {
var prev = self[index];
self[index] = toJavaScript(element);
return toJava(prev, true);
},
size : function() {
return self.length;
},
subList : function(from, to) {
return self.slice(from, to).list();
},
toArray : function(a) {
if ( typeof(a) != "object" )
return self.toJava(true);
var type = a.getClass().getName().slice(2, -1);
var java = self.collect(function(v) { return toJava(v, true); }).
filter(function(v) { return isTypeOrDerived(v, type); });
if ( a.length < java.length )
a = javaArray(type, java.length);
java.foreach(function(v, i) { a[i] = v; });
if ( a.length > java.length )
Arrays.fill(a, java.length, a.length, null);
return a;
}
};
};
Beyond.list = Beyond.collection;
Beyond.reverseList = function(list) {
return function(item) {
var i = item.length ? item[1] : list.listIterator();
return i.hasPrevious() ? [ toJavaScript(i.previous()) , i ] : [];
}.lazy();
};
var File = {
read : function(file, cache, reader) {
var rf = function(item) {
var stream = item[1] ?
item[1] :
new java.io.BufferedReader(reader ? file : new java.io.FileReader(file));
var line = stream.readLine();
return line != null ? [ String(line), stream ] : [];
}.lazy();
if ( cache )
rf = rf.cached();
rf.file = file;
return rf;
},
write : function(file, overwrite, writer) {
var st = new Stream(writer ? file : new java.io.FileWriter(file, !overwrite));
var handle = st.foreach(function(v, stream) {
if ( isDefined(v) ) {
var s = String(v) + "\n";
stream.write(s, 0, s.length);
stream.flush();
}
});
st.stop = function() {
this.detach(handle);
this.owner.Close();
};
return st;
},
process : function(input, output, f, append) {
if ( !f )
f = Function.NOP;
else if ( f.constructor === RegExp ) {
var re = f;
f = function(line) {
if ( re.test(line) )
return line;
};
}
return this.write(output, !append).extend(this.read(input).collect(f));
}
};
File.StdIn = File.read(new java.io.InputStreamReader(java.lang.System["in"]), false, true);
File.StdOut = File.write(new java.io.OutputStreamWriter(java.lang.System["out"]), false, true);
File.StdErr = File.write(new java.io.OutputStreamWriter(java.lang.System["err"]), false, true);

View file

@ -0,0 +1,221 @@
/*
BeyondStreams.JS
by Dan Shappir and Sjoerd Visscher
For more information see http://w3future.com/html/beyondJS
*/
if ( typeof(beyondVer) !== "number" || beyondVer < 0.99 )
alert("beyondStreams requires Beyond JS library ver 0.99 or higher");
var beyondStreamsVer = 0.91;
function Stream(owner) {
this.owner = owner;
this.callbacks = [];
}
var _STP = Stream.prototype;
_STP.foreach = function(f) {
return this.callbacks.push(f)-1;
};
_STP.detach = function(x) {
if ( isUndefined(x) || x === null ) {
if ( this.callbacks.isEmpty() )
return false;
this.callbacks.empty();
return true;
}
if ( typeof(x) == "function" || typeof(x) == "object" )
x = this.callbacks.search("===".curry(x));
if ( x < 0 || x >= this.callbacks.length )
return false;
if ( x == this.callbacks.length-1 )
this.callbacks.pop();
else {
this.callbacks[x] = this.undef;
if ( this.callbacks.filter(isDefined).isEmpty() )
this.callbacks.empty();
}
return true;
};
_STP.push = function() {
var self = this;
Array.from(arguments).foreach(function(v) {
self.callbacks.foreach(function(f, i) {
if ( typeof(f) == "function" ) {
if ( f.call(self.owner, v, self.owner, self) === false )
self.detach(i);
}
else if ( f(v, self.owner, self) === false )
self.detach(i);
});
});
return this;
};
_STP.append = function() {
this.push.apply(this, Array.from(arguments).filter(isDefined));
return this;
};
_STP.extend = function(x) {
if ( !x || isUndefined(x.foreach) )
this.push(x);
else {
var self = this;
x.foreach(function(v) { self.push(v); });
}
return this;
};
_STP.feed = function(st) {
return this.foreach(function(v) { st.push(v); });
};
_STP.pre = function(f) {
var st = new Stream(this.owner);
st.collect(f).feed(this);
return st;
};
_STP.fold = function(r, f) {
if ( isUndefined(f) ) {
f = arguments[0];
r = arguments[1];
}
if ( typeof(f) == "string" )
f = f.toFunction();
var st = new Stream(this.owner);
this.coalesce(function(x, v) {
r = isDefined(r) ? f(r, v) : v;
st.push(r);
});
return st;
};
_STP.search = function(f) {
if ( f.constructor === RegExp )
f = Function.from(f, "test");
else if ( typeof(f) != "function" ) {
var x = f;
f = function(y) { return y === x; };
}
var st = new Stream(this.owner);
this.foreach(function(v, owner, self) {
if ( f(v, owner, self) ) {
st.push(v);
return false;
}
});
return st;
};
_STP.delay = function(iMilliSeconds) {
var st = new Stream(this.owner);
this.foreach(function(v) { Function.from(st, "push").delay(iMilliSeconds)(v); });
return st;
};
_STP.buffer = function(f) {
if ( typeof(f) == "number" ) {
var count = f;
f = function(v, buffer) {
return buffer.length+1 >= count;
};
}
else if ( f.constructor === RegExp )
f = Function.from(f, "test");
var st = new Stream(this.owner), buffer = [];
this.foreach(function(v, owner, self) {
var r = f.call(owner, v, buffer, owner, self);
buffer.push(v);
if ( r ) {
st.push(buffer);
buffer = [];
}
});
return st;
};
Stream.from = function(x) {
if ( typeof(x.foreach) != "function" )
x = Array.from(x);
var st = new this(x);
x.foreach(function(v) { st.push(v); });
return st;
};
iteratable(Stream);
function timerStream(iMilliSeconds) {
var st = new Stream(iMilliSeconds);
var nInterval = setInterval(function() { st.push(new Date); }, iMilliSeconds);
st.stop = function() {
if ( isDefined(nInterval) ) {
clearInterval(nInterval);
nInterval = timerStream.undef;
}
};
return st;
}
function eventStream(element, eventName, noAttach) {
if ( typeof(element) == "string" )
element = element.element();
var st = new Stream(element);
if ( !noAttach && isDefined(element.attachEvent) ) {
function pushEvent(event) {
st.push(event ? event : window.event);
}
if ( element.attachEvent(eventName, pushEvent) ) {
st.stop = function() {
element.detachEvent(eventName, pushEvent);
};
return st;
}
}
function pushEvent(event) {
if ( !st.stopped )
st.push(event ? event : window.event);
}
var prev = eval("element." + eventName);
if ( !prev )
eval("element." + eventName + " = pushEvent");
else
eval("element." + eventName + " = prev.andThen(pushEvent)");
st.stop = function() {
st.stopped = true;
};
return st;
}
function propertyStream(element, propertyName) {
if ( typeof(element) == "string" )
element = element.element();
var st = new Stream(element);
var handle = st.foreach(Function.set(element, propertyName));
st.stop = function() {
this.detach(handle);
this.stop = Function.NOP;
};
return st;
}
function cancelDefaultAction(e) {
e.returnValue = false;
return e;
}
function cancelBubble(e) {
e.cancelBubble = true;
return e;
}
function logStream(log) {
var st = new Stream(log ? log : []);
var handle = st.foreach(function(v) {
st.owner.push([ new Date, v ]);
});
st.stop = function() {
st.detach(handle);
this.stop = Function.NOP;
};
st.flush = function() {
st.owner.empty();
};
return st;
}

View file

@ -0,0 +1,85 @@
/*
beyondXML.js
by Dan Shappir and Sjoerd Visscher
For more information see http://w3future.com/html/beyondJS
*/
if ( typeof(beyondVer) !== "number" || beyondVer < 0.95 )
alert("BeyondXML requires Beyond JS library ver 0.95 or higher");
var beyondXMLVer = 0.96;
var beyondXML = new function() {
var isMozilla = typeof(Document) != "undefined";
this.createDocument = function() {
return isMozilla ? new Document() : new ActiveXObject("Microsoft.XMLDOM");
};
this.parseFromString = function(doc, xml) {
return isMozilla ? doc = (new DOMParser()).parseFromString(xml, "text/xml") : doc.loadXML(xml);
};
this.load = function(doc, url) {
return doc.load(url);
};
this.invoke = function(method) {
return Function.from(null, method);
};
var origToArray = Array.from;
Array.from = function(x) {
var n = x.nextNode;
return typeof(n) != "object" ? origToArray.call(this, x) :
!n ? [] : Array.fill(function() { if ( n = x.nextNode ) return n; }, [ n ]);
};
if ( typeof(beyondLazyVer) == "number" && beyondLazyVer >= 0.95 ) {
var origToLazy = Lazy.from;
Lazy.from = function(x) {
var n = x.nextNode;
if ( typeof(n) != "object" )
return origToLazy.call(this, x);
return function(prev) {
if ( !prev.length ) x.reset();
var n = x.nextNode;
return n ? [ n ] : [];
}.lazy();
};
this.lazyLists = false;
this.list = function(list) {
return beyondXML.lazyLists ? Lazy.from(list) : Array.from(list);
};
}
};
function childNodes(n) {
return beyondXML.list(n.childNodes);
}
_SP.parseFromString = function(doc) {
if ( !doc )
doc = beyondXML.createDocument();
return beyondXML.parseFromString(doc, this) ? doc : null;
};
_SP.load = function(document) {
if ( !doc )
doc = beyondXML.createDocument();
return beyondXML.load(doc, this) ? doc : null;
};
_SP.selectSingleNode = function(node) {
return node.selectSingleNode(this);
};
_SP.selectNodes = function(node) {
return beyondXML.list(node.selectNodes(this));
};
_SP.nodeFromID = function(node) {
return node.ownerDocument.nodeFromID(this);
};
_SP.getElementsByTagName = function(node) {
return beyondXML.list(node.getElementsByTagName(this));
};
_SP.node = function(node) {
return node.getElementsByTagName(this).nextNode();
};
_SP.value = function(node) {
return this.node(node).nodeTypedValue;
};

38
www/extras/js/json.js Normal file
View file

@ -0,0 +1,38 @@
Array.prototype.______array='______array';var JSON={org:'http://www.JSON.org',copyright:'(c)2005 JSON.org',license:'http://www.crockford.com/JSON/license.html',stringify:function(arg){var c,i,l,s='',v;switch(typeof arg){case'object':if(arg){if(arg.______array=='______array'){for(i=0;i<arg.length;++i){v=this.stringify(arg[i]);if(s){s+=',';}
s+=v;}
return'['+s+']';}else if(typeof arg.toString!='undefined'){for(i in arg){v=arg[i];if(typeof v!='undefined'&&typeof v!='function'){v=this.stringify(v);if(s){s+=',';}
s+=this.stringify(i)+':'+v;}}
return'{'+s+'}';}}
return'null';case'number':return isFinite(arg)?String(arg):'null';case'string':l=arg.length;s='"';for(i=0;i<l;i+=1){c=arg.charAt(i);if(c>=' '){if(c=='\\'||c=='"'){s+='\\';}
s+=c;}else{switch(c){case'\b':s+='\\b';break;case'\f':s+='\\f';break;case'\n':s+='\\n';break;case'\r':s+='\\r';break;case'\t':s+='\\t';break;default:c=c.charCodeAt();s+='\\u00'+Math.floor(c/16).toString(16)+
(c%16).toString(16);}}}
return s+'"';case'boolean':return String(arg);default:return'null';}},parse:function(text){var at=0;var ch=' ';function error(m){throw{name:'JSONError',message:m,at:at-1,text:text};}
function next(){ch=text.charAt(at);at+=1;return ch;}
function white(){while(ch!==''&&ch<=' '){next();}}
function str(){var i,s='',t,u;if(ch=='"'){outer:while(next()){if(ch=='"'){next();return s;}else if(ch=='\\'){switch(next()){case'b':s+='\b';break;case'f':s+='\f';break;case'n':s+='\n';break;case'r':s+='\r';break;case't':s+='\t';break;case'u':u=0;for(i=0;i<4;i+=1){t=parseInt(next(),16);if(!isFinite(t)){break outer;}
u=u*16+t;}
s+=String.fromCharCode(u);break;default:s+=ch;}}else{s+=ch;}}}
error("Bad string");}
function arr(){var a=[];if(ch=='['){next();white();if(ch==']'){next();return a;}
while(ch){a.push(val());white();if(ch==']'){next();return a;}else if(ch!=','){break;}
next();white();}}
error("Bad array");}
function obj(){var k,o={};if(ch=='{'){next();white();if(ch=='}'){next();return o;}
while(ch){k=str();white();if(ch!=':'){break;}
next();o[k]=val();white();if(ch=='}'){next();return o;}else if(ch!=','){break;}
next();white();}}
error("Bad object");}
function num(){var n='',v;if(ch=='-'){n='-';next();}
while(ch>='0'&&ch<='9'){n+=ch;next();}
if(ch=='.'){n+='.';while(next()&&ch>='0'&&ch<='9'){n+=ch;}}
if(ch=='e'||ch=='E'){n+='e';next();if(ch=='-'||ch=='+'){n+=ch;next();}
while(ch>='0'&&ch<='9'){n+=ch;next();}}
v=+n;if(!isFinite(v)){error("Bad number");}else{return v;}}
function word(){switch(ch){case't':if(next()=='r'&&next()=='u'&&next()=='e'){next();return true;}
break;case'f':if(next()=='a'&&next()=='l'&&next()=='s'&&next()=='e'){next();return false;}
break;case'n':if(next()=='u'&&next()=='l'&&next()=='l'){next();return null;}
break;}
error("Syntax error");}
function val(){white();switch(ch){case'{':return obj();case'[':return arr();case'"':return str();case'-':return num();default:return ch>='0'&&ch<='9'?num():word();}}
return val();}};

View file

@ -0,0 +1,101 @@
// Copyright (c) 2005 Thomas Fuchs (http://script.aculo.us, http://mir.aculo.us)
//
// See scriptaculous.js for full license.
var Builder = {
NODEMAP: {
AREA: 'map',
CAPTION: 'table',
COL: 'table',
COLGROUP: 'table',
LEGEND: 'fieldset',
OPTGROUP: 'select',
OPTION: 'select',
PARAM: 'object',
TBODY: 'table',
TD: 'table',
TFOOT: 'table',
TH: 'table',
THEAD: 'table',
TR: 'table'
},
// note: For Firefox < 1.5, OPTION and OPTGROUP tags are currently broken,
// due to a Firefox bug
node: function(elementName) {
elementName = elementName.toUpperCase();
// try innerHTML approach
var parentTag = this.NODEMAP[elementName] || 'div';
var parentElement = document.createElement(parentTag);
try { // prevent IE "feature": http://dev.rubyonrails.org/ticket/2707
parentElement.innerHTML = "<" + elementName + "></" + elementName + ">";
} catch(e) {}
var element = parentElement.firstChild || null;
// see if browser added wrapping tags
if(element && (element.tagName != elementName))
element = element.getElementsByTagName(elementName)[0];
// fallback to createElement approach
if(!element) element = document.createElement(elementName);
// abort if nothing could be created
if(!element) return;
// attributes (or text)
if(arguments[1])
if(this._isStringOrNumber(arguments[1]) ||
(arguments[1] instanceof Array)) {
this._children(element, arguments[1]);
} else {
var attrs = this._attributes(arguments[1]);
if(attrs.length) {
try { // prevent IE "feature": http://dev.rubyonrails.org/ticket/2707
parentElement.innerHTML = "<" +elementName + " " +
attrs + "></" + elementName + ">";
} catch(e) {}
element = parentElement.firstChild || null;
// workaround firefox 1.0.X bug
if(!element) {
element = document.createElement(elementName);
for(attr in arguments[1])
element[attr == 'class' ? 'className' : attr] = arguments[1][attr];
}
if(element.tagName != elementName)
element = parentElement.getElementsByTagName(elementName)[0];
}
}
// text, or array of children
if(arguments[2])
this._children(element, arguments[2]);
return element;
},
_text: function(text) {
return document.createTextNode(text);
},
_attributes: function(attributes) {
var attrs = [];
for(attribute in attributes)
attrs.push((attribute=='className' ? 'class' : attribute) +
'="' + attributes[attribute].toString().escapeHTML() + '"');
return attrs.join(" ");
},
_children: function(element, children) {
if(typeof children=='object') { // array can hold nodes and text
children.flatten().each( function(e) {
if(typeof e=='object')
element.appendChild(e)
else
if(Builder._isStringOrNumber(e))
element.appendChild(Builder._text(e));
});
} else
if(Builder._isStringOrNumber(children))
element.appendChild(Builder._text(children));
},
_isStringOrNumber: function(param) {
return(typeof param=='string' || typeof param=='number');
}
}

750
www/extras/js/sau/controls.js vendored Normal file
View file

@ -0,0 +1,750 @@
// Copyright (c) 2005 Thomas Fuchs (http://script.aculo.us, http://mir.aculo.us)
// (c) 2005 Ivan Krstic (http://blogs.law.harvard.edu/ivan)
// (c) 2005 Jon Tirsen (http://www.tirsen.com)
// Contributors:
// Richard Livsey
// Rahul Bhargava
// Rob Wills
//
// See scriptaculous.js for full license.
// Autocompleter.Base handles all the autocompletion functionality
// that's independent of the data source for autocompletion. This
// includes drawing the autocompletion menu, observing keyboard
// and mouse events, and similar.
//
// Specific autocompleters need to provide, at the very least,
// a getUpdatedChoices function that will be invoked every time
// the text inside the monitored textbox changes. This method
// should get the text for which to provide autocompletion by
// invoking this.getToken(), NOT by directly accessing
// this.element.value. This is to allow incremental tokenized
// autocompletion. Specific auto-completion logic (AJAX, etc)
// belongs in getUpdatedChoices.
//
// Tokenized incremental autocompletion is enabled automatically
// when an autocompleter is instantiated with the 'tokens' option
// in the options parameter, e.g.:
// new Ajax.Autocompleter('id','upd', '/url/', { tokens: ',' });
// will incrementally autocomplete with a comma as the token.
// Additionally, ',' in the above example can be replaced with
// a token array, e.g. { tokens: [',', '\n'] } which
// enables autocompletion on multiple tokens. This is most
// useful when one of the tokens is \n (a newline), as it
// allows smart autocompletion after linebreaks.
var Autocompleter = {}
Autocompleter.Base = function() {};
Autocompleter.Base.prototype = {
baseInitialize: function(element, update, options) {
this.element = $(element);
this.update = $(update);
this.hasFocus = false;
this.changed = false;
this.active = false;
this.index = 0;
this.entryCount = 0;
if (this.setOptions)
this.setOptions(options);
else
this.options = options || {};
this.options.paramName = this.options.paramName || this.element.name;
this.options.tokens = this.options.tokens || [];
this.options.frequency = this.options.frequency || 0.4;
this.options.minChars = this.options.minChars || 1;
this.options.onShow = this.options.onShow ||
function(element, update){
if(!update.style.position || update.style.position=='absolute') {
update.style.position = 'absolute';
Position.clone(element, update, {setHeight: false, offsetTop: element.offsetHeight});
}
Effect.Appear(update,{duration:0.15});
};
this.options.onHide = this.options.onHide ||
function(element, update){ new Effect.Fade(update,{duration:0.15}) };
if (typeof(this.options.tokens) == 'string')
this.options.tokens = new Array(this.options.tokens);
this.observer = null;
this.element.setAttribute('autocomplete','off');
Element.hide(this.update);
Event.observe(this.element, "blur", this.onBlur.bindAsEventListener(this));
Event.observe(this.element, "keypress", this.onKeyPress.bindAsEventListener(this));
},
show: function() {
if(Element.getStyle(this.update, 'display')=='none') this.options.onShow(this.element, this.update);
if(!this.iefix &&
(navigator.appVersion.indexOf('MSIE')>0) &&
(navigator.userAgent.indexOf('Opera')<0) &&
(Element.getStyle(this.update, 'position')=='absolute')) {
new Insertion.After(this.update,
'<iframe id="' + this.update.id + '_iefix" '+
'style="display:none;position:absolute;filter:progid:DXImageTransform.Microsoft.Alpha(opacity=0);" ' +
'src="javascript:false;" frameborder="0" scrolling="no"></iframe>');
this.iefix = $(this.update.id+'_iefix');
}
if(this.iefix) setTimeout(this.fixIEOverlapping.bind(this), 50);
},
fixIEOverlapping: function() {
Position.clone(this.update, this.iefix);
this.iefix.style.zIndex = 1;
this.update.style.zIndex = 2;
Element.show(this.iefix);
},
hide: function() {
this.stopIndicator();
if(Element.getStyle(this.update, 'display')!='none') this.options.onHide(this.element, this.update);
if(this.iefix) Element.hide(this.iefix);
},
startIndicator: function() {
if(this.options.indicator) Element.show(this.options.indicator);
},
stopIndicator: function() {
if(this.options.indicator) Element.hide(this.options.indicator);
},
onKeyPress: function(event) {
if(this.active)
switch(event.keyCode) {
case Event.KEY_TAB:
case Event.KEY_RETURN:
this.selectEntry();
Event.stop(event);
case Event.KEY_ESC:
this.hide();
this.active = false;
Event.stop(event);
return;
case Event.KEY_LEFT:
case Event.KEY_RIGHT:
return;
case Event.KEY_UP:
this.markPrevious();
this.render();
if(navigator.appVersion.indexOf('AppleWebKit')>0) Event.stop(event);
return;
case Event.KEY_DOWN:
this.markNext();
this.render();
if(navigator.appVersion.indexOf('AppleWebKit')>0) Event.stop(event);
return;
}
else
if(event.keyCode==Event.KEY_TAB || event.keyCode==Event.KEY_RETURN)
return;
this.changed = true;
this.hasFocus = true;
if(this.observer) clearTimeout(this.observer);
this.observer =
setTimeout(this.onObserverEvent.bind(this), this.options.frequency*1000);
},
onHover: function(event) {
var element = Event.findElement(event, 'LI');
if(this.index != element.autocompleteIndex)
{
this.index = element.autocompleteIndex;
this.render();
}
Event.stop(event);
},
onClick: function(event) {
var element = Event.findElement(event, 'LI');
this.index = element.autocompleteIndex;
this.selectEntry();
this.hide();
},
onBlur: function(event) {
// needed to make click events working
setTimeout(this.hide.bind(this), 250);
this.hasFocus = false;
this.active = false;
},
render: function() {
if(this.entryCount > 0) {
for (var i = 0; i < this.entryCount; i++)
this.index==i ?
Element.addClassName(this.getEntry(i),"selected") :
Element.removeClassName(this.getEntry(i),"selected");
if(this.hasFocus) {
this.show();
this.active = true;
}
} else {
this.active = false;
this.hide();
}
},
markPrevious: function() {
if(this.index > 0) this.index--
else this.index = this.entryCount-1;
},
markNext: function() {
if(this.index < this.entryCount-1) this.index++
else this.index = 0;
},
getEntry: function(index) {
return this.update.firstChild.childNodes[index];
},
getCurrentEntry: function() {
return this.getEntry(this.index);
},
selectEntry: function() {
this.active = false;
this.updateElement(this.getCurrentEntry());
},
updateElement: function(selectedElement) {
if (this.options.updateElement) {
this.options.updateElement(selectedElement);
return;
}
var value = Element.collectTextNodesIgnoreClass(selectedElement, 'informal');
var lastTokenPos = this.findLastToken();
if (lastTokenPos != -1) {
var newValue = this.element.value.substr(0, lastTokenPos + 1);
var whitespace = this.element.value.substr(lastTokenPos + 1).match(/^\s+/);
if (whitespace)
newValue += whitespace[0];
this.element.value = newValue + value;
} else {
this.element.value = value;
}
this.element.focus();
if (this.options.afterUpdateElement)
this.options.afterUpdateElement(this.element, selectedElement);
},
updateChoices: function(choices) {
if(!this.changed && this.hasFocus) {
this.update.innerHTML = choices;
Element.cleanWhitespace(this.update);
Element.cleanWhitespace(this.update.firstChild);
if(this.update.firstChild && this.update.firstChild.childNodes) {
this.entryCount =
this.update.firstChild.childNodes.length;
for (var i = 0; i < this.entryCount; i++) {
var entry = this.getEntry(i);
entry.autocompleteIndex = i;
this.addObservers(entry);
}
} else {
this.entryCount = 0;
}
this.stopIndicator();
this.index = 0;
this.render();
}
},
addObservers: function(element) {
Event.observe(element, "mouseover", this.onHover.bindAsEventListener(this));
Event.observe(element, "click", this.onClick.bindAsEventListener(this));
},
onObserverEvent: function() {
this.changed = false;
if(this.getToken().length>=this.options.minChars) {
this.startIndicator();
this.getUpdatedChoices();
} else {
this.active = false;
this.hide();
}
},
getToken: function() {
var tokenPos = this.findLastToken();
if (tokenPos != -1)
var ret = this.element.value.substr(tokenPos + 1).replace(/^\s+/,'').replace(/\s+$/,'');
else
var ret = this.element.value;
return /\n/.test(ret) ? '' : ret;
},
findLastToken: function() {
var lastTokenPos = -1;
for (var i=0; i<this.options.tokens.length; i++) {
var thisTokenPos = this.element.value.lastIndexOf(this.options.tokens[i]);
if (thisTokenPos > lastTokenPos)
lastTokenPos = thisTokenPos;
}
return lastTokenPos;
}
}
Ajax.Autocompleter = Class.create();
Object.extend(Object.extend(Ajax.Autocompleter.prototype, Autocompleter.Base.prototype), {
initialize: function(element, update, url, options) {
this.baseInitialize(element, update, options);
this.options.asynchronous = true;
this.options.onComplete = this.onComplete.bind(this);
this.options.defaultParams = this.options.parameters || null;
this.url = url;
},
getUpdatedChoices: function() {
entry = encodeURIComponent(this.options.paramName) + '=' +
encodeURIComponent(this.getToken());
this.options.parameters = this.options.callback ?
this.options.callback(this.element, entry) : entry;
if(this.options.defaultParams)
this.options.parameters += '&' + this.options.defaultParams;
new Ajax.Request(this.url, this.options);
},
onComplete: function(request) {
this.updateChoices(request.responseText);
}
});
// The local array autocompleter. Used when you'd prefer to
// inject an array of autocompletion options into the page, rather
// than sending out Ajax queries, which can be quite slow sometimes.
//
// The constructor takes four parameters. The first two are, as usual,
// the id of the monitored textbox, and id of the autocompletion menu.
// The third is the array you want to autocomplete from, and the fourth
// is the options block.
//
// Extra local autocompletion options:
// - choices - How many autocompletion choices to offer
//
// - partialSearch - If false, the autocompleter will match entered
// text only at the beginning of strings in the
// autocomplete array. Defaults to true, which will
// match text at the beginning of any *word* in the
// strings in the autocomplete array. If you want to
// search anywhere in the string, additionally set
// the option fullSearch to true (default: off).
//
// - fullSsearch - Search anywhere in autocomplete array strings.
//
// - partialChars - How many characters to enter before triggering
// a partial match (unlike minChars, which defines
// how many characters are required to do any match
// at all). Defaults to 2.
//
// - ignoreCase - Whether to ignore case when autocompleting.
// Defaults to true.
//
// It's possible to pass in a custom function as the 'selector'
// option, if you prefer to write your own autocompletion logic.
// In that case, the other options above will not apply unless
// you support them.
Autocompleter.Local = Class.create();
Autocompleter.Local.prototype = Object.extend(new Autocompleter.Base(), {
initialize: function(element, update, array, options) {
this.baseInitialize(element, update, options);
this.options.array = array;
},
getUpdatedChoices: function() {
this.updateChoices(this.options.selector(this));
},
setOptions: function(options) {
this.options = Object.extend({
choices: 10,
partialSearch: true,
partialChars: 2,
ignoreCase: true,
fullSearch: false,
selector: function(instance) {
var ret = []; // Beginning matches
var partial = []; // Inside matches
var entry = instance.getToken();
var count = 0;
for (var i = 0; i < instance.options.array.length &&
ret.length < instance.options.choices ; i++) {
var elem = instance.options.array[i];
var foundPos = instance.options.ignoreCase ?
elem.toLowerCase().indexOf(entry.toLowerCase()) :
elem.indexOf(entry);
while (foundPos != -1) {
if (foundPos == 0 && elem.length != entry.length) {
ret.push("<li><strong>" + elem.substr(0, entry.length) + "</strong>" +
elem.substr(entry.length) + "</li>");
break;
} else if (entry.length >= instance.options.partialChars &&
instance.options.partialSearch && foundPos != -1) {
if (instance.options.fullSearch || /\s/.test(elem.substr(foundPos-1,1))) {
partial.push("<li>" + elem.substr(0, foundPos) + "<strong>" +
elem.substr(foundPos, entry.length) + "</strong>" + elem.substr(
foundPos + entry.length) + "</li>");
break;
}
}
foundPos = instance.options.ignoreCase ?
elem.toLowerCase().indexOf(entry.toLowerCase(), foundPos + 1) :
elem.indexOf(entry, foundPos + 1);
}
}
if (partial.length)
ret = ret.concat(partial.slice(0, instance.options.choices - ret.length))
return "<ul>" + ret.join('') + "</ul>";
}
}, options || {});
}
});
// AJAX in-place editor
//
// see documentation on http://wiki.script.aculo.us/scriptaculous/show/Ajax.InPlaceEditor
// Use this if you notice weird scrolling problems on some browsers,
// the DOM might be a bit confused when this gets called so do this
// waits 1 ms (with setTimeout) until it does the activation
Field.scrollFreeActivate = function(field) {
setTimeout(function() {
Field.activate(field);
}, 1);
}
Ajax.InPlaceEditor = Class.create();
Ajax.InPlaceEditor.defaultHighlightColor = "#FFFF99";
Ajax.InPlaceEditor.prototype = {
initialize: function(element, url, options) {
this.url = url;
this.element = $(element);
this.options = Object.extend({
okText: "ok",
cancelText: "cancel",
savingText: "Saving...",
clickToEditText: "Click to edit",
okText: "ok",
rows: 1,
onComplete: function(transport, element) {
new Effect.Highlight(element, {startcolor: this.options.highlightcolor});
},
onFailure: function(transport) {
alert("Error communicating with the server: " + transport.responseText.stripTags());
},
callback: function(form) {
return Form.serialize(form);
},
handleLineBreaks: true,
loadingText: 'Loading...',
savingClassName: 'inplaceeditor-saving',
loadingClassName: 'inplaceeditor-loading',
formClassName: 'inplaceeditor-form',
highlightcolor: Ajax.InPlaceEditor.defaultHighlightColor,
highlightendcolor: "#FFFFFF",
externalControl: null,
ajaxOptions: {}
}, options || {});
if(!this.options.formId && this.element.id) {
this.options.formId = this.element.id + "-inplaceeditor";
if ($(this.options.formId)) {
// there's already a form with that name, don't specify an id
this.options.formId = null;
}
}
if (this.options.externalControl) {
this.options.externalControl = $(this.options.externalControl);
}
this.originalBackground = Element.getStyle(this.element, 'background-color');
if (!this.originalBackground) {
this.originalBackground = "transparent";
}
this.element.title = this.options.clickToEditText;
this.onclickListener = this.enterEditMode.bindAsEventListener(this);
this.mouseoverListener = this.enterHover.bindAsEventListener(this);
this.mouseoutListener = this.leaveHover.bindAsEventListener(this);
Event.observe(this.element, 'click', this.onclickListener);
Event.observe(this.element, 'mouseover', this.mouseoverListener);
Event.observe(this.element, 'mouseout', this.mouseoutListener);
if (this.options.externalControl) {
Event.observe(this.options.externalControl, 'click', this.onclickListener);
Event.observe(this.options.externalControl, 'mouseover', this.mouseoverListener);
Event.observe(this.options.externalControl, 'mouseout', this.mouseoutListener);
}
},
enterEditMode: function(evt) {
if (this.saving) return;
if (this.editing) return;
this.editing = true;
this.onEnterEditMode();
if (this.options.externalControl) {
Element.hide(this.options.externalControl);
}
Element.hide(this.element);
this.createForm();
this.element.parentNode.insertBefore(this.form, this.element);
Field.scrollFreeActivate(this.editField);
// stop the event to avoid a page refresh in Safari
if (evt) {
Event.stop(evt);
}
return false;
},
createForm: function() {
this.form = document.createElement("form");
this.form.id = this.options.formId;
Element.addClassName(this.form, this.options.formClassName)
this.form.onsubmit = this.onSubmit.bind(this);
this.createEditField();
if (this.options.textarea) {
var br = document.createElement("br");
this.form.appendChild(br);
}
okButton = document.createElement("input");
okButton.type = "submit";
okButton.value = this.options.okText;
this.form.appendChild(okButton);
cancelLink = document.createElement("a");
cancelLink.href = "#";
cancelLink.appendChild(document.createTextNode(this.options.cancelText));
cancelLink.onclick = this.onclickCancel.bind(this);
this.form.appendChild(cancelLink);
},
hasHTMLLineBreaks: function(string) {
if (!this.options.handleLineBreaks) return false;
return string.match(/<br/i) || string.match(/<p>/i);
},
convertHTMLLineBreaks: function(string) {
return string.replace(/<br>/gi, "\n").replace(/<br\/>/gi, "\n").replace(/<\/p>/gi, "\n").replace(/<p>/gi, "");
},
createEditField: function() {
var text;
if(this.options.loadTextURL) {
text = this.options.loadingText;
} else {
text = this.getText();
}
if (this.options.rows == 1 && !this.hasHTMLLineBreaks(text)) {
this.options.textarea = false;
var textField = document.createElement("input");
textField.type = "text";
textField.name = "value";
textField.value = text;
textField.style.backgroundColor = this.options.highlightcolor;
var size = this.options.size || this.options.cols || 0;
if (size != 0) textField.size = size;
this.editField = textField;
} else {
this.options.textarea = true;
var textArea = document.createElement("textarea");
textArea.name = "value";
textArea.value = this.convertHTMLLineBreaks(text);
textArea.rows = this.options.rows;
textArea.cols = this.options.cols || 40;
this.editField = textArea;
}
if(this.options.loadTextURL) {
this.loadExternalText();
}
this.form.appendChild(this.editField);
},
getText: function() {
return this.element.innerHTML;
},
loadExternalText: function() {
Element.addClassName(this.form, this.options.loadingClassName);
this.editField.disabled = true;
new Ajax.Request(
this.options.loadTextURL,
Object.extend({
asynchronous: true,
onComplete: this.onLoadedExternalText.bind(this)
}, this.options.ajaxOptions)
);
},
onLoadedExternalText: function(transport) {
Element.removeClassName(this.form, this.options.loadingClassName);
this.editField.disabled = false;
this.editField.value = transport.responseText.stripTags();
},
onclickCancel: function() {
this.onComplete();
this.leaveEditMode();
return false;
},
onFailure: function(transport) {
this.options.onFailure(transport);
if (this.oldInnerHTML) {
this.element.innerHTML = this.oldInnerHTML;
this.oldInnerHTML = null;
}
return false;
},
onSubmit: function() {
// onLoading resets these so we need to save them away for the Ajax call
var form = this.form;
var value = this.editField.value;
// do this first, sometimes the ajax call returns before we get a chance to switch on Saving...
// which means this will actually switch on Saving... *after* we've left edit mode causing Saving...
// to be displayed indefinitely
this.onLoading();
new Ajax.Updater(
{
success: this.element,
// don't update on failure (this could be an option)
failure: null
},
this.url,
Object.extend({
parameters: this.options.callback(form, value),
onComplete: this.onComplete.bind(this),
onFailure: this.onFailure.bind(this)
}, this.options.ajaxOptions)
);
// stop the event to avoid a page refresh in Safari
if (arguments.length > 1) {
Event.stop(arguments[0]);
}
return false;
},
onLoading: function() {
this.saving = true;
this.removeForm();
this.leaveHover();
this.showSaving();
},
showSaving: function() {
this.oldInnerHTML = this.element.innerHTML;
this.element.innerHTML = this.options.savingText;
Element.addClassName(this.element, this.options.savingClassName);
this.element.style.backgroundColor = this.originalBackground;
Element.show(this.element);
},
removeForm: function() {
if(this.form) {
if (this.form.parentNode) Element.remove(this.form);
this.form = null;
}
},
enterHover: function() {
if (this.saving) return;
this.element.style.backgroundColor = this.options.highlightcolor;
if (this.effect) {
this.effect.cancel();
}
Element.addClassName(this.element, this.options.hoverClassName)
},
leaveHover: function() {
if (this.options.backgroundColor) {
this.element.style.backgroundColor = this.oldBackground;
}
Element.removeClassName(this.element, this.options.hoverClassName)
if (this.saving) return;
this.effect = new Effect.Highlight(this.element, {
startcolor: this.options.highlightcolor,
endcolor: this.options.highlightendcolor,
restorecolor: this.originalBackground
});
},
leaveEditMode: function() {
Element.removeClassName(this.element, this.options.savingClassName);
this.removeForm();
this.leaveHover();
this.element.style.backgroundColor = this.originalBackground;
Element.show(this.element);
if (this.options.externalControl) {
Element.show(this.options.externalControl);
}
this.editing = false;
this.saving = false;
this.oldInnerHTML = null;
this.onLeaveEditMode();
},
onComplete: function(transport) {
this.leaveEditMode();
this.options.onComplete.bind(this)(transport, this.element);
},
onEnterEditMode: function() {},
onLeaveEditMode: function() {},
dispose: function() {
if (this.oldInnerHTML) {
this.element.innerHTML = this.oldInnerHTML;
}
this.leaveEditMode();
Event.stopObserving(this.element, 'click', this.onclickListener);
Event.stopObserving(this.element, 'mouseover', this.mouseoverListener);
Event.stopObserving(this.element, 'mouseout', this.mouseoutListener);
if (this.options.externalControl) {
Event.stopObserving(this.options.externalControl, 'click', this.onclickListener);
Event.stopObserving(this.options.externalControl, 'mouseover', this.mouseoverListener);
Event.stopObserving(this.options.externalControl, 'mouseout', this.mouseoutListener);
}
}
};
// Delayed observer, like Form.Element.Observer,
// but waits for delay after last key input
// Ideal for live-search fields
Form.Element.DelayedObserver = Class.create();
Form.Element.DelayedObserver.prototype = {
initialize: function(element, delay, callback) {
this.delay = delay || 0.5;
this.element = $(element);
this.callback = callback;
this.timer = null;
this.lastValue = $F(this.element);
Event.observe(this.element,'keyup',this.delayedListener.bindAsEventListener(this));
},
delayedListener: function(event) {
if(this.lastValue == $F(this.element)) return;
if(this.timer) clearTimeout(this.timer);
this.timer = setTimeout(this.onTimerEvent.bind(this), this.delay * 1000);
this.lastValue = $F(this.element);
},
onTimerEvent: function() {
this.timer = null;
this.callback(this.element, $F(this.element));
}
};

579
www/extras/js/sau/dragdrop.js vendored Normal file
View file

@ -0,0 +1,579 @@
// Copyright (c) 2005 Thomas Fuchs (http://script.aculo.us, http://mir.aculo.us)
//
// See scriptaculous.js for full license.
/*--------------------------------------------------------------------------*/
var Droppables = {
drops: [],
remove: function(element) {
this.drops = this.drops.reject(function(d) { return d.element==$(element) });
},
add: function(element) {
element = $(element);
var options = Object.extend({
greedy: true,
hoverclass: null
}, arguments[1] || {});
// cache containers
if(options.containment) {
options._containers = [];
var containment = options.containment;
if((typeof containment == 'object') &&
(containment.constructor == Array)) {
containment.each( function(c) { options._containers.push($(c)) });
} else {
options._containers.push($(containment));
}
}
if(options.accept) options.accept = [options.accept].flatten();
Element.makePositioned(element); // fix IE
options.element = element;
this.drops.push(options);
},
isContained: function(element, drop) {
var parentNode = element.parentNode;
return drop._containers.detect(function(c) { return parentNode == c });
},
isAffected: function(point, element, drop) {
return (
(drop.element!=element) &&
((!drop._containers) ||
this.isContained(element, drop)) &&
((!drop.accept) ||
(Element.classNames(element).detect(
function(v) { return drop.accept.include(v) } ) )) &&
Position.within(drop.element, point[0], point[1]) );
},
deactivate: function(drop) {
if(drop.hoverclass)
Element.removeClassName(drop.element, drop.hoverclass);
this.last_active = null;
},
activate: function(drop) {
if(drop.hoverclass)
Element.addClassName(drop.element, drop.hoverclass);
this.last_active = drop;
},
show: function(point, element) {
if(!this.drops.length) return;
if(this.last_active) this.deactivate(this.last_active);
this.drops.each( function(drop) {
if(Droppables.isAffected(point, element, drop)) {
if(drop.onHover)
drop.onHover(element, drop.element, Position.overlap(drop.overlap, drop.element));
if(drop.greedy) {
Droppables.activate(drop);
throw $break;
}
}
});
},
fire: function(event, element) {
if(!this.last_active) return;
Position.prepare();
if (this.isAffected([Event.pointerX(event), Event.pointerY(event)], element, this.last_active))
if (this.last_active.onDrop)
this.last_active.onDrop(element, this.last_active.element, event);
},
reset: function() {
if(this.last_active)
this.deactivate(this.last_active);
}
}
var Draggables = {
drags: [],
observers: [],
register: function(draggable) {
if(this.drags.length == 0) {
this.eventMouseUp = this.endDrag.bindAsEventListener(this);
this.eventMouseMove = this.updateDrag.bindAsEventListener(this);
this.eventKeypress = this.keyPress.bindAsEventListener(this);
Event.observe(document, "mouseup", this.eventMouseUp);
Event.observe(document, "mousemove", this.eventMouseMove);
Event.observe(document, "keypress", this.eventKeypress);
}
this.drags.push(draggable);
},
unregister: function(draggable) {
this.drags = this.drags.reject(function(d) { return d==draggable });
if(this.drags.length == 0) {
Event.stopObserving(document, "mouseup", this.eventMouseUp);
Event.stopObserving(document, "mousemove", this.eventMouseMove);
Event.stopObserving(document, "keypress", this.eventKeypress);
}
},
activate: function(draggable) {
window.focus(); // allows keypress events if window isn't currently focused, fails for Safari
this.activeDraggable = draggable;
},
deactivate: function(draggbale) {
this.activeDraggable = null;
},
updateDrag: function(event) {
if(!this.activeDraggable) return;
var pointer = [Event.pointerX(event), Event.pointerY(event)];
// Mozilla-based browsers fire successive mousemove events with
// the same coordinates, prevent needless redrawing (moz bug?)
if(this._lastPointer && (this._lastPointer.inspect() == pointer.inspect())) return;
this._lastPointer = pointer;
this.activeDraggable.updateDrag(event, pointer);
},
endDrag: function(event) {
if(!this.activeDraggable) return;
this._lastPointer = null;
this.activeDraggable.endDrag(event);
},
keyPress: function(event) {
if(this.activeDraggable)
this.activeDraggable.keyPress(event);
},
addObserver: function(observer) {
this.observers.push(observer);
this._cacheObserverCallbacks();
},
removeObserver: function(element) { // element instead of observer fixes mem leaks
this.observers = this.observers.reject( function(o) { return o.element==element });
this._cacheObserverCallbacks();
},
notify: function(eventName, draggable, event) { // 'onStart', 'onEnd', 'onDrag'
if(this[eventName+'Count'] > 0)
this.observers.each( function(o) {
if(o[eventName]) o[eventName](eventName, draggable, event);
});
},
_cacheObserverCallbacks: function() {
['onStart','onEnd','onDrag'].each( function(eventName) {
Draggables[eventName+'Count'] = Draggables.observers.select(
function(o) { return o[eventName]; }
).length;
});
}
}
/*--------------------------------------------------------------------------*/
var Draggable = Class.create();
Draggable.prototype = {
initialize: function(element) {
var options = Object.extend({
handle: false,
starteffect: function(element) {
new Effect.Opacity(element, {duration:0.2, from:1.0, to:0.7});
},
reverteffect: function(element, top_offset, left_offset) {
var dur = Math.sqrt(Math.abs(top_offset^2)+Math.abs(left_offset^2))*0.02;
new Effect.MoveBy(element, -top_offset, -left_offset, {duration:dur});
},
endeffect: function(element) {
new Effect.Opacity(element, {duration:0.2, from:0.7, to:1.0});
},
zindex: 1000,
revert: false,
snap: false // false, or xy or [x,y] or function(x,y){ return [x,y] }
}, arguments[1] || {});
this.element = $(element);
if(options.handle && (typeof options.handle == 'string'))
this.handle = Element.childrenWithClassName(this.element, options.handle)[0];
if(!this.handle) this.handle = $(options.handle);
if(!this.handle) this.handle = this.element;
Element.makePositioned(this.element); // fix IE
this.delta = this.currentDelta();
this.options = options;
this.dragging = false;
this.eventMouseDown = this.initDrag.bindAsEventListener(this);
Event.observe(this.handle, "mousedown", this.eventMouseDown);
Draggables.register(this);
},
destroy: function() {
Event.stopObserving(this.handle, "mousedown", this.eventMouseDown);
Draggables.unregister(this);
},
currentDelta: function() {
return([
parseInt(this.element.style.left || '0'),
parseInt(this.element.style.top || '0')]);
},
initDrag: function(event) {
if(Event.isLeftClick(event)) {
// abort on form elements, fixes a Firefox issue
var src = Event.element(event);
if(src.tagName && (
src.tagName=='INPUT' ||
src.tagName=='SELECT' ||
src.tagName=='BUTTON' ||
src.tagName=='TEXTAREA')) return;
var pointer = [Event.pointerX(event), Event.pointerY(event)];
var pos = Position.cumulativeOffset(this.element);
this.offset = [0,1].map( function(i) { return (pointer[i] - pos[i]) });
Draggables.activate(this);
Event.stop(event);
}
},
startDrag: function(event) {
this.dragging = true;
if(this.options.zindex) {
this.originalZ = parseInt(Element.getStyle(this.element,'z-index') || 0);
this.element.style.zIndex = this.options.zindex;
}
if(this.options.ghosting) {
this._clone = this.element.cloneNode(true);
Position.absolutize(this.element);
this.element.parentNode.insertBefore(this._clone, this.element);
}
Draggables.notify('onStart', this, event);
if(this.options.starteffect) this.options.starteffect(this.element);
},
updateDrag: function(event, pointer) {
if(!this.dragging) this.startDrag(event);
Position.prepare();
Droppables.show(pointer, this.element);
Draggables.notify('onDrag', this, event);
this.draw(pointer);
if(this.options.change) this.options.change(this);
// fix AppleWebKit rendering
if(navigator.appVersion.indexOf('AppleWebKit')>0) window.scrollBy(0,0);
Event.stop(event);
},
finishDrag: function(event, success) {
this.dragging = false;
if(this.options.ghosting) {
Position.relativize(this.element);
Element.remove(this._clone);
this._clone = null;
}
if(success) Droppables.fire(event, this.element);
Draggables.notify('onEnd', this, event);
var revert = this.options.revert;
if(revert && typeof revert == 'function') revert = revert(this.element);
var d = this.currentDelta();
if(revert && this.options.reverteffect) {
this.options.reverteffect(this.element,
d[1]-this.delta[1], d[0]-this.delta[0]);
} else {
this.delta = d;
}
if(this.options.zindex)
this.element.style.zIndex = this.originalZ;
if(this.options.endeffect)
this.options.endeffect(this.element);
Draggables.deactivate(this);
Droppables.reset();
},
keyPress: function(event) {
if(!event.keyCode==Event.KEY_ESC) return;
this.finishDrag(event, false);
Event.stop(event);
},
endDrag: function(event) {
if(!this.dragging) return;
this.finishDrag(event, true);
Event.stop(event);
},
draw: function(point) {
var pos = Position.cumulativeOffset(this.element);
var d = this.currentDelta();
pos[0] -= d[0]; pos[1] -= d[1];
var p = [0,1].map(function(i){ return (point[i]-pos[i]-this.offset[i]) }.bind(this));
if(this.options.snap) {
if(typeof this.options.snap == 'function') {
p = this.options.snap(p[0],p[1]);
} else {
if(this.options.snap instanceof Array) {
p = p.map( function(v, i) {
return Math.round(v/this.options.snap[i])*this.options.snap[i] }.bind(this))
} else {
p = p.map( function(v) {
return Math.round(v/this.options.snap)*this.options.snap }.bind(this))
}
}}
var style = this.element.style;
if((!this.options.constraint) || (this.options.constraint=='horizontal'))
style.left = p[0] + "px";
if((!this.options.constraint) || (this.options.constraint=='vertical'))
style.top = p[1] + "px";
if(style.visibility=="hidden") style.visibility = ""; // fix gecko rendering
}
}
/*--------------------------------------------------------------------------*/
var SortableObserver = Class.create();
SortableObserver.prototype = {
initialize: function(element, observer) {
this.element = $(element);
this.observer = observer;
this.lastValue = Sortable.serialize(this.element);
},
onStart: function() {
this.lastValue = Sortable.serialize(this.element);
},
onEnd: function() {
Sortable.unmark();
if(this.lastValue != Sortable.serialize(this.element))
this.observer(this.element)
}
}
var Sortable = {
sortables: new Array(),
options: function(element){
element = $(element);
return this.sortables.detect(function(s) { return s.element == element });
},
destroy: function(element){
element = $(element);
this.sortables.findAll(function(s) { return s.element == element }).each(function(s){
Draggables.removeObserver(s.element);
s.droppables.each(function(d){ Droppables.remove(d) });
s.draggables.invoke('destroy');
});
this.sortables = this.sortables.reject(function(s) { return s.element == element });
},
create: function(element) {
element = $(element);
var options = Object.extend({
element: element,
tag: 'li', // assumes li children, override with tag: 'tagname'
dropOnEmpty: false,
tree: false, // fixme: unimplemented
overlap: 'vertical', // one of 'vertical', 'horizontal'
constraint: 'vertical', // one of 'vertical', 'horizontal', false
containment: element, // also takes array of elements (or id's); or false
handle: false, // or a CSS class
only: false,
hoverclass: null,
ghosting: false,
format: null,
onChange: Prototype.emptyFunction,
onUpdate: Prototype.emptyFunction
}, arguments[1] || {});
// clear any old sortable with same element
this.destroy(element);
// build options for the draggables
var options_for_draggable = {
revert: true,
ghosting: options.ghosting,
constraint: options.constraint,
handle: options.handle };
if(options.starteffect)
options_for_draggable.starteffect = options.starteffect;
if(options.reverteffect)
options_for_draggable.reverteffect = options.reverteffect;
else
if(options.ghosting) options_for_draggable.reverteffect = function(element) {
element.style.top = 0;
element.style.left = 0;
};
if(options.endeffect)
options_for_draggable.endeffect = options.endeffect;
if(options.zindex)
options_for_draggable.zindex = options.zindex;
// build options for the droppables
var options_for_droppable = {
overlap: options.overlap,
containment: options.containment,
hoverclass: options.hoverclass,
onHover: Sortable.onHover,
greedy: !options.dropOnEmpty
}
// fix for gecko engine
Element.cleanWhitespace(element);
options.draggables = [];
options.droppables = [];
// make it so
// drop on empty handling
if(options.dropOnEmpty) {
Droppables.add(element,
{containment: options.containment, onHover: Sortable.onEmptyHover, greedy: false});
options.droppables.push(element);
}
(this.findElements(element, options) || []).each( function(e) {
// handles are per-draggable
var handle = options.handle ?
Element.childrenWithClassName(e, options.handle)[0] : e;
options.draggables.push(
new Draggable(e, Object.extend(options_for_draggable, { handle: handle })));
Droppables.add(e, options_for_droppable);
options.droppables.push(e);
});
// keep reference
this.sortables.push(options);
// for onupdate
Draggables.addObserver(new SortableObserver(element, options.onUpdate));
},
// return all suitable-for-sortable elements in a guaranteed order
findElements: function(element, options) {
if(!element.hasChildNodes()) return null;
var elements = [];
$A(element.childNodes).each( function(e) {
if(e.tagName && e.tagName==options.tag.toUpperCase() &&
(!options.only || (Element.hasClassName(e, options.only))))
elements.push(e);
if(options.tree) {
var grandchildren = this.findElements(e, options);
if(grandchildren) elements.push(grandchildren);
}
});
return (elements.length>0 ? elements.flatten() : null);
},
onHover: function(element, dropon, overlap) {
if(overlap>0.5) {
Sortable.mark(dropon, 'before');
if(dropon.previousSibling != element) {
var oldParentNode = element.parentNode;
element.style.visibility = "hidden"; // fix gecko rendering
dropon.parentNode.insertBefore(element, dropon);
if(dropon.parentNode!=oldParentNode)
Sortable.options(oldParentNode).onChange(element);
Sortable.options(dropon.parentNode).onChange(element);
}
} else {
Sortable.mark(dropon, 'after');
var nextElement = dropon.nextSibling || null;
if(nextElement != element) {
var oldParentNode = element.parentNode;
element.style.visibility = "hidden"; // fix gecko rendering
dropon.parentNode.insertBefore(element, nextElement);
if(dropon.parentNode!=oldParentNode)
Sortable.options(oldParentNode).onChange(element);
Sortable.options(dropon.parentNode).onChange(element);
}
}
},
onEmptyHover: function(element, dropon) {
if(element.parentNode!=dropon) {
var oldParentNode = element.parentNode;
dropon.appendChild(element);
Sortable.options(oldParentNode).onChange(element);
Sortable.options(dropon).onChange(element);
}
},
unmark: function() {
if(Sortable._marker) Element.hide(Sortable._marker);
},
mark: function(dropon, position) {
// mark on ghosting only
var sortable = Sortable.options(dropon.parentNode);
if(sortable && !sortable.ghosting) return;
if(!Sortable._marker) {
Sortable._marker = $('dropmarker') || document.createElement('DIV');
Element.hide(Sortable._marker);
Element.addClassName(Sortable._marker, 'dropmarker');
Sortable._marker.style.position = 'absolute';
document.getElementsByTagName("body").item(0).appendChild(Sortable._marker);
}
var offsets = Position.cumulativeOffset(dropon);
Sortable._marker.style.left = offsets[0] + 'px';
Sortable._marker.style.top = offsets[1] + 'px';
if(position=='after')
if(sortable.overlap == 'horizontal')
Sortable._marker.style.left = (offsets[0]+dropon.clientWidth) + 'px';
else
Sortable._marker.style.top = (offsets[1]+dropon.clientHeight) + 'px';
Element.show(Sortable._marker);
},
serialize: function(element) {
element = $(element);
var sortableOptions = this.options(element);
var options = Object.extend({
tag: sortableOptions.tag,
only: sortableOptions.only,
name: element.id,
format: sortableOptions.format || /^[^_]*_(.*)$/
}, arguments[1] || {});
return $(this.findElements(element, options) || []).map( function(item) {
return (encodeURIComponent(options.name) + "[]=" +
encodeURIComponent(item.id.match(options.format) ? item.id.match(options.format)[1] : ''));
}).join(";");
}
}

841
www/extras/js/sau/effects.js vendored Normal file
View file

@ -0,0 +1,841 @@
// Copyright (c) 2005 Thomas Fuchs (http://script.aculo.us, http://mir.aculo.us)
// Contributors:
// Justin Palmer (http://encytemedia.com/)
// Mark Pilgrim (http://diveintomark.org/)
// Martin Bialasinki
//
// See scriptaculous.js for full license.
/* ------------- element ext -------------- */
// converts rgb() and #xxx to #xxxxxx format,
// returns self (or first argument) if not convertable
String.prototype.parseColor = function() {
var color = '#';
if(this.slice(0,4) == 'rgb(') {
var cols = this.slice(4,this.length-1).split(',');
var i=0; do { color += parseInt(cols[i]).toColorPart() } while (++i<3);
} else {
if(this.slice(0,1) == '#') {
if(this.length==4) for(var i=1;i<4;i++) color += (this.charAt(i) + this.charAt(i)).toLowerCase();
if(this.length==7) color = this.toLowerCase();
}
}
return(color.length==7 ? color : (arguments[0] || this));
}
Element.collectTextNodesIgnoreClass = function(element, ignoreclass) {
var children = $(element).childNodes;
var text = '';
var classtest = new RegExp('^([^ ]+ )*' + ignoreclass+ '( [^ ]+)*$','i');
for (var i = 0; i < children.length; i++) {
if(children[i].nodeType==3) {
text+=children[i].nodeValue;
} else {
if((!children[i].className.match(classtest)) && children[i].hasChildNodes())
text += Element.collectTextNodesIgnoreClass(children[i], ignoreclass);
}
}
return text;
}
Element.setStyle = function(element, style) {
element = $(element);
for(k in style) element.style[k.camelize()] = style[k];
}
Element.setContentZoom = function(element, percent) {
Element.setStyle(element, {fontSize: (percent/100) + 'em'});
if(navigator.appVersion.indexOf('AppleWebKit')>0) window.scrollBy(0,0);
}
Element.getOpacity = function(element){
var opacity;
if (opacity = Element.getStyle(element, 'opacity'))
return parseFloat(opacity);
if (opacity = (Element.getStyle(element, 'filter') || '').match(/alpha\(opacity=(.*)\)/))
if(opacity[1]) return parseFloat(opacity[1]) / 100;
return 1.0;
}
Element.setOpacity = function(element, value){
element= $(element);
if (value == 1){
Element.setStyle(element, {opacity: 0.999999});
if(/MSIE/.test(navigator.userAgent))
Element.setStyle(element, {filter: Element.getStyle(element,'filter').replace(/alpha\([^\)]*\)/gi,'')});
} else {
if(value < 0.00001) value = 0;
Element.setStyle(element, {opacity: value});
if(/MSIE/.test(navigator.userAgent))
Element.setStyle(element,
{ filter: Element.getStyle(element,'filter').replace(/alpha\([^\)]*\)/gi,'') +
'alpha(opacity='+value*100+')' });
}
}
Element.getInlineOpacity = function(element){
return $(element).style.opacity || '';
}
Element.childrenWithClassName = function(element, className) {
return $A($(element).getElementsByTagName('*')).select(
function(c) { return Element.hasClassName(c, className) });
}
Array.prototype.call = function() {
var args = arguments;
this.each(function(f){ f.apply(this, args) });
}
/*--------------------------------------------------------------------------*/
var Effect = {
tagifyText: function(element) {
var tagifyStyle = 'position:relative';
if(/MSIE/.test(navigator.userAgent)) tagifyStyle += ';zoom:1';
element = $(element);
$A(element.childNodes).each( function(child) {
if(child.nodeType==3) {
child.nodeValue.toArray().each( function(character) {
element.insertBefore(
Builder.node('span',{style: tagifyStyle},
character == ' ' ? String.fromCharCode(160) : character),
child);
});
Element.remove(child);
}
});
},
multiple: function(element, effect) {
var elements;
if(((typeof element == 'object') ||
(typeof element == 'function')) &&
(element.length))
elements = element;
else
elements = $(element).childNodes;
var options = Object.extend({
speed: 0.1,
delay: 0.0
}, arguments[2] || {});
$A(elements).each( function(element, index) {
new effect(element, Object.extend(options, { delay: options.delay + index * options.speed }));
});
}
};
var Effect2 = Effect; // deprecated
/* ------------- transitions ------------- */
Effect.Transitions = {}
Effect.Transitions.linear = function(pos) {
return pos;
}
Effect.Transitions.sinoidal = function(pos) {
return (-Math.cos(pos*Math.PI)/2) + 0.5;
}
Effect.Transitions.reverse = function(pos) {
return 1-pos;
}
Effect.Transitions.flicker = function(pos) {
return ((-Math.cos(pos*Math.PI)/4) + 0.75) + Math.random()/4;
}
Effect.Transitions.wobble = function(pos) {
return (-Math.cos(pos*Math.PI*(9*pos))/2) + 0.5;
}
Effect.Transitions.pulse = function(pos) {
return (Math.floor(pos*10) % 2 == 0 ?
(pos*10-Math.floor(pos*10)) : 1-(pos*10-Math.floor(pos*10)));
}
Effect.Transitions.none = function(pos) {
return 0;
}
Effect.Transitions.full = function(pos) {
return 1;
}
/* ------------- core effects ------------- */
Effect.Queue = {
effects: [],
_each: function(iterator) {
this.effects._each(iterator);
},
interval: null,
add: function(effect) {
var timestamp = new Date().getTime();
switch(effect.options.queue) {
case 'front':
// move unstarted effects after this effect
this.effects.findAll(function(e){ return e.state=='idle' }).each( function(e) {
e.startOn += effect.finishOn;
e.finishOn += effect.finishOn;
});
break;
case 'end':
// start effect after last queued effect has finished
timestamp = this.effects.pluck('finishOn').max() || timestamp;
break;
}
effect.startOn += timestamp;
effect.finishOn += timestamp;
this.effects.push(effect);
if(!this.interval)
this.interval = setInterval(this.loop.bind(this), 40);
},
remove: function(effect) {
this.effects = this.effects.reject(function(e) { return e==effect });
if(this.effects.length == 0) {
clearInterval(this.interval);
this.interval = null;
}
},
loop: function() {
var timePos = new Date().getTime();
this.effects.invoke('loop', timePos);
}
}
Object.extend(Effect.Queue, Enumerable);
Effect.Base = function() {};
Effect.Base.prototype = {
position: null,
setOptions: function(options) {
this.options = Object.extend({
transition: Effect.Transitions.sinoidal,
duration: 1.0, // seconds
fps: 25.0, // max. 25fps due to Effect.Queue implementation
sync: false, // true for combining
from: 0.0,
to: 1.0,
delay: 0.0,
queue: 'parallel'
}, options || {});
},
start: function(options) {
this.setOptions(options || {});
this.currentFrame = 0;
this.state = 'idle';
this.startOn = this.options.delay*1000;
this.finishOn = this.startOn + (this.options.duration*1000);
this.event('beforeStart');
if(!this.options.sync) Effect.Queue.add(this);
},
loop: function(timePos) {
if(timePos >= this.startOn) {
if(timePos >= this.finishOn) {
this.render(1.0);
this.cancel();
this.event('beforeFinish');
if(this.finish) this.finish();
this.event('afterFinish');
return;
}
var pos = (timePos - this.startOn) / (this.finishOn - this.startOn);
var frame = Math.round(pos * this.options.fps * this.options.duration);
if(frame > this.currentFrame) {
this.render(pos);
this.currentFrame = frame;
}
}
},
render: function(pos) {
if(this.state == 'idle') {
this.state = 'running';
this.event('beforeSetup');
if(this.setup) this.setup();
this.event('afterSetup');
}
if(this.options.transition) pos = this.options.transition(pos);
pos *= (this.options.to-this.options.from);
pos += this.options.from;
this.position = pos;
this.event('beforeUpdate');
if(this.update) this.update(pos);
this.event('afterUpdate');
},
cancel: function() {
if(!this.options.sync) Effect.Queue.remove(this);
this.state = 'finished';
},
event: function(eventName) {
if(this.options[eventName + 'Internal']) this.options[eventName + 'Internal'](this);
if(this.options[eventName]) this.options[eventName](this);
}
}
Effect.Parallel = Class.create();
Object.extend(Object.extend(Effect.Parallel.prototype, Effect.Base.prototype), {
initialize: function(effects) {
this.effects = effects || [];
this.start(arguments[1]);
},
update: function(position) {
this.effects.invoke('render', position);
},
finish: function(position) {
this.effects.each( function(effect) {
effect.render(1.0);
effect.cancel();
effect.event('beforeFinish');
if(effect.finish) effect.finish(position);
effect.event('afterFinish');
});
}
});
Effect.Opacity = Class.create();
Object.extend(Object.extend(Effect.Opacity.prototype, Effect.Base.prototype), {
initialize: function(element) {
this.element = $(element);
// make this work on IE on elements without 'layout'
if(/MSIE/.test(navigator.userAgent) && (!this.element.hasLayout))
Element.setStyle(this.element, {zoom: 1});
var options = Object.extend({
from: Element.getOpacity(this.element) || 0.0,
to: 1.0
}, arguments[1] || {});
this.start(options);
},
update: function(position) {
Element.setOpacity(this.element, position);
}
});
Effect.MoveBy = Class.create();
Object.extend(Object.extend(Effect.MoveBy.prototype, Effect.Base.prototype), {
initialize: function(element, toTop, toLeft) {
this.element = $(element);
this.toTop = toTop;
this.toLeft = toLeft;
this.start(arguments[3]);
},
setup: function() {
// Bug in Opera: Opera returns the "real" position of a static element or
// relative element that does not have top/left explicitly set.
// ==> Always set top and left for position relative elements in your stylesheets
// (to 0 if you do not need them)
Element.makePositioned(this.element);
this.originalTop = parseFloat(Element.getStyle(this.element,'top') || '0');
this.originalLeft = parseFloat(Element.getStyle(this.element,'left') || '0');
},
update: function(position) {
Element.setStyle(this.element, {
top: this.toTop * position + this.originalTop + 'px',
left: this.toLeft * position + this.originalLeft + 'px'
});
}
});
Effect.Scale = Class.create();
Object.extend(Object.extend(Effect.Scale.prototype, Effect.Base.prototype), {
initialize: function(element, percent) {
this.element = $(element)
var options = Object.extend({
scaleX: true,
scaleY: true,
scaleContent: true,
scaleFromCenter: false,
scaleMode: 'box', // 'box' or 'contents' or {} with provided values
scaleFrom: 100.0,
scaleTo: percent
}, arguments[2] || {});
this.start(options);
},
setup: function() {
this.restoreAfterFinish = this.options.restoreAfterFinish || false;
this.elementPositioning = Element.getStyle(this.element,'position');
this.originalStyle = {};
['top','left','width','height','fontSize'].each( function(k) {
this.originalStyle[k] = this.element.style[k];
}.bind(this));
this.originalTop = this.element.offsetTop;
this.originalLeft = this.element.offsetLeft;
var fontSize = Element.getStyle(this.element,'font-size') || '100%';
['em','px','%'].each( function(fontSizeType) {
if(fontSize.indexOf(fontSizeType)>0) {
this.fontSize = parseFloat(fontSize);
this.fontSizeType = fontSizeType;
}
}.bind(this));
this.factor = (this.options.scaleTo - this.options.scaleFrom)/100;
this.dims = null;
if(this.options.scaleMode=='box')
this.dims = [this.element.clientHeight, this.element.clientWidth];
if(/^content/.test(this.options.scaleMode))
this.dims = [this.element.scrollHeight, this.element.scrollWidth];
if(!this.dims)
this.dims = [this.options.scaleMode.originalHeight,
this.options.scaleMode.originalWidth];
},
update: function(position) {
var currentScale = (this.options.scaleFrom/100.0) + (this.factor * position);
if(this.options.scaleContent && this.fontSize)
Element.setStyle(this.element, {fontSize: this.fontSize * currentScale + this.fontSizeType });
this.setDimensions(this.dims[0] * currentScale, this.dims[1] * currentScale);
},
finish: function(position) {
if (this.restoreAfterFinish) Element.setStyle(this.element, this.originalStyle);
},
setDimensions: function(height, width) {
var d = {};
if(this.options.scaleX) d.width = width + 'px';
if(this.options.scaleY) d.height = height + 'px';
if(this.options.scaleFromCenter) {
var topd = (height - this.dims[0])/2;
var leftd = (width - this.dims[1])/2;
if(this.elementPositioning == 'absolute') {
if(this.options.scaleY) d.top = this.originalTop-topd + 'px';
if(this.options.scaleX) d.left = this.originalLeft-leftd + 'px';
} else {
if(this.options.scaleY) d.top = -topd + 'px';
if(this.options.scaleX) d.left = -leftd + 'px';
}
}
Element.setStyle(this.element, d);
}
});
Effect.Highlight = Class.create();
Object.extend(Object.extend(Effect.Highlight.prototype, Effect.Base.prototype), {
initialize: function(element) {
this.element = $(element);
var options = Object.extend({ startcolor: '#ffff99' }, arguments[1] || {});
this.start(options);
},
setup: function() {
// Prevent executing on elements not in the layout flow
if(Element.getStyle(this.element, 'display')=='none') { this.cancel(); return; }
// Disable background image during the effect
this.oldStyle = {
backgroundImage: Element.getStyle(this.element, 'background-image') };
Element.setStyle(this.element, {backgroundImage: 'none'});
if(!this.options.endcolor)
this.options.endcolor = Element.getStyle(this.element, 'background-color').parseColor('#ffffff');
if(!this.options.restorecolor)
this.options.restorecolor = Element.getStyle(this.element, 'background-color');
// init color calculations
this._base = $R(0,2).map(function(i){ return parseInt(this.options.startcolor.slice(i*2+1,i*2+3),16) }.bind(this));
this._delta = $R(0,2).map(function(i){ return parseInt(this.options.endcolor.slice(i*2+1,i*2+3),16)-this._base[i] }.bind(this));
},
update: function(position) {
Element.setStyle(this.element,{backgroundColor: $R(0,2).inject('#',function(m,v,i){
return m+(Math.round(this._base[i]+(this._delta[i]*position)).toColorPart()); }.bind(this)) });
},
finish: function() {
Element.setStyle(this.element, Object.extend(this.oldStyle, {
backgroundColor: this.options.restorecolor
}));
}
});
Effect.ScrollTo = Class.create();
Object.extend(Object.extend(Effect.ScrollTo.prototype, Effect.Base.prototype), {
initialize: function(element) {
this.element = $(element);
this.start(arguments[1] || {});
},
setup: function() {
Position.prepare();
var offsets = Position.cumulativeOffset(this.element);
var max = window.innerHeight ?
window.height - window.innerHeight :
document.body.scrollHeight -
(document.documentElement.clientHeight ?
document.documentElement.clientHeight : document.body.clientHeight);
this.scrollStart = Position.deltaY;
this.delta = (offsets[1] > max ? max : offsets[1]) - this.scrollStart;
},
update: function(position) {
Position.prepare();
window.scrollTo(Position.deltaX,
this.scrollStart + (position*this.delta));
}
});
/* ------------- combination effects ------------- */
Effect.Fade = function(element) {
var oldOpacity = Element.getInlineOpacity(element);
var options = Object.extend({
from: Element.getOpacity(element) || 1.0,
to: 0.0,
afterFinishInternal: function(effect) { with(Element) {
if(effect.options.to!=0) return;
hide(effect.element);
setStyle(effect.element, {opacity: oldOpacity}); }}
}, arguments[1] || {});
return new Effect.Opacity(element,options);
}
Effect.Appear = function(element) {
var options = Object.extend({
from: (Element.getStyle(element, 'display') == 'none' ? 0.0 : Element.getOpacity(element) || 0.0),
to: 1.0,
beforeSetup: function(effect) { with(Element) {
setOpacity(effect.element, effect.options.from);
show(effect.element); }}
}, arguments[1] || {});
return new Effect.Opacity(element,options);
}
Effect.Puff = function(element) {
element = $(element);
var oldStyle = { opacity: Element.getInlineOpacity(element), position: Element.getStyle(element, 'position') };
return new Effect.Parallel(
[ new Effect.Scale(element, 200,
{ sync: true, scaleFromCenter: true, scaleContent: true, restoreAfterFinish: true }),
new Effect.Opacity(element, { sync: true, to: 0.0 } ) ],
Object.extend({ duration: 1.0,
beforeSetupInternal: function(effect) { with(Element) {
setStyle(effect.effects[0].element, {position: 'absolute'}); }},
afterFinishInternal: function(effect) { with(Element) {
hide(effect.effects[0].element);
setStyle(effect.effects[0].element, oldStyle); }}
}, arguments[1] || {})
);
}
Effect.BlindUp = function(element) {
element = $(element);
Element.makeClipping(element);
return new Effect.Scale(element, 0,
Object.extend({ scaleContent: false,
scaleX: false,
restoreAfterFinish: true,
afterFinishInternal: function(effect) { with(Element) {
[hide, undoClipping].call(effect.element); }}
}, arguments[1] || {})
);
}
Effect.BlindDown = function(element) {
element = $(element);
var oldHeight = Element.getStyle(element, 'height');
var elementDimensions = Element.getDimensions(element);
return new Effect.Scale(element, 100,
Object.extend({ scaleContent: false,
scaleX: false,
scaleFrom: 0,
scaleMode: {originalHeight: elementDimensions.height, originalWidth: elementDimensions.width},
restoreAfterFinish: true,
afterSetup: function(effect) { with(Element) {
makeClipping(effect.element);
setStyle(effect.element, {height: '0px'});
show(effect.element);
}},
afterFinishInternal: function(effect) { with(Element) {
undoClipping(effect.element);
setStyle(effect.element, {height: oldHeight});
}}
}, arguments[1] || {})
);
}
Effect.SwitchOff = function(element) {
element = $(element);
var oldOpacity = Element.getInlineOpacity(element);
return new Effect.Appear(element, {
duration: 0.4,
from: 0,
transition: Effect.Transitions.flicker,
afterFinishInternal: function(effect) {
new Effect.Scale(effect.element, 1, {
duration: 0.3, scaleFromCenter: true,
scaleX: false, scaleContent: false, restoreAfterFinish: true,
beforeSetup: function(effect) { with(Element) {
[makePositioned,makeClipping].call(effect.element);
}},
afterFinishInternal: function(effect) { with(Element) {
[hide,undoClipping,undoPositioned].call(effect.element);
setStyle(effect.element, {opacity: oldOpacity});
}}
})
}
});
}
Effect.DropOut = function(element) {
element = $(element);
var oldStyle = {
top: Element.getStyle(element, 'top'),
left: Element.getStyle(element, 'left'),
opacity: Element.getInlineOpacity(element) };
return new Effect.Parallel(
[ new Effect.MoveBy(element, 100, 0, { sync: true }),
new Effect.Opacity(element, { sync: true, to: 0.0 }) ],
Object.extend(
{ duration: 0.5,
beforeSetup: function(effect) { with(Element) {
makePositioned(effect.effects[0].element); }},
afterFinishInternal: function(effect) { with(Element) {
[hide, undoPositioned].call(effect.effects[0].element);
setStyle(effect.effects[0].element, oldStyle); }}
}, arguments[1] || {}));
}
Effect.Shake = function(element) {
element = $(element);
var oldStyle = {
top: Element.getStyle(element, 'top'),
left: Element.getStyle(element, 'left') };
return new Effect.MoveBy(element, 0, 20,
{ duration: 0.05, afterFinishInternal: function(effect) {
new Effect.MoveBy(effect.element, 0, -40,
{ duration: 0.1, afterFinishInternal: function(effect) {
new Effect.MoveBy(effect.element, 0, 40,
{ duration: 0.1, afterFinishInternal: function(effect) {
new Effect.MoveBy(effect.element, 0, -40,
{ duration: 0.1, afterFinishInternal: function(effect) {
new Effect.MoveBy(effect.element, 0, 40,
{ duration: 0.1, afterFinishInternal: function(effect) {
new Effect.MoveBy(effect.element, 0, -20,
{ duration: 0.05, afterFinishInternal: function(effect) { with(Element) {
undoPositioned(effect.element);
setStyle(effect.element, oldStyle);
}}}) }}) }}) }}) }}) }});
}
Effect.SlideDown = function(element) {
element = $(element);
Element.cleanWhitespace(element);
// SlideDown need to have the content of the element wrapped in a container element with fixed height!
var oldInnerBottom = Element.getStyle(element.firstChild, 'bottom');
var elementDimensions = Element.getDimensions(element);
return new Effect.Scale(element, 100, Object.extend({
scaleContent: false,
scaleX: false,
scaleFrom: 0,
scaleMode: {originalHeight: elementDimensions.height, originalWidth: elementDimensions.width},
restoreAfterFinish: true,
afterSetup: function(effect) { with(Element) {
makePositioned(effect.element.firstChild);
if(window.opera) setStyle(effect.element, {top: ''});
makeClipping(effect.element);
setStyle(effect.element, {height: '0px'});
show(element); }},
afterUpdateInternal: function(effect) { with(Element) {
setStyle(effect.element.firstChild, {bottom:
(effect.dims[0] - effect.element.clientHeight) + 'px' }); }},
afterFinishInternal: function(effect) { with(Element) {
undoClipping(effect.element);
undoPositioned(effect.element.firstChild);
setStyle(effect.element.firstChild, {bottom: oldInnerBottom}); }}
}, arguments[1] || {})
);
}
Effect.SlideUp = function(element) {
element = $(element);
Element.cleanWhitespace(element);
var oldInnerBottom = Element.getStyle(element.firstChild, 'bottom');
return new Effect.Scale(element, 0,
Object.extend({ scaleContent: false,
scaleX: false,
scaleMode: 'box',
scaleFrom: 100,
restoreAfterFinish: true,
beforeStartInternal: function(effect) { with(Element) {
makePositioned(effect.element.firstChild);
if(window.opera) setStyle(effect.element, {top: ''});
makeClipping(effect.element);
show(element); }},
afterUpdateInternal: function(effect) { with(Element) {
setStyle(effect.element.firstChild, {bottom:
(effect.dims[0] - effect.element.clientHeight) + 'px' }); }},
afterFinishInternal: function(effect) { with(Element) {
[hide, undoClipping].call(effect.element);
undoPositioned(effect.element.firstChild);
setStyle(effect.element.firstChild, {bottom: oldInnerBottom}); }}
}, arguments[1] || {})
);
}
// Bug in opera makes the TD containing this element expand for a instance after finish
Effect.Squish = function(element) {
return new Effect.Scale(element, window.opera ? 1 : 0,
{ restoreAfterFinish: true,
beforeSetup: function(effect) { with(Element) {
makeClipping(effect.element); }},
afterFinishInternal: function(effect) { with(Element) {
hide(effect.element);
undoClipping(effect.element); }}
});
}
Effect.Grow = function(element) {
element = $(element);
var options = Object.extend({
direction: 'center',
moveTransistion: Effect.Transitions.sinoidal,
scaleTransition: Effect.Transitions.sinoidal,
opacityTransition: Effect.Transitions.full
}, arguments[1] || {});
var oldStyle = {
top: element.style.top,
left: element.style.left,
height: element.style.height,
width: element.style.width,
opacity: Element.getInlineOpacity(element) };
var dims = Element.getDimensions(element);
var initialMoveX, initialMoveY;
var moveX, moveY;
switch (options.direction) {
case 'top-left':
initialMoveX = initialMoveY = moveX = moveY = 0;
break;
case 'top-right':
initialMoveX = dims.width;
initialMoveY = moveY = 0;
moveX = -dims.width;
break;
case 'bottom-left':
initialMoveX = moveX = 0;
initialMoveY = dims.height;
moveY = -dims.height;
break;
case 'bottom-right':
initialMoveX = dims.width;
initialMoveY = dims.height;
moveX = -dims.width;
moveY = -dims.height;
break;
case 'center':
initialMoveX = dims.width / 2;
initialMoveY = dims.height / 2;
moveX = -dims.width / 2;
moveY = -dims.height / 2;
break;
}
return new Effect.MoveBy(element, initialMoveY, initialMoveX, {
duration: 0.01,
beforeSetup: function(effect) { with(Element) {
hide(effect.element);
makeClipping(effect.element);
makePositioned(effect.element);
}},
afterFinishInternal: function(effect) {
new Effect.Parallel(
[ new Effect.Opacity(effect.element, { sync: true, to: 1.0, from: 0.0, transition: options.opacityTransition }),
new Effect.MoveBy(effect.element, moveY, moveX, { sync: true, transition: options.moveTransition }),
new Effect.Scale(effect.element, 100, {
scaleMode: { originalHeight: dims.height, originalWidth: dims.width },
sync: true, scaleFrom: window.opera ? 1 : 0, transition: options.scaleTransition, restoreAfterFinish: true})
], Object.extend({
beforeSetup: function(effect) { with(Element) {
setStyle(effect.effects[0].element, {height: '0px'});
show(effect.effects[0].element); }},
afterFinishInternal: function(effect) { with(Element) {
[undoClipping, undoPositioned].call(effect.effects[0].element);
setStyle(effect.effects[0].element, oldStyle); }}
}, options)
)
}
});
}
Effect.Shrink = function(element) {
element = $(element);
var options = Object.extend({
direction: 'center',
moveTransistion: Effect.Transitions.sinoidal,
scaleTransition: Effect.Transitions.sinoidal,
opacityTransition: Effect.Transitions.none
}, arguments[1] || {});
var oldStyle = {
top: element.style.top,
left: element.style.left,
height: element.style.height,
width: element.style.width,
opacity: Element.getInlineOpacity(element) };
var dims = Element.getDimensions(element);
var moveX, moveY;
switch (options.direction) {
case 'top-left':
moveX = moveY = 0;
break;
case 'top-right':
moveX = dims.width;
moveY = 0;
break;
case 'bottom-left':
moveX = 0;
moveY = dims.height;
break;
case 'bottom-right':
moveX = dims.width;
moveY = dims.height;
break;
case 'center':
moveX = dims.width / 2;
moveY = dims.height / 2;
break;
}
return new Effect.Parallel(
[ new Effect.Opacity(element, { sync: true, to: 0.0, from: 1.0, transition: options.opacityTransition }),
new Effect.Scale(element, window.opera ? 1 : 0, { sync: true, transition: options.scaleTransition, restoreAfterFinish: true}),
new Effect.MoveBy(element, moveY, moveX, { sync: true, transition: options.moveTransition })
], Object.extend({
beforeStartInternal: function(effect) { with(Element) {
[makePositioned, makeClipping].call(effect.effects[0].element) }},
afterFinishInternal: function(effect) { with(Element) {
[hide, undoClipping, undoPositioned].call(effect.effects[0].element);
setStyle(effect.effects[0].element, oldStyle); }}
}, options)
);
}
Effect.Pulsate = function(element) {
element = $(element);
var options = arguments[1] || {};
var oldOpacity = Element.getInlineOpacity(element);
var transition = options.transition || Effect.Transitions.sinoidal;
var reverser = function(pos){ return transition(1-Effect.Transitions.pulse(pos)) };
reverser.bind(transition);
return new Effect.Opacity(element,
Object.extend(Object.extend({ duration: 3.0, from: 0,
afterFinishInternal: function(effect) { Element.setStyle(el, {opacity: oldOpacity}); }
}, options), {transition: reverser}));
}
Effect.Fold = function(element) {
element = $(element);
var oldStyle = {
top: element.style.top,
left: element.style.left,
width: element.style.width,
height: element.style.height };
Element.makeClipping(element);
return new Effect.Scale(element, 5, Object.extend({
scaleContent: false,
scaleX: false,
afterFinishInternal: function(effect) {
new Effect.Scale(element, 1, {
scaleContent: false,
scaleY: false,
afterFinishInternal: function(effect) { with(Element) {
[hide, undoClipping].call(effect.element);
setStyle(effect.element, oldStyle);
}} });
}}, arguments[1] || {}));
}

1758
www/extras/js/sau/prototype.js vendored Normal file

File diff suppressed because it is too large Load diff

View file

@ -0,0 +1,48 @@
// Copyright (c) 2005 Thomas Fuchs (http://script.aculo.us, http://mir.aculo.us)
//
// Permission is hereby granted, free of charge, to any person obtaining
// a copy of this software and associated documentation files (the
// "Software"), to deal in the Software without restriction, including
// without limitation the rights to use, copy, modify, merge, publish,
// distribute, sublicense, and/or sell copies of the Software, and to
// permit persons to whom the Software is furnished to do so, subject to
// the following conditions:
//
// The above copyright notice and this permission notice shall be
// included in all copies or substantial portions of the Software.
//
// THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
// EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
// MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
// NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE
// LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION
// OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION
// WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
var Scriptaculous = {
Version: '1.5_rc5',
require: function(libraryName) {
// inserting via DOM fails in Safari 2.0, so brute force approach
document.write('<script type="text/javascript" src="'+libraryName+'"></script>');
},
load: function() {
if((typeof Prototype=='undefined') ||
parseFloat(Prototype.Version.split(".")[0] + "." +
Prototype.Version.split(".")[1]) < 1.4)
throw("script.aculo.us requires the Prototype JavaScript framework >= 1.4.0");
var scriptTags = document.getElementsByTagName("script");
for(var i=0;i<scriptTags.length;i++) {
if(scriptTags[i].src && scriptTags[i].src.match(/scriptaculous\.js(\?.*)?$/)) {
var path = scriptTags[i].src.replace(/scriptaculous\.js(\?.*)?$/,'');
this.require(path + 'builder.js');
this.require(path + 'effects.js');
this.require(path + 'dragdrop.js');
this.require(path + 'controls.js');
this.require(path + 'slider.js');
break;
}
}
}
}
Scriptaculous.load();

275
www/extras/js/sau/slider.js Normal file
View file

@ -0,0 +1,275 @@
// Copyright (c) 2005 Marty Haught, Thomas Fuchs
//
// See http://script.aculo.us for more info
//
// Permission is hereby granted, free of charge, to any person obtaining
// a copy of this software and associated documentation files (the
// "Software"), to deal in the Software without restriction, including
// without limitation the rights to use, copy, modify, merge, publish,
// distribute, sublicense, and/or sell copies of the Software, and to
// permit persons to whom the Software is furnished to do so, subject to
// the following conditions:
//
// The above copyright notice and this permission notice shall be
// included in all copies or substantial portions of the Software.
//
// THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
// EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
// MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
// NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE
// LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION
// OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION
// WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
if(!Control) var Control = {};
Control.Slider = Class.create();
// options:
// axis: 'vertical', or 'horizontal' (default)
//
// callbacks:
// onChange(value)
// onSlide(value)
Control.Slider.prototype = {
initialize: function(handle, track, options) {
var slider = this;
if(handle instanceof Array) {
this.handles = handle.collect( function(e) { return $(e) });
} else {
this.handles = [$(handle)];
}
this.track = $(track);
this.options = options || {};
this.axis = this.options.axis || 'horizontal';
this.increment = this.options.increment || 1;
this.step = parseInt(this.options.step || '1');
this.range = this.options.range || $R(0,1);
this.value = 0; // assure backwards compat
this.values = this.handles.map( function() { return 0 });
this.spans = this.options.spans ? this.options.spans.map(function(s){ return $(s) }) : false;
this.options.startSpan = $(this.options.startSpan || null);
this.options.endSpan = $(this.options.endSpan || null);
this.restricted = this.options.restricted || false;
this.maximum = this.options.maximum || this.range.end;
this.minimum = this.options.minimum || this.range.start;
// Will be used to align the handle onto the track, if necessary
this.alignX = parseInt(this.options.alignX || '0');
this.alignY = parseInt(this.options.alignY || '0');
this.trackLength = this.maximumOffset() - this.minimumOffset();
this.active = false;
this.dragging = false;
this.disabled = false;
if(this.options.disabled) this.setDisabled();
// Allowed values array
this.allowedValues = this.options.values ? this.options.values.sortBy(Prototype.K) : false;
if(this.allowedValues) {
this.minimum = this.allowedValues.min();
this.maximum = this.allowedValues.max();
}
this.eventMouseDown = this.startDrag.bindAsEventListener(this);
this.eventMouseUp = this.endDrag.bindAsEventListener(this);
this.eventMouseMove = this.update.bindAsEventListener(this);
// Initialize handles in reverse (make sure first handle is active)
this.handles.each( function(h,i) {
i = slider.handles.length-1-i;
slider.setValue(parseFloat(
(slider.options.sliderValue instanceof Array ?
slider.options.sliderValue[i] : slider.options.sliderValue) ||
slider.range.start), i);
Element.makePositioned(h); // fix IE
Event.observe(h, "mousedown", slider.eventMouseDown);
});
Event.observe(this.track, "mousedown", this.eventMouseDown);
Event.observe(document, "mouseup", this.eventMouseUp);
Event.observe(document, "mousemove", this.eventMouseMove);
this.initialized = true;
},
dispose: function() {
var slider = this;
Event.stopObserving(this.track, "mousedown", this.eventMouseDown);
Event.stopObserving(document, "mouseup", this.eventMouseUp);
Event.stopObserving(document, "mousemove", this.eventMouseMove);
this.handles.each( function(h) {
Event.stopObserving(h, "mousedown", slider.eventMouseDown);
});
},
setDisabled: function(){
this.disabled = true;
},
setEnabled: function(){
this.disabled = false;
},
getNearestValue: function(value){
if(this.allowedValues){
if(value >= this.allowedValues.max()) return(this.allowedValues.max());
if(value <= this.allowedValues.min()) return(this.allowedValues.min());
var offset = Math.abs(this.allowedValues[0] - value);
var newValue = this.allowedValues[0];
this.allowedValues.each( function(v) {
var currentOffset = Math.abs(v - value);
if(currentOffset <= offset){
newValue = v;
offset = currentOffset;
}
});
return newValue;
}
if(value > this.range.end) return this.range.end;
if(value < this.range.start) return this.range.start;
return value;
},
setValue: function(sliderValue, handleIdx){
if(!this.active) {
this.activeHandle = this.handles[handleIdx];
this.activeHandleIdx = handleIdx;
this.updateStyles();
}
handleIdx = handleIdx || this.activeHandleIdx || 0;
if(this.initialized && this.restricted) {
if((handleIdx>0) && (sliderValue<this.values[handleIdx-1]))
sliderValue = this.values[handleIdx-1];
if((handleIdx < (this.handles.length-1)) && (sliderValue>this.values[handleIdx+1]))
sliderValue = this.values[handleIdx+1];
}
sliderValue = this.getNearestValue(sliderValue);
this.values[handleIdx] = sliderValue;
this.value = this.values[0]; // assure backwards compat
this.handles[handleIdx].style[this.isVertical() ? 'top' : 'left'] =
this.translateToPx(sliderValue);
this.drawSpans();
if(!this.dragging || !this.event) this.updateFinished();
},
setValueBy: function(delta, handleIdx) {
this.setValue(this.values[handleIdx || this.activeHandleIdx || 0] + delta,
handleIdx || this.activeHandleIdx || 0);
},
translateToPx: function(value) {
return Math.round((this.trackLength / (this.range.end - this.range.start)) * (value - this.range.start)) + "px";
},
translateToValue: function(offset) {
return ((offset/this.trackLength) * (this.range.end - this.range.start)) + this.range.start;
},
getRange: function(range) {
var v = this.values.sortBy(Prototype.K);
range = range || 0;
return $R(v[range],v[range+1]);
},
minimumOffset: function(){
return(this.isVertical() ? this.alignY : this.alignX);
},
maximumOffset: function(){
return(this.isVertical() ?
this.track.offsetHeight - this.alignY : this.track.offsetWidth - this.alignX);
},
isVertical: function(){
return (this.axis == 'vertical');
},
drawSpans: function() {
var slider = this;
if(this.spans)
$R(0, this.spans.length-1).each(function(r) { slider.setSpan(slider.spans[r], slider.getRange(r)) });
if(this.options.startSpan)
this.setSpan(this.options.startSpan,
$R(0, this.values.length>1 ? this.getRange(0).min() : this.value ));
if(this.options.endSpan)
this.setSpan(this.options.endSpan,
$R(this.values.length>1 ? this.getRange(this.spans.length-1).max() : this.value, this.maximum));
},
setSpan: function(span, range) {
if(this.isVertical()) {
span.style.top = this.translateToPx(range.start);
span.style.height = this.translateToPx(range.end - range.start);
} else {
span.style.left = this.translateToPx(range.start);
span.style.width = this.translateToPx(range.end - range.start);
}
},
updateStyles: function() {
this.handles.each( function(h){ Element.removeClassName(h, 'selected') });
Element.addClassName(this.activeHandle, 'selected');
},
startDrag: function(event) {
if(Event.isLeftClick(event)) {
if(!this.disabled){
this.active = true;
var handle = Event.element(event);
var pointer = [Event.pointerX(event), Event.pointerY(event)];
if(handle==this.track) {
var offsets = Position.cumulativeOffset(this.track);
this.event = event;
this.setValue(this.translateToValue(
this.isVertical() ? pointer[1]-offsets[1] : pointer[0]-offsets[0]
));
} else {
// find the handle (prevents issues with Safari)
while((this.handles.indexOf(handle) == -1) && handle.parentNode)
handle = handle.parentNode;
this.activeHandle = handle;
this.activeHandleIdx = this.handles.indexOf(this.activeHandle);
this.updateStyles();
var offsets = Position.cumulativeOffset(this.activeHandle);
this.offsetX = (pointer[0] - offsets[0]);
this.offsetY = (pointer[1] - offsets[1]);
}
}
Event.stop(event);
}
},
update: function(event) {
if(this.active) {
if(!this.dragging) this.dragging = true;
this.draw(event);
// fix AppleWebKit rendering
if(navigator.appVersion.indexOf('AppleWebKit')>0) window.scrollBy(0,0);
Event.stop(event);
}
},
draw: function(event) {
var pointer = [Event.pointerX(event), Event.pointerY(event)];
var offsets = Position.cumulativeOffset(this.track);
pointer[0] -= this.offsetX + offsets[0];
pointer[1] -= this.offsetY + offsets[1];
this.event = event;
this.setValue(this.translateToValue( this.isVertical() ? pointer[1] : pointer[0] ));
if(this.initialized && this.options.onSlide) this.options.onSlide(this.values.length>1 ? this.values : this.value, this);
},
endDrag: function(event) {
if(this.active && this.dragging) {
this.finishDrag(event, true);
Event.stop(event);
}
this.active = false;
this.dragging = false;
},
finishDrag: function(event, success) {
this.active = false;
this.dragging = false;
this.updateFinished();
},
updateFinished: function() {
if(this.initialized && this.options.onChange)
this.options.onChange(this.values.length>1 ? this.values : this.value, this);
this.event = null;
}
}

View file

@ -0,0 +1,363 @@
// Copyright (c) 2005 Thomas Fuchs (http://script.aculo.us, http://mir.aculo.us)
// (c) 2005 Jon Tirsen (http://www.tirsen.com)
// (c) 2005 Michael Schuerig (http://www.schuerig.de/michael/)
//
// See scriptaculous.js for full license.
// experimental, Firefox-only
Event.simulateMouse = function(element, eventName) {
var options = Object.extend({
pointerX: 0,
pointerY: 0,
buttons: 0
}, arguments[2] || {});
var oEvent = document.createEvent("MouseEvents");
oEvent.initMouseEvent(eventName, true, true, document.defaultView,
options.buttons, options.pointerX, options.pointerY, options.pointerX, options.pointerY,
false, false, false, false, 0, $(element));
if(this.mark) Element.remove(this.mark);
this.mark = document.createElement('div');
this.mark.appendChild(document.createTextNode(" "));
document.body.appendChild(this.mark);
this.mark.style.position = 'absolute';
this.mark.style.top = options.pointerY + "px";
this.mark.style.left = options.pointerX + "px";
this.mark.style.width = "5px";
this.mark.style.height = "5px;";
this.mark.style.borderTop = "1px solid red;"
this.mark.style.borderLeft = "1px solid red;"
if(this.step)
alert('['+new Date().getTime().toString()+'] '+eventName+'/'+Test.Unit.inspect(options));
$(element).dispatchEvent(oEvent);
};
// Note: Due to a fix in Firefox 1.0.5/6 that probably fixed "too much", this doesn't work in 1.0.6 or DP2.
// You need to downgrade to 1.0.4 for now to get this working
// See https://bugzilla.mozilla.org/show_bug.cgi?id=289940 for the fix that fixed too much
Event.simulateKey = function(element, eventName) {
var options = Object.extend({
ctrlKey: false,
altKey: false,
shiftKey: false,
metaKey: false,
keyCode: 0,
charCode: 0
}, arguments[2] || {});
var oEvent = document.createEvent("KeyEvents");
oEvent.initKeyEvent(eventName, true, true, window,
options.ctrlKey, options.altKey, options.shiftKey, options.metaKey,
options.keyCode, options.charCode );
$(element).dispatchEvent(oEvent);
};
Event.simulateKeys = function(element, command) {
for(var i=0; i<command.length; i++) {
Event.simulateKey(element,'keypress',{charCode:command.charCodeAt(i)});
}
};
var Test = {}
Test.Unit = {};
// security exception workaround
Test.Unit.inspect = function(obj) {
var info = [];
if(typeof obj=="string" ||
typeof obj=="number") {
return obj;
} else {
for(property in obj)
if(typeof obj[property]!="function")
info.push(property + ' => ' +
(typeof obj[property] == "string" ?
'"' + obj[property] + '"' :
obj[property]));
}
return ("'" + obj + "' #" + typeof obj +
": {" + info.join(", ") + "}");
}
Test.Unit.Logger = Class.create();
Test.Unit.Logger.prototype = {
initialize: function(log) {
this.log = $(log);
if (this.log) {
this._createLogTable();
}
},
start: function(testName) {
if (!this.log) return;
this.testName = testName;
this.lastLogLine = document.createElement('tr');
this.statusCell = document.createElement('td');
this.nameCell = document.createElement('td');
this.nameCell.appendChild(document.createTextNode(testName));
this.messageCell = document.createElement('td');
this.lastLogLine.appendChild(this.statusCell);
this.lastLogLine.appendChild(this.nameCell);
this.lastLogLine.appendChild(this.messageCell);
this.loglines.appendChild(this.lastLogLine);
},
finish: function(status, summary) {
if (!this.log) return;
this.lastLogLine.className = status;
this.statusCell.innerHTML = status;
this.messageCell.innerHTML = this._toHTML(summary);
},
message: function(message) {
if (!this.log) return;
this.messageCell.innerHTML = this._toHTML(message);
},
summary: function(summary) {
if (!this.log) return;
this.logsummary.innerHTML = this._toHTML(summary);
},
_createLogTable: function() {
this.log.innerHTML =
'<div id="logsummary"></div>' +
'<table id="logtable">' +
'<thead><tr><th>Status</th><th>Test</th><th>Message</th></tr></thead>' +
'<tbody id="loglines"></tbody>' +
'</table>';
this.logsummary = $('logsummary')
this.loglines = $('loglines');
},
_toHTML: function(txt) {
return txt.escapeHTML().replace(/\n/g,"<br/>");
}
}
Test.Unit.Runner = Class.create();
Test.Unit.Runner.prototype = {
initialize: function(testcases) {
this.options = Object.extend({
testLog: 'testlog'
}, arguments[1] || {});
this.options.resultsURL = this.parseResultsURLQueryParameter();
if (this.options.testLog) {
this.options.testLog = $(this.options.testLog) || null;
}
if(this.options.tests) {
this.tests = [];
for(var i = 0; i < this.options.tests.length; i++) {
if(/^test/.test(this.options.tests[i])) {
this.tests.push(new Test.Unit.Testcase(this.options.tests[i], testcases[this.options.tests[i]], testcases["setup"], testcases["teardown"]));
}
}
} else {
if (this.options.test) {
this.tests = [new Test.Unit.Testcase(this.options.test, testcases[this.options.test], testcases["setup"], testcases["teardown"])];
} else {
this.tests = [];
for(var testcase in testcases) {
if(/^test/.test(testcase)) {
this.tests.push(new Test.Unit.Testcase(testcase, testcases[testcase], testcases["setup"], testcases["teardown"]));
}
}
}
}
this.currentTest = 0;
this.logger = new Test.Unit.Logger(this.options.testLog);
setTimeout(this.runTests.bind(this), 1000);
},
parseResultsURLQueryParameter: function() {
return window.location.search.parseQuery()["resultsURL"];
},
// Returns:
// "ERROR" if there was an error,
// "FAILURE" if there was a failure, or
// "SUCCESS" if there was neither
getResult: function() {
var hasFailure = false;
for(var i=0;i<this.tests.length;i++) {
if (this.tests[i].errors > 0) {
return "ERROR";
}
if (this.tests[i].failures > 0) {
hasFailure = true;
}
}
if (hasFailure) {
return "FAILURE";
} else {
return "SUCCESS";
}
},
postResults: function() {
if (this.options.resultsURL) {
new Ajax.Request(this.options.resultsURL,
{ method: 'get', parameters: 'result=' + this.getResult(), asynchronous: false });
}
},
runTests: function() {
var test = this.tests[this.currentTest];
if (!test) {
// finished!
this.postResults();
this.logger.summary(this.summary());
return;
}
if(!test.isWaiting) {
this.logger.start(test.name);
}
test.run();
if(test.isWaiting) {
this.logger.message("Waiting for " + test.timeToWait + "ms");
setTimeout(this.runTests.bind(this), test.timeToWait || 1000);
} else {
this.logger.finish(test.status(), test.summary());
this.currentTest++;
// tail recursive, hopefully the browser will skip the stackframe
this.runTests();
}
},
summary: function() {
var assertions = 0;
var failures = 0;
var errors = 0;
var messages = [];
for(var i=0;i<this.tests.length;i++) {
assertions += this.tests[i].assertions;
failures += this.tests[i].failures;
errors += this.tests[i].errors;
}
return (
this.tests.length + " tests, " +
assertions + " assertions, " +
failures + " failures, " +
errors + " errors");
}
}
Test.Unit.Assertions = Class.create();
Test.Unit.Assertions.prototype = {
initialize: function() {
this.assertions = 0;
this.failures = 0;
this.errors = 0;
this.messages = [];
},
summary: function() {
return (
this.assertions + " assertions, " +
this.failures + " failures, " +
this.errors + " errors" + "\n" +
this.messages.join("\n"));
},
pass: function() {
this.assertions++;
},
fail: function(message) {
this.failures++;
this.messages.push("Failure: " + message);
},
error: function(error) {
this.errors++;
this.messages.push(error.name + ": "+ error.message + "(" + Test.Unit.inspect(error) +")");
},
status: function() {
if (this.failures > 0) return 'failed';
if (this.errors > 0) return 'error';
return 'passed';
},
assert: function(expression) {
var message = arguments[1] || 'assert: got "' + Test.Unit.inspect(expression) + '"';
try { expression ? this.pass() :
this.fail(message); }
catch(e) { this.error(e); }
},
assertEqual: function(expected, actual) {
var message = arguments[2] || "assertEqual";
try { (expected == actual) ? this.pass() :
this.fail(message + ': expected "' + Test.Unit.inspect(expected) +
'", actual "' + Test.Unit.inspect(actual) + '"'); }
catch(e) { this.error(e); }
},
assertNotEqual: function(expected, actual) {
var message = arguments[2] || "assertNotEqual";
try { (expected != actual) ? this.pass() :
this.fail(message + ': got "' + Test.Unit.inspect(actual) + '"'); }
catch(e) { this.error(e); }
},
assertNull: function(obj) {
var message = arguments[1] || 'assertNull'
try { (obj==null) ? this.pass() :
this.fail(message + ': got "' + Test.Unit.inspect(obj) + '"'); }
catch(e) { this.error(e); }
},
assertHidden: function(element) {
var message = arguments[1] || 'assertHidden';
this.assertEqual("none", element.style.display, message);
},
assertNotNull: function(object) {
var message = arguments[1] || 'assertNotNull';
this.assert(object != null, message);
},
assertInstanceOf: function(expected, actual) {
var message = arguments[2] || 'assertInstanceOf';
try {
(actual instanceof expected) ? this.pass() :
this.fail(message + ": object was not an instance of the expected type"); }
catch(e) { this.error(e); }
},
assertNotInstanceOf: function(expected, actual) {
var message = arguments[2] || 'assertNotInstanceOf';
try {
!(actual instanceof expected) ? this.pass() :
this.fail(message + ": object was an instance of the not expected type"); }
catch(e) { this.error(e); }
},
_isVisible: function(element) {
element = $(element);
if(!element.parentNode) return true;
this.assertNotNull(element);
if(element.style && Element.getStyle(element, 'display') == 'none')
return false;
return this._isVisible(element.parentNode);
},
assertNotVisible: function(element) {
this.assert(!this._isVisible(element), Test.Unit.inspect(element) + " was not hidden and didn't have a hidden parent either. " + ("" || arguments[1]));
},
assertVisible: function(element) {
this.assert(this._isVisible(element), Test.Unit.inspect(element) + " was not visible. " + ("" || arguments[1]));
}
}
Test.Unit.Testcase = Class.create();
Object.extend(Object.extend(Test.Unit.Testcase.prototype, Test.Unit.Assertions.prototype), {
initialize: function(name, test, setup, teardown) {
Test.Unit.Assertions.prototype.initialize.bind(this)();
this.name = name;
this.test = test || function() {};
this.setup = setup || function() {};
this.teardown = teardown || function() {};
this.isWaiting = false;
this.timeToWait = 1000;
},
wait: function(time, nextPart) {
this.isWaiting = true;
this.test = nextPart;
this.timeToWait = time;
},
run: function() {
try {
try {
if (!this.isWaiting) this.setup.bind(this)();
this.isWaiting = false;
this.test.bind(this)();
} finally {
if(!this.isWaiting) {
this.teardown.bind(this)();
}
}
}
catch(e) { this.error(e); }
}
});