2011-07-29 18:48:43 +00:00
|
|
|
// vim: et:ts=4:sw=4:sts=4:ft=javascript
|
2011-05-16 08:24:46 +00:00
|
|
|
|
|
|
|
/*@
|
|
|
|
Ox.TreeList <f:Ox.Element> TreeList Object
|
|
|
|
() -> <f> TreeList Object
|
|
|
|
(options) -> <f> TreeList Object
|
|
|
|
(options, self) -> <f> TreeList Object
|
|
|
|
options <o> Options object
|
|
|
|
data <f|null> data to be parsed to items, needs documentation
|
|
|
|
items <a|[]> array of items
|
|
|
|
max <n|-1> maximum number of items that can be selected, -1 unlimited
|
|
|
|
min <n|0> minimum number of items that have to be selected
|
|
|
|
selected <a|[]> selected ids
|
|
|
|
width <n|256> list width
|
|
|
|
self <o> shared private variable
|
|
|
|
@*/
|
|
|
|
|
2011-04-22 22:03:10 +00:00
|
|
|
Ox.TreeList = function(options, self) {
|
|
|
|
|
2011-05-07 21:07:53 +00:00
|
|
|
// fixme: expanding the last item should cause some scroll
|
|
|
|
|
2011-06-19 17:48:32 +00:00
|
|
|
self = self || {};
|
|
|
|
var that = Ox.Element({}, self)
|
2011-04-22 22:03:10 +00:00
|
|
|
.defaults({
|
|
|
|
data: null,
|
|
|
|
items: [],
|
|
|
|
max: -1,
|
|
|
|
min: 0,
|
|
|
|
selected: [],
|
|
|
|
width: 256
|
|
|
|
})
|
|
|
|
.options(options || {});
|
|
|
|
|
|
|
|
if (self.options.data) {
|
|
|
|
self.options.items = [];
|
|
|
|
//Ox.print('d', self.options.data, 'i', self.options.items)
|
|
|
|
Ox.forEach(self.options.data, function(value, key) {
|
|
|
|
self.options.items.push(parseData(key, value));
|
|
|
|
});
|
|
|
|
//Ox.print('d', self.options.data, 'i', self.options.items)
|
|
|
|
}
|
|
|
|
|
2011-06-19 17:48:32 +00:00
|
|
|
that.$element = Ox.List({
|
2011-04-22 22:03:10 +00:00
|
|
|
construct: constructItem,
|
|
|
|
itemHeight: 16,
|
|
|
|
items: parseItems(),
|
|
|
|
itemWidth: self.options.width,
|
|
|
|
max: self.options.max,
|
|
|
|
min: self.options.min,
|
2011-05-12 03:29:35 +00:00
|
|
|
unique: 'id'
|
2011-04-22 22:03:10 +00:00
|
|
|
}, $.extend({}, self))
|
|
|
|
.addClass('OxTextList OxTreeList')
|
|
|
|
.css({
|
|
|
|
width: self.options.width + 'px'
|
|
|
|
})
|
|
|
|
.click(clickItem)
|
|
|
|
.bindEvent({
|
|
|
|
toggle: toggleItems
|
|
|
|
});
|
|
|
|
|
|
|
|
function clickItem(e) {
|
|
|
|
var $target = $(e.target),
|
|
|
|
$item, id, item;
|
2011-05-11 13:38:19 +00:00
|
|
|
if ($target.is('.OxToggle')) {
|
2011-04-22 22:03:10 +00:00
|
|
|
$item = $target.parent().parent();
|
|
|
|
id = $item.data('id');
|
|
|
|
item = getItemById(id);
|
2011-05-11 13:38:19 +00:00
|
|
|
toggleItem(item, !item.expanded);
|
2011-04-22 22:03:10 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
function constructItem(data) {
|
|
|
|
var $item = $('<div>'),
|
|
|
|
padding = (data.level + !data.items) * 16 - 8;
|
|
|
|
if (data.level || !data.items) {
|
|
|
|
$('<div>')
|
|
|
|
.addClass('OxCell OxTarget')
|
|
|
|
.css({
|
|
|
|
width: padding + 'px',
|
|
|
|
})
|
|
|
|
.appendTo($item);
|
|
|
|
}
|
|
|
|
if (data.items) {
|
|
|
|
$('<div>')
|
|
|
|
.addClass('OxCell')
|
|
|
|
.css({
|
|
|
|
width: '8px',
|
|
|
|
})
|
|
|
|
.append(
|
|
|
|
// fixme: need Ox.Icon()
|
|
|
|
$('<img>')
|
|
|
|
.addClass('OxToggle')
|
|
|
|
.attr({
|
|
|
|
src: Ox.UI.getImagePath(
|
|
|
|
'symbol' + (data.expanded ? 'Down' : 'Right') + '.svg'
|
|
|
|
)
|
|
|
|
})
|
|
|
|
.css({
|
|
|
|
width: '10px',
|
|
|
|
height: '10px',
|
|
|
|
padding: '3px'
|
|
|
|
})
|
|
|
|
)
|
|
|
|
.appendTo($item);
|
|
|
|
}
|
|
|
|
$('<div>')
|
|
|
|
.addClass('OxCell OxTarget')
|
|
|
|
.css({
|
|
|
|
width: (self.options.width - padding - 32 + !data.items * 16) + 'px'
|
|
|
|
})
|
|
|
|
.html(data.title)
|
|
|
|
.appendTo($item);
|
|
|
|
return $item;
|
|
|
|
}
|
|
|
|
|
|
|
|
function getItemById(id, items, level) {
|
2011-06-19 17:48:32 +00:00
|
|
|
var ret = null;
|
|
|
|
items = items || self.options.items;
|
|
|
|
level = level || 0;
|
2011-04-22 22:03:10 +00:00
|
|
|
Ox.forEach(items, function(item) {
|
|
|
|
if (item.id == id) {
|
|
|
|
ret = $.extend(item, {
|
|
|
|
level: level
|
|
|
|
});
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
if (item.items) {
|
|
|
|
ret = getItemById(id, item.items, level + 1);
|
|
|
|
if (ret) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
});
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
function parseData(key, value) {
|
|
|
|
//Ox.print('parseData', key, value)
|
|
|
|
var ret = {
|
2011-07-26 14:33:52 +00:00
|
|
|
id: Ox.uid(),
|
2011-07-26 14:36:31 +00:00
|
|
|
title: key.toString() + ': '
|
2011-04-22 22:03:10 +00:00
|
|
|
},
|
|
|
|
type = Ox.typeOf(value);
|
|
|
|
if (type == 'array' || type == 'object') {
|
2011-07-26 14:36:31 +00:00
|
|
|
ret.title += Ox.toTitleCase(type);
|
2011-04-22 22:03:10 +00:00
|
|
|
ret.items = Ox.map(Ox.sort(Ox.keys(value)), function(k) {
|
2011-07-26 14:33:52 +00:00
|
|
|
return parseData(k, value[k]);
|
2011-04-22 22:03:10 +00:00
|
|
|
});
|
|
|
|
} else {
|
2011-07-26 14:36:31 +00:00
|
|
|
ret.title += (
|
2011-04-22 22:03:10 +00:00
|
|
|
type == 'function' ?
|
2011-07-26 14:36:31 +00:00
|
|
|
value.toString().split('{')[0] :
|
|
|
|
JSON.stringify(value)
|
2011-06-19 17:48:32 +00:00
|
|
|
);
|
2011-04-22 22:03:10 +00:00
|
|
|
}
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
function parseItems(items, level) {
|
2011-06-19 17:48:32 +00:00
|
|
|
var ret = [];
|
|
|
|
items = items || self.options.items;
|
|
|
|
level = level || 0;
|
2011-04-22 22:03:10 +00:00
|
|
|
items.forEach(function(item, i) {
|
|
|
|
var item_ = $.extend({
|
|
|
|
level: level
|
|
|
|
}, item, item.items ? {
|
|
|
|
items: !!item.expanded ?
|
|
|
|
parseItems(item.items, level + 1) : []
|
|
|
|
} : {});
|
|
|
|
ret.push(item_);
|
|
|
|
item.items && $.merge(ret, item_.items);
|
|
|
|
});
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
function toggleItem(item, expanded) {
|
2011-05-07 15:56:29 +00:00
|
|
|
var $img, pos;
|
2011-04-22 22:03:10 +00:00
|
|
|
item.expanded = expanded;
|
2011-05-11 13:53:29 +00:00
|
|
|
//getItemById(item.id).expanded = expanded;
|
2011-04-22 22:03:10 +00:00
|
|
|
$.each(that.$element.find('.OxItem'), function(i, v) {
|
|
|
|
var $item = $(v);
|
|
|
|
if ($item.data('id') == item.id) {
|
2011-05-11 13:38:19 +00:00
|
|
|
$img = $item.find('.OxToggle');
|
2011-04-22 22:03:10 +00:00
|
|
|
pos = $item.data('position');
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
});
|
|
|
|
$img.attr({
|
|
|
|
src: Ox.UI.getImagePath(
|
2011-05-11 13:53:29 +00:00
|
|
|
'symbol' + (expanded ? 'Down' : 'Right') + '.svg'
|
2011-04-22 22:03:10 +00:00
|
|
|
)
|
|
|
|
});
|
2011-05-11 13:38:19 +00:00
|
|
|
that.$element.value(item.id, 'expanded', expanded);
|
2011-05-11 13:53:29 +00:00
|
|
|
expanded ?
|
2011-04-22 22:03:10 +00:00
|
|
|
that.$element.addItems(pos + 1, parseItems(item.items, item.level + 1)) :
|
|
|
|
that.$element.removeItems(pos + 1, parseItems(item.items, item.level + 1).length);
|
|
|
|
}
|
|
|
|
|
|
|
|
function toggleItems(event, data) {
|
|
|
|
data.ids.forEach(function(id, i) {
|
|
|
|
var item = getItemById(id);
|
|
|
|
if (item.items && data.expanded != !!item.expanded) {
|
|
|
|
toggleItem(item, data.expanded);
|
|
|
|
}
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
2011-04-29 12:40:51 +00:00
|
|
|
self.setOption = function(key, value) {
|
2011-04-22 22:03:10 +00:00
|
|
|
if (key == 'data') {
|
|
|
|
|
2011-05-12 03:29:35 +00:00
|
|
|
} else if (key == 'width') {
|
|
|
|
|
2011-04-22 22:03:10 +00:00
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
return that;
|
|
|
|
|
|
|
|
};
|