2011-11-05 16:46:53 +00:00
|
|
|
'use strict';
|
|
|
|
|
2011-05-16 10:49:48 +00:00
|
|
|
/*@
|
|
|
|
Ox.OptionGroup <f> OptionGroup
|
2011-09-19 06:33:52 +00:00
|
|
|
Helper object, used by ButtonGroup, CheckboxGroup, Select and Menu
|
2011-05-16 10:49:48 +00:00
|
|
|
(items, min, max, property) -> <f> OptionGroup
|
|
|
|
items <a> array of items
|
|
|
|
min <n> minimum number of selected items
|
|
|
|
max <n> maximum number of selected items
|
2011-12-21 13:42:47 +00:00
|
|
|
property <s|'checked'> property to check
|
2011-05-16 10:49:48 +00:00
|
|
|
@*/
|
|
|
|
|
2011-04-22 22:03:10 +00:00
|
|
|
Ox.OptionGroup = function(items, min, max, property) {
|
|
|
|
|
2011-06-19 17:48:32 +00:00
|
|
|
var length = items.length;
|
|
|
|
property = property || 'checked';
|
|
|
|
max = max == -1 ? length : max;
|
2011-04-22 22:03:10 +00:00
|
|
|
|
|
|
|
function getLastBefore(pos) {
|
|
|
|
// returns the position of the last checked item before position pos
|
|
|
|
var last = -1;
|
|
|
|
// fixme: why is length not == items.length here?
|
2011-09-17 18:36:09 +00:00
|
|
|
Ox.forEach(Ox.merge(
|
2011-04-22 22:03:10 +00:00
|
|
|
pos > 0 ? Ox.range(pos - 1, -1, -1) : [],
|
|
|
|
pos < items.length - 1 ? Ox.range(items.length - 1, pos, -1) : []
|
|
|
|
), function(v) {
|
|
|
|
if (items[v][property]) {
|
|
|
|
last = v;
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
});
|
|
|
|
return last;
|
|
|
|
}
|
|
|
|
|
|
|
|
function getNumber() {
|
|
|
|
// returns the number of checked items
|
2011-12-21 13:42:47 +00:00
|
|
|
return items.reduce(function(prev, curr) {
|
2011-12-22 15:47:46 +00:00
|
|
|
return prev + !!curr[property];
|
2011-12-21 13:42:47 +00:00
|
|
|
}, 0);
|
2011-04-22 22:03:10 +00:00
|
|
|
}
|
|
|
|
|
2011-05-16 10:49:48 +00:00
|
|
|
/*@
|
2011-12-21 13:42:47 +00:00
|
|
|
[property] <f> returns an array with the positions of all checked items
|
|
|
|
() -> <a> positions of checked items
|
2011-05-16 10:49:48 +00:00
|
|
|
@*/
|
2011-12-21 13:42:47 +00:00
|
|
|
// FIXME: isn't value more useful in all cases?
|
2011-04-22 22:03:10 +00:00
|
|
|
this[property] = function() {
|
2011-12-21 13:42:47 +00:00
|
|
|
return Ox.map(items, function(item, i) {
|
|
|
|
return item[property] ? i : null;
|
|
|
|
});
|
2011-04-22 22:03:10 +00:00
|
|
|
};
|
|
|
|
|
2011-05-16 10:49:48 +00:00
|
|
|
/*@
|
|
|
|
init <f> init group
|
|
|
|
() -> <a> returns items
|
|
|
|
@*/
|
2011-04-22 22:03:10 +00:00
|
|
|
this.init = function() {
|
|
|
|
var num = getNumber(),
|
|
|
|
count = 0;
|
|
|
|
//if (num < min || num > max) {
|
|
|
|
items.forEach(function(item) {
|
|
|
|
if (Ox.isUndefined(item[property])) {
|
|
|
|
item[property] = false;
|
|
|
|
}
|
|
|
|
if (item[property]) {
|
|
|
|
count++;
|
|
|
|
if (count > max) {
|
|
|
|
item[property] = false;
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
if (num < min) {
|
|
|
|
item[property] = true;
|
|
|
|
num++;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
});
|
|
|
|
//}
|
|
|
|
return items;
|
|
|
|
};
|
|
|
|
|
2011-05-16 10:49:48 +00:00
|
|
|
/*@
|
|
|
|
toggle <f> toggle options
|
|
|
|
(pos) -> <a> returns toggled state
|
|
|
|
@*/
|
2011-04-22 22:03:10 +00:00
|
|
|
this.toggle = function(pos) {
|
|
|
|
var last,
|
|
|
|
num = getNumber(),
|
|
|
|
toggled = [];
|
|
|
|
if (!items[pos][property]) { // check
|
|
|
|
if (num >= max) {
|
|
|
|
last = getLastBefore(pos);
|
|
|
|
items[last][property] = false;
|
|
|
|
toggled.push(last);
|
|
|
|
}
|
|
|
|
if (!items[pos][property]) {
|
|
|
|
items[pos][property] = true;
|
|
|
|
toggled.push(pos);
|
|
|
|
}
|
|
|
|
} else { // uncheck
|
|
|
|
if (num > min) {
|
|
|
|
items[pos][property] = false;
|
|
|
|
toggled.push(pos);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return toggled;
|
|
|
|
}
|
|
|
|
|
2012-05-21 10:38:18 +00:00
|
|
|
/*@
|
|
|
|
value <f> get value
|
|
|
|
@*/
|
2011-12-21 13:42:47 +00:00
|
|
|
this.value = function() {
|
|
|
|
var value = Ox.map(items, function(item) {
|
|
|
|
return item[property] ? item.id : null;
|
|
|
|
});
|
|
|
|
return max == 1 ? (value.length ? value[0] : '') : value;
|
|
|
|
};
|
|
|
|
|
2011-04-22 22:03:10 +00:00
|
|
|
return this;
|
|
|
|
|
|
|
|
}
|