mirror of
http://git.whoc.org.uk/git/password-manager.git
synced 2025-01-11 00:50:03 +01:00
812 lines
23 KiB
JavaScript
812 lines
23 KiB
JavaScript
|
/*
|
||
|
|
||
|
Copyright 2008-2013 Clipperz Srl
|
||
|
|
||
|
This file is part of Clipperz, the online password manager.
|
||
|
For further information about its features and functionalities please
|
||
|
refer to http://www.clipperz.com.
|
||
|
|
||
|
* Clipperz is free software: you can redistribute it and/or modify it
|
||
|
under the terms of the GNU Affero General Public License as published
|
||
|
by the Free Software Foundation, either version 3 of the License, or
|
||
|
(at your option) any later version.
|
||
|
|
||
|
* Clipperz is distributed in the hope that it will be useful, but
|
||
|
WITHOUT ANY WARRANTY; without even the implied warranty of
|
||
|
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.
|
||
|
See the GNU Affero General Public License for more details.
|
||
|
|
||
|
* You should have received a copy of the GNU Affero General Public
|
||
|
License along with Clipperz. If not, see http://www.gnu.org/licenses/.
|
||
|
|
||
|
*/
|
||
|
|
||
|
/***
|
||
|
|
||
|
MochiKit.Iter 1.5
|
||
|
|
||
|
See <http://mochikit.com/> for documentation, downloads, license, etc.
|
||
|
|
||
|
(c) 2005 Bob Ippolito. All rights Reserved.
|
||
|
|
||
|
***/
|
||
|
|
||
|
MochiKit.Base.module(MochiKit, 'Iter', '1.5', ['Base']);
|
||
|
|
||
|
MochiKit.Base.update(MochiKit.Iter, {
|
||
|
/** @id MochiKit.Iter.registerIteratorFactory */
|
||
|
registerIteratorFactory: function (name, check, iterfactory, /* optional */ override) {
|
||
|
MochiKit.Iter.iteratorRegistry.register(name, check, iterfactory, override);
|
||
|
},
|
||
|
|
||
|
/** @id MochiKit.Iter.isIterable */
|
||
|
isIterable: function(o) {
|
||
|
return o != null &&
|
||
|
(typeof(o.next) == "function" || typeof(o.iter) == "function");
|
||
|
},
|
||
|
|
||
|
/** @id MochiKit.Iter.iter */
|
||
|
iter: function (iterable, /* optional */ sentinel) {
|
||
|
var self = MochiKit.Iter;
|
||
|
if (arguments.length == 2) {
|
||
|
return self.takewhile(
|
||
|
function (a) { return a != sentinel; },
|
||
|
iterable
|
||
|
);
|
||
|
}
|
||
|
if (typeof(iterable.next) == 'function') {
|
||
|
return iterable;
|
||
|
} else if (typeof(iterable.iter) == 'function') {
|
||
|
return iterable.iter();
|
||
|
/*
|
||
|
} else if (typeof(iterable.__iterator__) == 'function') {
|
||
|
//
|
||
|
// XXX: We can't support JavaScript 1.7 __iterator__ directly
|
||
|
// because of Object.prototype.__iterator__
|
||
|
//
|
||
|
return iterable.__iterator__();
|
||
|
*/
|
||
|
}
|
||
|
|
||
|
try {
|
||
|
return self.iteratorRegistry.match(iterable);
|
||
|
} catch (e) {
|
||
|
var m = MochiKit.Base;
|
||
|
if (e == m.NotFound) {
|
||
|
e = new TypeError(typeof(iterable) + ": " + m.repr(iterable) + " is not iterable");
|
||
|
}
|
||
|
throw e;
|
||
|
}
|
||
|
},
|
||
|
|
||
|
/** @id MochiKit.Iter.count */
|
||
|
count: function (n) {
|
||
|
if (!n) {
|
||
|
n = 0;
|
||
|
}
|
||
|
var m = MochiKit.Base;
|
||
|
return {
|
||
|
repr: function () { return "count(" + n + ")"; },
|
||
|
toString: m.forwardCall("repr"),
|
||
|
next: m.counter(n)
|
||
|
};
|
||
|
},
|
||
|
|
||
|
/** @id MochiKit.Iter.cycle */
|
||
|
cycle: function (p) {
|
||
|
var self = MochiKit.Iter;
|
||
|
var m = MochiKit.Base;
|
||
|
var lst = [];
|
||
|
var iterator = self.iter(p);
|
||
|
return {
|
||
|
repr: function () { return "cycle(...)"; },
|
||
|
toString: m.forwardCall("repr"),
|
||
|
next: function () {
|
||
|
try {
|
||
|
var rval = iterator.next();
|
||
|
lst.push(rval);
|
||
|
return rval;
|
||
|
} catch (e) {
|
||
|
if (e != self.StopIteration) {
|
||
|
throw e;
|
||
|
}
|
||
|
if (lst.length === 0) {
|
||
|
this.next = function () {
|
||
|
throw self.StopIteration;
|
||
|
};
|
||
|
} else {
|
||
|
var i = -1;
|
||
|
this.next = function () {
|
||
|
i = (i + 1) % lst.length;
|
||
|
return lst[i];
|
||
|
};
|
||
|
}
|
||
|
return this.next();
|
||
|
}
|
||
|
}
|
||
|
};
|
||
|
},
|
||
|
|
||
|
/** @id MochiKit.Iter.repeat */
|
||
|
repeat: function (elem, /* optional */n) {
|
||
|
var m = MochiKit.Base;
|
||
|
if (typeof(n) == 'undefined') {
|
||
|
return {
|
||
|
repr: function () {
|
||
|
return "repeat(" + m.repr(elem) + ")";
|
||
|
},
|
||
|
toString: m.forwardCall("repr"),
|
||
|
next: function () {
|
||
|
return elem;
|
||
|
}
|
||
|
};
|
||
|
}
|
||
|
return {
|
||
|
repr: function () {
|
||
|
return "repeat(" + m.repr(elem) + ", " + n + ")";
|
||
|
},
|
||
|
toString: m.forwardCall("repr"),
|
||
|
next: function () {
|
||
|
if (n <= 0) {
|
||
|
throw MochiKit.Iter.StopIteration;
|
||
|
}
|
||
|
n -= 1;
|
||
|
return elem;
|
||
|
}
|
||
|
};
|
||
|
},
|
||
|
|
||
|
/** @id MochiKit.Iter.next */
|
||
|
next: function (iterator) {
|
||
|
return iterator.next();
|
||
|
},
|
||
|
|
||
|
/** @id MochiKit.Iter.izip */
|
||
|
izip: function (p, q/*, ...*/) {
|
||
|
var m = MochiKit.Base;
|
||
|
var self = MochiKit.Iter;
|
||
|
var next = self.next;
|
||
|
var iterables = m.map(self.iter, arguments);
|
||
|
return {
|
||
|
repr: function () { return "izip(...)"; },
|
||
|
toString: m.forwardCall("repr"),
|
||
|
next: function () { return m.map(next, iterables); }
|
||
|
};
|
||
|
},
|
||
|
|
||
|
/** @id MochiKit.Iter.ifilter */
|
||
|
ifilter: function (pred, seq) {
|
||
|
var m = MochiKit.Base;
|
||
|
seq = MochiKit.Iter.iter(seq);
|
||
|
if (pred === null) {
|
||
|
pred = m.operator.truth;
|
||
|
}
|
||
|
return {
|
||
|
repr: function () { return "ifilter(...)"; },
|
||
|
toString: m.forwardCall("repr"),
|
||
|
next: function () {
|
||
|
while (true) {
|
||
|
var rval = seq.next();
|
||
|
if (pred(rval)) {
|
||
|
return rval;
|
||
|
}
|
||
|
}
|
||
|
// mozilla warnings aren't too bright
|
||
|
return undefined;
|
||
|
}
|
||
|
};
|
||
|
},
|
||
|
|
||
|
/** @id MochiKit.Iter.ifilterfalse */
|
||
|
ifilterfalse: function (pred, seq) {
|
||
|
var m = MochiKit.Base;
|
||
|
seq = MochiKit.Iter.iter(seq);
|
||
|
if (pred === null) {
|
||
|
pred = m.operator.truth;
|
||
|
}
|
||
|
return {
|
||
|
repr: function () { return "ifilterfalse(...)"; },
|
||
|
toString: m.forwardCall("repr"),
|
||
|
next: function () {
|
||
|
while (true) {
|
||
|
var rval = seq.next();
|
||
|
if (!pred(rval)) {
|
||
|
return rval;
|
||
|
}
|
||
|
}
|
||
|
// mozilla warnings aren't too bright
|
||
|
return undefined;
|
||
|
}
|
||
|
};
|
||
|
},
|
||
|
|
||
|
/** @id MochiKit.Iter.islice */
|
||
|
islice: function (seq/*, [start,] stop[, step] */) {
|
||
|
var self = MochiKit.Iter;
|
||
|
var m = MochiKit.Base;
|
||
|
seq = self.iter(seq);
|
||
|
var start = 0;
|
||
|
var stop = 0;
|
||
|
var step = 1;
|
||
|
var i = -1;
|
||
|
if (arguments.length == 2) {
|
||
|
stop = arguments[1];
|
||
|
} else if (arguments.length == 3) {
|
||
|
start = arguments[1];
|
||
|
stop = arguments[2];
|
||
|
} else {
|
||
|
start = arguments[1];
|
||
|
stop = arguments[2];
|
||
|
step = arguments[3];
|
||
|
}
|
||
|
return {
|
||
|
repr: function () {
|
||
|
return "islice(" + ["...", start, stop, step].join(", ") + ")";
|
||
|
},
|
||
|
toString: m.forwardCall("repr"),
|
||
|
next: function () {
|
||
|
if (start >= stop) {
|
||
|
throw self.StopIteration;
|
||
|
}
|
||
|
|
||
|
var rval;
|
||
|
while (i < start) {
|
||
|
rval = seq.next();
|
||
|
i++;
|
||
|
}
|
||
|
start += step;
|
||
|
return rval;
|
||
|
}
|
||
|
};
|
||
|
},
|
||
|
|
||
|
/** @id MochiKit.Iter.imap */
|
||
|
imap: function (fun, p, q/*, ...*/) {
|
||
|
var m = MochiKit.Base;
|
||
|
var self = MochiKit.Iter;
|
||
|
var iterables = m.map(self.iter, m.extend(null, arguments, 1));
|
||
|
var map = m.map;
|
||
|
var next = self.next;
|
||
|
return {
|
||
|
repr: function () { return "imap(...)"; },
|
||
|
toString: m.forwardCall("repr"),
|
||
|
next: function () {
|
||
|
return fun.apply(this, map(next, iterables));
|
||
|
}
|
||
|
};
|
||
|
},
|
||
|
|
||
|
/** @id MochiKit.Iter.applymap */
|
||
|
applymap: function (fun, seq, self) {
|
||
|
seq = MochiKit.Iter.iter(seq);
|
||
|
var m = MochiKit.Base;
|
||
|
return {
|
||
|
repr: function () { return "applymap(...)"; },
|
||
|
toString: m.forwardCall("repr"),
|
||
|
next: function () {
|
||
|
return fun.apply(self, seq.next());
|
||
|
}
|
||
|
};
|
||
|
},
|
||
|
|
||
|
/** @id MochiKit.Iter.chain */
|
||
|
chain: function (p, q/*, ...*/) {
|
||
|
// dumb fast path
|
||
|
var self = MochiKit.Iter;
|
||
|
var m = MochiKit.Base;
|
||
|
if (arguments.length == 1) {
|
||
|
return self.iter(arguments[0]);
|
||
|
}
|
||
|
var argiter = m.map(self.iter, arguments);
|
||
|
return {
|
||
|
repr: function () { return "chain(...)"; },
|
||
|
toString: m.forwardCall("repr"),
|
||
|
next: function () {
|
||
|
while (argiter.length > 1) {
|
||
|
try {
|
||
|
return argiter[0].next();
|
||
|
} catch (e) {
|
||
|
if (e != self.StopIteration) {
|
||
|
throw e;
|
||
|
}
|
||
|
argiter.shift();
|
||
|
}
|
||
|
}
|
||
|
if (argiter.length == 1) {
|
||
|
// optimize last element
|
||
|
var arg = argiter.shift();
|
||
|
this.next = m.bind("next", arg);
|
||
|
return this.next();
|
||
|
}
|
||
|
throw self.StopIteration;
|
||
|
}
|
||
|
};
|
||
|
},
|
||
|
|
||
|
/** @id MochiKit.Iter.takewhile */
|
||
|
takewhile: function (pred, seq) {
|
||
|
var self = MochiKit.Iter;
|
||
|
seq = self.iter(seq);
|
||
|
return {
|
||
|
repr: function () { return "takewhile(...)"; },
|
||
|
toString: MochiKit.Base.forwardCall("repr"),
|
||
|
next: function () {
|
||
|
var rval = seq.next();
|
||
|
if (!pred(rval)) {
|
||
|
this.next = function () {
|
||
|
throw self.StopIteration;
|
||
|
};
|
||
|
this.next();
|
||
|
}
|
||
|
return rval;
|
||
|
}
|
||
|
};
|
||
|
},
|
||
|
|
||
|
/** @id MochiKit.Iter.dropwhile */
|
||
|
dropwhile: function (pred, seq) {
|
||
|
seq = MochiKit.Iter.iter(seq);
|
||
|
var m = MochiKit.Base;
|
||
|
var bind = m.bind;
|
||
|
return {
|
||
|
"repr": function () { return "dropwhile(...)"; },
|
||
|
"toString": m.forwardCall("repr"),
|
||
|
"next": function () {
|
||
|
while (true) {
|
||
|
var rval = seq.next();
|
||
|
if (!pred(rval)) {
|
||
|
break;
|
||
|
}
|
||
|
}
|
||
|
this.next = bind("next", seq);
|
||
|
return rval;
|
||
|
}
|
||
|
};
|
||
|
},
|
||
|
|
||
|
_tee: function (ident, sync, iterable) {
|
||
|
sync.pos[ident] = -1;
|
||
|
var m = MochiKit.Base;
|
||
|
var listMin = m.listMin;
|
||
|
return {
|
||
|
repr: function () { return "tee(" + ident + ", ...)"; },
|
||
|
toString: m.forwardCall("repr"),
|
||
|
next: function () {
|
||
|
var rval;
|
||
|
var i = sync.pos[ident];
|
||
|
|
||
|
if (i == sync.max) {
|
||
|
rval = iterable.next();
|
||
|
sync.deque.push(rval);
|
||
|
sync.max += 1;
|
||
|
sync.pos[ident] += 1;
|
||
|
} else {
|
||
|
rval = sync.deque[i - sync.min];
|
||
|
sync.pos[ident] += 1;
|
||
|
if (i == sync.min && listMin(sync.pos) != sync.min) {
|
||
|
sync.min += 1;
|
||
|
sync.deque.shift();
|
||
|
}
|
||
|
}
|
||
|
return rval;
|
||
|
}
|
||
|
};
|
||
|
},
|
||
|
|
||
|
/** @id MochiKit.Iter.tee */
|
||
|
tee: function (iterable, n/* = 2 */) {
|
||
|
var rval = [];
|
||
|
var sync = {
|
||
|
"pos": [],
|
||
|
"deque": [],
|
||
|
"max": -1,
|
||
|
"min": -1
|
||
|
};
|
||
|
if (arguments.length == 1 || typeof(n) == "undefined" || n === null) {
|
||
|
n = 2;
|
||
|
}
|
||
|
var self = MochiKit.Iter;
|
||
|
iterable = self.iter(iterable);
|
||
|
var _tee = self._tee;
|
||
|
for (var i = 0; i < n; i++) {
|
||
|
rval.push(_tee(i, sync, iterable));
|
||
|
}
|
||
|
return rval;
|
||
|
},
|
||
|
|
||
|
/** @id MochiKit.Iter.list */
|
||
|
list: function (iterable) {
|
||
|
// Fast-path for Array and Array-like
|
||
|
var rval;
|
||
|
if (iterable instanceof Array) {
|
||
|
return iterable.slice();
|
||
|
}
|
||
|
// this is necessary to avoid a Safari crash
|
||
|
if (typeof(iterable) == "function" &&
|
||
|
!(iterable instanceof Function) &&
|
||
|
typeof(iterable.length) == 'number') {
|
||
|
rval = [];
|
||
|
for (var i = 0; i < iterable.length; i++) {
|
||
|
rval.push(iterable[i]);
|
||
|
}
|
||
|
return rval;
|
||
|
}
|
||
|
|
||
|
var self = MochiKit.Iter;
|
||
|
iterable = self.iter(iterable);
|
||
|
rval = [];
|
||
|
var a_val;
|
||
|
try {
|
||
|
while (true) {
|
||
|
a_val = iterable.next();
|
||
|
rval.push(a_val);
|
||
|
}
|
||
|
} catch (e) {
|
||
|
if (e != self.StopIteration) {
|
||
|
throw e;
|
||
|
}
|
||
|
return rval;
|
||
|
}
|
||
|
// mozilla warnings aren't too bright
|
||
|
return undefined;
|
||
|
},
|
||
|
|
||
|
|
||
|
/** @id MochiKit.Iter.reduce */
|
||
|
reduce: function (fn, iterable, /* optional */initial) {
|
||
|
var i = 0;
|
||
|
var x = initial;
|
||
|
var self = MochiKit.Iter;
|
||
|
iterable = self.iter(iterable);
|
||
|
if (arguments.length < 3) {
|
||
|
try {
|
||
|
x = iterable.next();
|
||
|
} catch (e) {
|
||
|
if (e == self.StopIteration) {
|
||
|
e = new TypeError("reduce() of empty sequence with no initial value");
|
||
|
}
|
||
|
throw e;
|
||
|
}
|
||
|
i++;
|
||
|
}
|
||
|
try {
|
||
|
while (true) {
|
||
|
x = fn(x, iterable.next());
|
||
|
}
|
||
|
} catch (e) {
|
||
|
if (e != self.StopIteration) {
|
||
|
throw e;
|
||
|
}
|
||
|
}
|
||
|
return x;
|
||
|
},
|
||
|
|
||
|
/** @id MochiKit.Iter.range */
|
||
|
range: function (/* [start,] stop[, step] */) {
|
||
|
var start = 0;
|
||
|
var stop = 0;
|
||
|
var step = 1;
|
||
|
if (arguments.length == 1) {
|
||
|
stop = arguments[0];
|
||
|
} else if (arguments.length == 2) {
|
||
|
start = arguments[0];
|
||
|
stop = arguments[1];
|
||
|
} else if (arguments.length == 3) {
|
||
|
start = arguments[0];
|
||
|
stop = arguments[1];
|
||
|
step = arguments[2];
|
||
|
} else {
|
||
|
throw new TypeError("range() takes 1, 2, or 3 arguments!");
|
||
|
}
|
||
|
if (step === 0) {
|
||
|
throw new TypeError("range() step must not be 0");
|
||
|
}
|
||
|
return {
|
||
|
next: function () {
|
||
|
if ((step > 0 && start >= stop) || (step < 0 && start <= stop)) {
|
||
|
throw MochiKit.Iter.StopIteration;
|
||
|
}
|
||
|
var rval = start;
|
||
|
start += step;
|
||
|
return rval;
|
||
|
},
|
||
|
repr: function () {
|
||
|
return "range(" + [start, stop, step].join(", ") + ")";
|
||
|
},
|
||
|
toString: MochiKit.Base.forwardCall("repr")
|
||
|
};
|
||
|
},
|
||
|
|
||
|
/** @id MochiKit.Iter.sum */
|
||
|
sum: function (iterable, start/* = 0 */) {
|
||
|
if (typeof(start) == "undefined" || start === null) {
|
||
|
start = 0;
|
||
|
}
|
||
|
var x = start;
|
||
|
var self = MochiKit.Iter;
|
||
|
iterable = self.iter(iterable);
|
||
|
try {
|
||
|
while (true) {
|
||
|
x += iterable.next();
|
||
|
}
|
||
|
} catch (e) {
|
||
|
if (e != self.StopIteration) {
|
||
|
throw e;
|
||
|
}
|
||
|
}
|
||
|
return x;
|
||
|
},
|
||
|
|
||
|
/** @id MochiKit.Iter.exhaust */
|
||
|
exhaust: function (iterable) {
|
||
|
var self = MochiKit.Iter;
|
||
|
iterable = self.iter(iterable);
|
||
|
try {
|
||
|
while (true) {
|
||
|
iterable.next();
|
||
|
}
|
||
|
} catch (e) {
|
||
|
if (e != self.StopIteration) {
|
||
|
throw e;
|
||
|
}
|
||
|
}
|
||
|
},
|
||
|
|
||
|
/** @id MochiKit.Iter.forEach */
|
||
|
forEach: function (iterable, func, /* optional */obj) {
|
||
|
var m = MochiKit.Base;
|
||
|
var self = MochiKit.Iter;
|
||
|
if (arguments.length > 2) {
|
||
|
func = m.bind(func, obj);
|
||
|
}
|
||
|
// fast path for array
|
||
|
if (m.isArrayLike(iterable) && !self.isIterable(iterable)) {
|
||
|
try {
|
||
|
for (var i = 0; i < iterable.length; i++) {
|
||
|
func(iterable[i]);
|
||
|
}
|
||
|
} catch (e) {
|
||
|
if (e != self.StopIteration) {
|
||
|
throw e;
|
||
|
}
|
||
|
}
|
||
|
} else {
|
||
|
self.exhaust(self.imap(func, iterable));
|
||
|
}
|
||
|
},
|
||
|
|
||
|
/** @id MochiKit.Iter.every */
|
||
|
every: function (iterable, func) {
|
||
|
var self = MochiKit.Iter;
|
||
|
try {
|
||
|
self.ifilterfalse(func, iterable).next();
|
||
|
return false;
|
||
|
} catch (e) {
|
||
|
if (e != self.StopIteration) {
|
||
|
throw e;
|
||
|
}
|
||
|
return true;
|
||
|
}
|
||
|
},
|
||
|
|
||
|
/** @id MochiKit.Iter.sorted */
|
||
|
sorted: function (iterable, /* optional */cmp) {
|
||
|
var rval = MochiKit.Iter.list(iterable);
|
||
|
if (arguments.length == 1) {
|
||
|
cmp = MochiKit.Base.compare;
|
||
|
}
|
||
|
rval.sort(cmp);
|
||
|
return rval;
|
||
|
},
|
||
|
|
||
|
/** @id MochiKit.Iter.reversed */
|
||
|
reversed: function (iterable) {
|
||
|
var rval = MochiKit.Iter.list(iterable);
|
||
|
rval.reverse();
|
||
|
return rval;
|
||
|
},
|
||
|
|
||
|
/** @id MochiKit.Iter.some */
|
||
|
some: function (iterable, func) {
|
||
|
var self = MochiKit.Iter;
|
||
|
try {
|
||
|
self.ifilter(func, iterable).next();
|
||
|
return true;
|
||
|
} catch (e) {
|
||
|
if (e != self.StopIteration) {
|
||
|
throw e;
|
||
|
}
|
||
|
return false;
|
||
|
}
|
||
|
},
|
||
|
|
||
|
/** @id MochiKit.Iter.iextend */
|
||
|
iextend: function (lst, iterable) {
|
||
|
var m = MochiKit.Base;
|
||
|
var self = MochiKit.Iter;
|
||
|
if (m.isArrayLike(iterable) && !self.isIterable(iterable)) {
|
||
|
// fast-path for array-like
|
||
|
for (var i = 0; i < iterable.length; i++) {
|
||
|
lst.push(iterable[i]);
|
||
|
}
|
||
|
} else {
|
||
|
iterable = self.iter(iterable);
|
||
|
try {
|
||
|
while (true) {
|
||
|
lst.push(iterable.next());
|
||
|
}
|
||
|
} catch (e) {
|
||
|
if (e != self.StopIteration) {
|
||
|
throw e;
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
return lst;
|
||
|
},
|
||
|
|
||
|
/** @id MochiKit.Iter.groupby */
|
||
|
groupby: function(iterable, /* optional */ keyfunc) {
|
||
|
var m = MochiKit.Base;
|
||
|
var self = MochiKit.Iter;
|
||
|
if (arguments.length < 2) {
|
||
|
keyfunc = m.operator.identity;
|
||
|
}
|
||
|
iterable = self.iter(iterable);
|
||
|
|
||
|
// shared
|
||
|
var pk = undefined;
|
||
|
var k = undefined;
|
||
|
var v;
|
||
|
|
||
|
function fetch() {
|
||
|
v = iterable.next();
|
||
|
k = keyfunc(v);
|
||
|
};
|
||
|
|
||
|
function eat() {
|
||
|
var ret = v;
|
||
|
v = undefined;
|
||
|
return ret;
|
||
|
};
|
||
|
|
||
|
var first = true;
|
||
|
var compare = m.compare;
|
||
|
return {
|
||
|
repr: function () { return "groupby(...)"; },
|
||
|
next: function() {
|
||
|
// iterator-next
|
||
|
|
||
|
// iterate until meet next group
|
||
|
while (compare(k, pk) === 0) {
|
||
|
fetch();
|
||
|
if (first) {
|
||
|
first = false;
|
||
|
break;
|
||
|
}
|
||
|
}
|
||
|
pk = k;
|
||
|
return [k, {
|
||
|
next: function() {
|
||
|
// subiterator-next
|
||
|
if (v == undefined) { // Is there something to eat?
|
||
|
fetch();
|
||
|
}
|
||
|
if (compare(k, pk) !== 0) {
|
||
|
throw self.StopIteration;
|
||
|
}
|
||
|
return eat();
|
||
|
}
|
||
|
}];
|
||
|
}
|
||
|
};
|
||
|
},
|
||
|
|
||
|
/** @id MochiKit.Iter.groupby_as_array */
|
||
|
groupby_as_array: function (iterable, /* optional */ keyfunc) {
|
||
|
var m = MochiKit.Base;
|
||
|
var self = MochiKit.Iter;
|
||
|
if (arguments.length < 2) {
|
||
|
keyfunc = m.operator.identity;
|
||
|
}
|
||
|
|
||
|
iterable = self.iter(iterable);
|
||
|
var result = [];
|
||
|
var first = true;
|
||
|
var prev_key;
|
||
|
var compare = m.compare;
|
||
|
while (true) {
|
||
|
try {
|
||
|
var value = iterable.next();
|
||
|
var key = keyfunc(value);
|
||
|
} catch (e) {
|
||
|
if (e == self.StopIteration) {
|
||
|
break;
|
||
|
}
|
||
|
throw e;
|
||
|
}
|
||
|
if (first || compare(key, prev_key) !== 0) {
|
||
|
var values = [];
|
||
|
result.push([key, values]);
|
||
|
}
|
||
|
values.push(value);
|
||
|
first = false;
|
||
|
prev_key = key;
|
||
|
}
|
||
|
return result;
|
||
|
},
|
||
|
|
||
|
/** @id MochiKit.Iter.arrayLikeIter */
|
||
|
arrayLikeIter: function (iterable) {
|
||
|
var i = 0;
|
||
|
return {
|
||
|
repr: function () { return "arrayLikeIter(...)"; },
|
||
|
toString: MochiKit.Base.forwardCall("repr"),
|
||
|
next: function () {
|
||
|
if (i >= iterable.length) {
|
||
|
throw MochiKit.Iter.StopIteration;
|
||
|
}
|
||
|
return iterable[i++];
|
||
|
}
|
||
|
};
|
||
|
},
|
||
|
|
||
|
/** @id MochiKit.Iter.hasIterateNext */
|
||
|
hasIterateNext: function (iterable) {
|
||
|
return (iterable && typeof(iterable.iterateNext) == "function");
|
||
|
},
|
||
|
|
||
|
/** @id MochiKit.Iter.iterateNextIter */
|
||
|
iterateNextIter: function (iterable) {
|
||
|
return {
|
||
|
repr: function () { return "iterateNextIter(...)"; },
|
||
|
toString: MochiKit.Base.forwardCall("repr"),
|
||
|
next: function () {
|
||
|
var rval = iterable.iterateNext();
|
||
|
if (rval === null || rval === undefined) {
|
||
|
throw MochiKit.Iter.StopIteration;
|
||
|
}
|
||
|
return rval;
|
||
|
}
|
||
|
};
|
||
|
}
|
||
|
});
|
||
|
|
||
|
|
||
|
MochiKit.Iter.__new__ = function () {
|
||
|
var m = MochiKit.Base;
|
||
|
// Re-use StopIteration if exists (e.g. SpiderMonkey)
|
||
|
if (typeof(StopIteration) != "undefined") {
|
||
|
this.StopIteration = StopIteration;
|
||
|
} else {
|
||
|
/** @id MochiKit.Iter.StopIteration */
|
||
|
this.StopIteration = new m.NamedError("StopIteration");
|
||
|
}
|
||
|
this.iteratorRegistry = new m.AdapterRegistry();
|
||
|
// Register the iterator factory for arrays
|
||
|
this.registerIteratorFactory(
|
||
|
"arrayLike",
|
||
|
m.isArrayLike,
|
||
|
this.arrayLikeIter
|
||
|
);
|
||
|
|
||
|
this.registerIteratorFactory(
|
||
|
"iterateNext",
|
||
|
this.hasIterateNext,
|
||
|
this.iterateNextIter
|
||
|
);
|
||
|
|
||
|
m.nameFunctions(this);
|
||
|
|
||
|
};
|
||
|
|
||
|
MochiKit.Iter.__new__();
|
||
|
|
||
|
//
|
||
|
// XXX: Internet Explorer blows
|
||
|
//
|
||
|
if (MochiKit.__export__) {
|
||
|
reduce = MochiKit.Iter.reduce;
|
||
|
}
|
||
|
|
||
|
MochiKit.Base._exportSymbols(this, MochiKit.Iter);
|