Pretty checkboxed multi-select widget

This commit is contained in:
Nathan Gray 2012-03-15 22:57:55 +00:00
parent ea77d9704e
commit 3a865977ff
4 changed files with 281 additions and 14 deletions

View File

@ -220,6 +220,7 @@ var et2_customfields_list = et2_baseWidget.extend([et2_IDetachedDOM], {
if(!this.options.customfields) return; if(!this.options.customfields) return;
for(var field_name in this.options.customfields) for(var field_name in this.options.customfields)
{ {
// Make sure widget is created, and has the needed function
if(!this.widgets[field_name] || !this.widgets[field_name].set_value) continue; if(!this.widgets[field_name] || !this.widgets[field_name].set_value) continue;
var value = _value[this.prefix + field_name] ? _value[this.prefix + field_name] : null; var value = _value[this.prefix + field_name] ? _value[this.prefix + field_name] : null;
this.widgets[field_name].set_value(value); this.widgets[field_name].set_value(value);

View File

@ -561,7 +561,12 @@ var et2_nextmatch = et2_DOMWidget.extend(et2_IResizeable, {
// Build the popup // Build the popup
if(!this.selectPopup) if(!this.selectPopup)
{ {
var select = et2_createWidget("select", {multiple: true, rows: 8}, this); var select = et2_createWidget("select", {
multiple: true,
rows: 8,
empty_label:this.egw().lang("select columns"),
selected_first: false
}, this);
select.set_select_options(columns); select.set_select_options(columns);
select.set_value(columns_selected); select.set_value(columns_selected);
@ -629,9 +634,8 @@ var et2_nextmatch = et2_DOMWidget.extend(et2_IResizeable, {
self.selectPopup.toggle(); self.selectPopup.toggle();
} }
this.selectPopup = jQuery(document.createElement("fieldset")) this.selectPopup = jQuery(document.createElement("div"))
.addClass("colselection ui-dialog") .addClass("colselection ui-dialog ui-widget-content")
.append("<legend>"+this.egw().lang("Select columns")+"</legend>")
.append(select.getDOMNode()) .append(select.getDOMNode())
.append(okButton.getDOMNode()) .append(okButton.getDOMNode())
.append(cancelButton.getDOMNode()) .append(cancelButton.getDOMNode())

View File

@ -46,6 +46,12 @@ var et2_selectbox = et2_inputWidget.extend({
"name": "Select options", "name": "Select options",
"default": {}, "default": {},
"description": "Internaly used to hold the select options." "description": "Internaly used to hold the select options."
},
"selected_first": {
"name": "Selected options first",
"type": "boolean",
"default": true,
"description": "For multi-selects, put the selected options at the top of the list when first loaded"
} }
}, },
@ -53,17 +59,35 @@ var et2_selectbox = et2_inputWidget.extend({
init: function() { init: function() {
this._super.apply(this, arguments); this._super.apply(this, arguments);
this.input = null;
// Allow no other widgets inside this one // Allow no other widgets inside this one
this.supportedWidgetClasses = []; this.supportedWidgetClasses = [];
// Legacy options could have row count or empty label in first slot // Legacy options could have row count or empty label in first slot
if(typeof this.options.rows == "string" && isNaN(this.options.rows)) { if(typeof this.options.rows == "string")
this.options.empty_label = this.egw().lang(this.options.rows); {
this.options.rows = 1; if(isNaN(this.options.rows))
{
this.options.empty_label = this.egw().lang(this.options.rows);
this.options.rows = 1;
}
else
{
this.options.rows = parseInt(this.options.rows);
}
}
if(this.options.rows > 1)
{
this.options.multiple = true;
this.createMultiSelect();
}
else
{
this.createInputWidget();
} }
if(this.options.rows > 1) this.options.multiple = true;
this.createInputWidget();
}, },
destroy: function() { destroy: function() {
@ -117,11 +141,19 @@ var et2_selectbox = et2_inputWidget.extend({
} }
}, },
/**
* Add an option to regular drop-down select
*/
_appendOptionElement: function(_value, _label, _title) { _appendOptionElement: function(_value, _label, _title) {
if(_value == "" && (_label == null || _label == "")) { if(_value == "" && (_label == null || _label == "")) {
_label = this.options.empty_label; _label = this.options.empty_label;
} }
if(this.input == null)
{
return this._appendMultiOption(_value, _label, _title);
}
var option = $j(document.createElement("option")) var option = $j(document.createElement("option"))
.attr("value", _value) .attr("value", _value)
.text(_label+""); .text(_label+"");
@ -133,6 +165,61 @@ var et2_selectbox = et2_inputWidget.extend({
option.appendTo(this.input); option.appendTo(this.input);
}, },
/**
* Append a value to multi-select
*/
_appendMultiOption: function(_value, _label, _title) {
var option_data = null;
if(typeof _label == "object")
{
option_data = _label;
_label = option_data.label;
}
// Already in header
if(_label == this.options.empty_label) return;
var opt_id = this.id + "_opt_" + _value;
var label = jQuery(document.createElement("label"))
.attr("for", opt_id)
.hover(
function() {jQuery(this).addClass("ui-state-hover")},
function() {jQuery(this).removeClass("ui-state-hover")}
);
var option = jQuery(document.createElement("input"))
.attr("type", "checkbox")
.attr("id",opt_id)
.attr("value", _value)
.appendTo(label);
if(typeof _title !== "undefined")
{
option.attr("title",_title);
}
// Some special stuff for categories
if(option_data)
{
if(option_data.data.icon)
{
var img = this.egw().image(option_data.data.icon);
jQuery(document.createElement("img"))
.attr("src", img)
.appendTo(label);
}
if(option_data.data.color)
{
label.css("background-color",option_data.data.color);
}
}
label.append(jQuery("<span>"+_label+"</span>"))
var li = jQuery(document.createElement("li")).append(label);
li.appendTo(this.multiOptions);
},
/**
* Create a regular drop-down select box
*/
createInputWidget: function() { createInputWidget: function() {
// Create the base input widget // Create the base input widget
this.input = $j(document.createElement("select")) this.input = $j(document.createElement("select"))
@ -155,6 +242,73 @@ var et2_selectbox = et2_inputWidget.extend({
} }
}, },
/**
* Create a list of checkboxes
*/
createMultiSelect: function() {
var node = jQuery(document.createElement("div"))
.addClass("et2_selectbox");
var header = jQuery(document.createElement("div"))
.addClass("ui-widget-header ui-helper-clearfix")
.appendTo(node);
if(this.options.empty_label)
{
jQuery(document.createElement("span"))
.text(this.options.empty_label)
.addClass("ui-multiselect-header")
.appendTo(header);
}
// Set up for options to be added later
var options = this.multiOptions = jQuery(document.createElement("ul"));
this.multiOptions.addClass("ui-multiselect-checkboxes ui-helper-reset")
.css("height", 1.9*this.options.rows + "em")
.appendTo(node);
if(this.options.rows >= 5)
{
// Check / uncheck all
var header_controls = {
check: {
icon_class: 'ui-icon-check',
label: 'Check all',
click: function(e) {
jQuery("input",e.data).attr("checked", true);
}
},
uncheck: {
icon_class: 'ui-icon-closethick',
label: 'Uncheck all',
click: function(e) {
jQuery("input",e.data).attr("checked", false);
}
}
}
var controls = jQuery(document.createElement("ul"))
.addClass('ui-helper-reset')
.appendTo(header);
for(var key in header_controls)
{
jQuery(document.createElement("li"))
.addClass("et2_clickable")
.click(options, header_controls[key].click)
.append('<span class="ui-icon ' + header_controls[key].icon_class + '"/>')
.append("<span>"+this.egw().lang(header_controls[key].label)+"</span>")
.appendTo(controls);
}
}
else if (header[0].childNodes.length == 0)
{
// Hide header - nothing there but padding
header.hide();
}
this.setDOMNode(node[0]);
},
loadFromXML: function(_node) { loadFromXML: function(_node) {
// Read the option-tags // Read the option-tags
var options = et2_directChildrenByTagName(_node, "options"); var options = et2_directChildrenByTagName(_node, "options");
@ -169,6 +323,14 @@ var et2_selectbox = et2_inputWidget.extend({
}, },
set_value: function(_value) { set_value: function(_value) {
if(typeof _value == "string" && _value.match(/[,0-9]+$/) !== null)
{
_value = _value.split(',');
}
if(this.input == null)
{
return this.set_multi_value(_value);
}
jQuery("option",this.input).attr("selected", false); jQuery("option",this.input).attr("selected", false);
this.value = _value; this.value = _value;
if(typeof _value == "array") if(typeof _value == "array")
@ -190,11 +352,41 @@ var et2_selectbox = et2_inputWidget.extend({
if(jQuery("option[value='"+_value+"']", this.input).attr("selected", true).length == 0) if(jQuery("option[value='"+_value+"']", this.input).attr("selected", true).length == 0)
{ {
this.egw().debug("warning", "Tried to set value that isn't an option", this, _value); this.egw().debug("warning", "Tried to set value that isn't an option", this, _value);
//console.trace();
} }
} }
}, },
set_multi_value: function(_value) {
jQuery("input",this.multiOptions).attr("checked", false);
if(typeof _value == "array")
{
for(var i = 0; i < _value.length; i++)
{
jQuery("input[value='"+_value[i]+"']", this.multiOptions).attr("checked", true);
}
}
else if (typeof _value == "object")
{
for(var i in _value)
{
jQuery("input[value='"+_value[i]+"']", this.multiOptions).attr("checked", true);
}
}
else
{
if(jQuery("input[value='"+_value+"']", this.multiOptions).attr("checked", true).length == 0)
{
this.egw().debug("warning", "Tried to set value that isn't an option", this, _value);
}
}
// Sort selected to the top
if(this.selected_first)
{
this.multiOptions.find("li:has(input:checked)").prependTo(this.multiOptions);
}
},
/** /**
* The set_select_options function is added, as the select options have to be * The set_select_options function is added, as the select options have to be
* added after the "option"-widgets were added to selectbox. * added after the "option"-widgets were added to selectbox.
@ -220,9 +412,18 @@ var et2_selectbox = et2_inputWidget.extend({
if (typeof _options[key] === 'object') if (typeof _options[key] === 'object')
{ {
this._appendOptionElement(key, if(this.input == null)
_options[key]["label"] ? _options[key]["label"] : "", {
_options[key]["title"] ? _options[key]["title"] : ""); // Allow some special extras for objects by passing the whole thing
_options[key]["label"] = _options[key]["label"] ? _options[key]["label"] : "";
this._appendMultiOption(key, _options[key], _options[key]["title"]);
}
else
{
this._appendOptionElement(key,
_options[key]["label"] ? _options[key]["label"] : "",
_options[key]["title"] ? _options[key]["title"] : "");
}
} }
else else
{ {
@ -231,6 +432,19 @@ var et2_selectbox = et2_inputWidget.extend({
} }
// Sometimes value gets set before options // Sometimes value gets set before options
if(this.value) this.set_value(this.value); if(this.value) this.set_value(this.value);
},
getValue: function() {
if(this.input == null)
{
var value = [];
jQuery("input:checked",this.multiOptions).each(function(){value.push(this.value);});
return value;
}
else
{
return this._super.apply(this, arguments);
}
} }
}); });

View File

@ -174,6 +174,54 @@ button.et2_button_text:focus {
font-style: italic; font-style: italic;
} }
/**
* Multi-select widget
*/
.et2_selectbox .ui-widget-header {
padding: 2px 6px 0px 6px;
}
.et2_selectbox .ui-widget-header ul {
float: right;
}
.et2_selectbox .ui-widget-header li {
float: left;
padding: 2px;
padding-top: 0px
}
.et2_selectbox .ui-widget-header li span.ui-icon {
float: left;
margin-top: -2px;
}
.et2_selectbox .ui-multiselect-checkboxes {
overflow-y: scroll;
position: relative;
text-align: left;
}
.et2_selectbox .ui-multiselect-checkboxes li {
margin: 0px;
clear: both;
padding-left: 26px;
padding-right: 3px;
text-decoration: none;
list-style-image: none;
list-style-type: none;
text-indent: -26px;
}
.et2_selectbox .ui-multiselect-checkboxes label {
display: block;
border: 1px solid transparent;
}
.et2_selectbox input[type="checkbox"] {
margin: 3px;
}
.et2_selectbox .ui-multiselect-checkboxes img {
float: right;
height: 1.8em;
}
/**
* Date / Time widgets
*/
span.et2_date { span.et2_date {
min-width: 130px; min-width: 130px;
} }
@ -537,7 +585,7 @@ label input, label span, label div, label select, label textarea {
} }
.nextmatch_header > .filters { .nextmatch_header > .filters {
padding-left: 5px; padding-left: 5px;
} }
.nextmatch_header .et2_button_icon { .nextmatch_header .et2_button_icon {
vertical-align: middle; vertical-align: middle;
} }