forked from extern/egroupware
353 lines
7.8 KiB
JavaScript
353 lines
7.8 KiB
JavaScript
|
/**
|
||
|
* eGroupWare eTemplate2 - JS Dropdown Button object
|
||
|
*
|
||
|
* @license http://opensource.org/licenses/gpl-license.php GPL - GNU General Public License
|
||
|
* @package etemplate
|
||
|
* @subpackage api
|
||
|
* @link http://www.egroupware.org
|
||
|
* @author Nathan Gray
|
||
|
* @copyright Nathan Gray 2013
|
||
|
* @version $Id$
|
||
|
*/
|
||
|
|
||
|
"use strict";
|
||
|
|
||
|
/*egw:uses
|
||
|
jquery.jquery;
|
||
|
jquery.jquery-ui;
|
||
|
et2_baseWidget;
|
||
|
*/
|
||
|
|
||
|
/**
|
||
|
* A split button - a button with a dropdown list
|
||
|
*
|
||
|
* There are several parts to the button UI:
|
||
|
* - Container: This is what is percieved as the dropdown button, the whole package together
|
||
|
* - Button: The part on the left that can be clicked
|
||
|
* - Arrow: The button to display the choices
|
||
|
* - Menu: The list of choices
|
||
|
*
|
||
|
* Menu options are passed via the select_options. They are normally ID => Title pairs,
|
||
|
* as for a select box, but the title can also be full HTML if needed.
|
||
|
*/
|
||
|
var et2_dropdown_button = et2_inputWidget.extend({
|
||
|
|
||
|
attributes: {
|
||
|
"label": {
|
||
|
"name": "caption",
|
||
|
"type": "string",
|
||
|
"description": "Label of the button",
|
||
|
"translate": true,
|
||
|
"default": "Select..."
|
||
|
},
|
||
|
"label_updates": {
|
||
|
"name": "Label updates",
|
||
|
"type": "boolean",
|
||
|
"description": "Button label updates when an option is selected from the menu",
|
||
|
"default": true
|
||
|
},
|
||
|
"image": {
|
||
|
"name": "Icon",
|
||
|
"type": "string",
|
||
|
"description": "Add an icon"
|
||
|
},
|
||
|
"ro_image": {
|
||
|
"name": "Read-only Icon",
|
||
|
"type": "string",
|
||
|
"description": "Use this icon instead of hiding for read-only"
|
||
|
},
|
||
|
"onclick": {
|
||
|
"name": "onclick",
|
||
|
"type": "string",
|
||
|
"description": "JS code which gets executed when the button is clicked"
|
||
|
},
|
||
|
"select_options": {
|
||
|
"type": "any",
|
||
|
"name": "Select options",
|
||
|
"default": {},
|
||
|
"description": "Select options for dropdown. Can be a simple key => value list, or value can be full HTML",
|
||
|
// Skip normal initialization for this one
|
||
|
"ignore": true
|
||
|
},
|
||
|
"accesskey": {
|
||
|
"name": "Access Key",
|
||
|
"type": "string",
|
||
|
"default": et2_no_init,
|
||
|
"description": "Alt + <key> activates widget"
|
||
|
},
|
||
|
"tabindex": {
|
||
|
"name": "Tab index",
|
||
|
"type": "integer",
|
||
|
"default": et2_no_init,
|
||
|
"description": "Specifies the tab order of a widget when the 'tab' button is used for navigating."
|
||
|
},
|
||
|
// No such thing as a required button
|
||
|
"required": {
|
||
|
"ignore": true,
|
||
|
}
|
||
|
},
|
||
|
|
||
|
internal_ids: {
|
||
|
div: "",
|
||
|
button: "",
|
||
|
menu: ""
|
||
|
},
|
||
|
|
||
|
div: null,
|
||
|
buttons: null,
|
||
|
button: null,
|
||
|
menu: null,
|
||
|
|
||
|
/**
|
||
|
* Default menu, so there is something for the widget browser / editor to show
|
||
|
*/
|
||
|
default_menu: '<ul> \
|
||
|
<li id="opt_1.1"><a href="javascript:void(0);"><img src="' + egw().image("navbar") + '"/>Option-1.1</a></li>\
|
||
|
<li id="opt_1.2"><a href="javascript:void(0);">Option-1.2</a></li>\
|
||
|
<li id="opt_1.3"><a href="javascript:void(0);">Option-1.3</a></li>\
|
||
|
<li id="opt_1.4"><a href="javascript:void(0);">Option-1.4<br>\
|
||
|
<small>with second line</small>\
|
||
|
</a></li>\
|
||
|
<li id="opt_1.5"><a href="javascript:void(0);">Option-1.5</a></li>\
|
||
|
</ul>',
|
||
|
|
||
|
init: function() {
|
||
|
this._super.apply(this, arguments);
|
||
|
|
||
|
this.clicked = false;
|
||
|
|
||
|
var self = this;
|
||
|
|
||
|
// Create the individual UI elements
|
||
|
|
||
|
// Menu is a UL
|
||
|
this.menu = $j(this.default_menu).attr("id",this.internal_ids.menu)
|
||
|
.hide()
|
||
|
.menu({
|
||
|
selected: function(event,ui) {
|
||
|
self.onselect(event,ui.item);
|
||
|
}
|
||
|
});
|
||
|
|
||
|
this.buttons = $j(document.createElement("div"))
|
||
|
.addClass("et2_dropdown");
|
||
|
|
||
|
// Main "wrapper" div
|
||
|
this.div = $j(document.createElement("div"))
|
||
|
.attr("id", this.internal_ids.div)
|
||
|
.append(this.buttons)
|
||
|
.append(this.menu);
|
||
|
|
||
|
// Left side - activates click action
|
||
|
this.button = $j(document.createElement("button"))
|
||
|
.attr("id", this.internal_ids.button)
|
||
|
.addClass("ui-widget ui-corner-left").removeClass("ui-corner-all")
|
||
|
.appendTo(this.buttons);
|
||
|
|
||
|
// Right side - shows dropdown
|
||
|
this.arrow = $j(document.createElement("button"))
|
||
|
.addClass("ui-widget ui-corner-right").removeClass("ui-corner-all")
|
||
|
.click(function() {
|
||
|
// Clicking it again hides menu
|
||
|
if(self.menu.is(":visible"))
|
||
|
{
|
||
|
self.menu.hide();
|
||
|
return false;
|
||
|
}
|
||
|
// Show menu dropdown
|
||
|
var menu = self.menu.show().position({
|
||
|
my: "left top",
|
||
|
at: "left bottom",
|
||
|
of: self.buttons
|
||
|
});
|
||
|
// Hide menu if clicked elsewhere
|
||
|
$j( document ).one( "click", function() {
|
||
|
menu.hide();
|
||
|
});
|
||
|
return false;
|
||
|
})
|
||
|
// This is the actual down arrow icon
|
||
|
.append("<div class='ui-icon ui-icon-triangle-1-s'/>")
|
||
|
.appendTo(this.buttons);
|
||
|
|
||
|
// Common button UI
|
||
|
this.buttons.children("button")
|
||
|
.addClass("ui-state-default")
|
||
|
.hover(
|
||
|
function() {$j(this).addClass("ui-state-hover");},
|
||
|
function() {$j(this).removeClass("ui-state-hover");}
|
||
|
);
|
||
|
|
||
|
// Icon
|
||
|
this.image = jQuery(document.createElement("img"))
|
||
|
|
||
|
this.setDOMNode(this.div[0]);
|
||
|
},
|
||
|
|
||
|
destroy: function() {
|
||
|
// Destroy widget
|
||
|
this.menu.menu("destroy");
|
||
|
|
||
|
// Null children
|
||
|
this.image = null;
|
||
|
this.button = null;
|
||
|
this.arrow = null;
|
||
|
this.buttons = null;
|
||
|
this.menu = null;
|
||
|
|
||
|
// Remove
|
||
|
this.div.empty().remove();
|
||
|
},
|
||
|
|
||
|
set_id: function(_id) {
|
||
|
this._super.apply(this, arguments);
|
||
|
|
||
|
// Update internal IDs - not really needed since we refer by internal
|
||
|
// javascript reference, but good to keep up to date
|
||
|
this.internal_ids = {
|
||
|
div: this.id + "_wrapper",
|
||
|
button: this.id,
|
||
|
menu: this.id + "_menu"
|
||
|
};
|
||
|
for(var key in this.internal_ids)
|
||
|
{
|
||
|
if(this[key] == null) continue;
|
||
|
this[key].attr("id", this.internal_ids[key]);
|
||
|
}
|
||
|
},
|
||
|
|
||
|
/**
|
||
|
* Set if the button label changes to match the selected option
|
||
|
*
|
||
|
* @param updates boolean Turn updating on or off
|
||
|
*/
|
||
|
set_label_updates: function(updates) {
|
||
|
this.label_updates = updates;
|
||
|
},
|
||
|
|
||
|
set_accesskey: function(key) {
|
||
|
jQuery(this.node).attr("accesskey", key);
|
||
|
},
|
||
|
set_ro_image: function(_image) {
|
||
|
if(this.options.readonly)
|
||
|
{
|
||
|
this.set_image(_image);
|
||
|
}
|
||
|
},
|
||
|
|
||
|
set_image: function(_image) {
|
||
|
if(!this.isInTree() || this.image == null) return;
|
||
|
var found_image = false;
|
||
|
if(!_image.trim())
|
||
|
{
|
||
|
this.image.hide();
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
this.image.show();
|
||
|
}
|
||
|
|
||
|
var src = this.egw().image(_image);
|
||
|
if(src)
|
||
|
{
|
||
|
this.image.attr("src", src);
|
||
|
found_image = true;
|
||
|
}
|
||
|
// allow url's too
|
||
|
else if (_image[0] == '/' || _image.substr(0,4) == 'http')
|
||
|
{
|
||
|
this.image.attr('src', _image);
|
||
|
found_image = true;
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
this.image.hide();
|
||
|
}
|
||
|
},
|
||
|
|
||
|
onclick: function(_node) {
|
||
|
this.clicked = true;
|
||
|
|
||
|
// Execute the JS code connected to the event handler
|
||
|
if (this.options.onclick)
|
||
|
{
|
||
|
// Exectute the legacy JS code
|
||
|
if (!(et2_compileLegacyJS(this.options.onclick, this, _node))())
|
||
|
{
|
||
|
this.clicked = false;
|
||
|
return false;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
this.clicked = false;
|
||
|
},
|
||
|
|
||
|
onselect: function(event, selected_node) {
|
||
|
this.set_value(selected_node.attr("id"));
|
||
|
this.change(selected_node);
|
||
|
},
|
||
|
|
||
|
set_label: function(_value) {
|
||
|
if (this.button)
|
||
|
{
|
||
|
this.label = _value;
|
||
|
|
||
|
this.button.text(_value)
|
||
|
.prepend(this.image);
|
||
|
}
|
||
|
},
|
||
|
|
||
|
/**
|
||
|
* Set the options for the dropdown
|
||
|
*
|
||
|
* @param options Object ID => Label pairs
|
||
|
*/
|
||
|
set_select_options: function(options) {
|
||
|
this.menu.first().empty();
|
||
|
|
||
|
// Allow more complicated content, if passed
|
||
|
if(typeof options == "string")
|
||
|
{
|
||
|
this.menu.append(options);
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
for(var key in options)
|
||
|
{
|
||
|
this.menu.first().append("<li id='"+key+"'><a href='javascript:void(0);'>"+options[key]+"</a></li>");
|
||
|
}
|
||
|
}
|
||
|
this.menu.menu("refresh");
|
||
|
},
|
||
|
|
||
|
/**
|
||
|
* Set tab index
|
||
|
*/
|
||
|
set_tabindex: function(index) {
|
||
|
jQuery(this.button).attr("tabindex", index);
|
||
|
},
|
||
|
|
||
|
set_value: function(new_value) {
|
||
|
var menu_item = $j("[id='"+new_value+"']",this.menu);
|
||
|
if(menu_item.length)
|
||
|
{
|
||
|
this.value = new_value;
|
||
|
if(this.label_updates)
|
||
|
{
|
||
|
this.set_label(menu_item.text());
|
||
|
}
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
this.value = null;
|
||
|
if(this.label_updates)
|
||
|
{
|
||
|
this.set_label(this.options.label);
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
});
|
||
|
|
||
|
et2_register_widget(et2_dropdown_button, ["dropdown_button"]);
|
||
|
|