2013-04-09 10:45:59 +02:00
|
|
|
/**
|
|
|
|
* EGroupware clientside Application javascript base 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
|
|
|
|
* @version $Id$
|
|
|
|
*/
|
|
|
|
|
|
|
|
"use strict";
|
|
|
|
|
|
|
|
/*egw:uses
|
2013-07-19 17:22:54 +02:00
|
|
|
egw_inheritance;
|
2013-04-09 10:45:59 +02:00
|
|
|
*/
|
|
|
|
|
2013-11-04 21:54:23 +01:00
|
|
|
/**
|
|
|
|
* Object to collect instanciated appliction objects
|
|
|
|
*
|
|
|
|
* Attributes classes collects loaded application classes,
|
|
|
|
* which can get instanciated:
|
|
|
|
*
|
|
|
|
* app[appname] = new app.classes[appname]();
|
|
|
|
*
|
|
|
|
* On destruction only app[appname] gets deleted, app.classes[appname] need to be used again!
|
|
|
|
*
|
|
|
|
* @type object
|
|
|
|
*/
|
|
|
|
window.app = {classes: {}};
|
2013-04-09 10:45:59 +02:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Common base class for application javascript
|
|
|
|
* Each app should extend as needed.
|
|
|
|
*
|
|
|
|
* All application javascript should be inside. Intitialization goes in init(),
|
2013-11-04 21:54:23 +01:00
|
|
|
* clean-up code goes in destroy(). Initialization is done once all js is loaded.
|
2013-04-09 10:45:59 +02:00
|
|
|
*
|
|
|
|
* var app.appname = AppJS.extend({
|
|
|
|
* // Actually set this one, the rest is example
|
|
|
|
* appname: appname,
|
|
|
|
*
|
|
|
|
* internal_var: 1000,
|
|
|
|
*
|
|
|
|
* init: function()
|
|
|
|
* {
|
|
|
|
* // Call the super
|
|
|
|
* this._super.apply(this, arguments);
|
2013-11-04 21:54:23 +01:00
|
|
|
*
|
2013-04-09 10:45:59 +02:00
|
|
|
* // Init the stuff
|
2013-11-04 21:54:23 +01:00
|
|
|
* if ( egw.preference('dateformat', 'common') )
|
2013-04-09 10:45:59 +02:00
|
|
|
* {
|
|
|
|
* // etc
|
|
|
|
* }
|
|
|
|
* },
|
|
|
|
* _private: function()
|
|
|
|
* {
|
|
|
|
* // Underscore private by convention
|
|
|
|
* }
|
|
|
|
* });
|
|
|
|
*/
|
2013-11-04 21:54:23 +01:00
|
|
|
var AppJS = Class.extend(
|
|
|
|
{
|
2013-04-09 10:45:59 +02:00
|
|
|
/**
|
|
|
|
* Internal application name - override this
|
|
|
|
*/
|
|
|
|
appname: '',
|
2013-11-04 21:54:23 +01:00
|
|
|
|
2013-08-20 21:20:10 +02:00
|
|
|
/**
|
|
|
|
* Internal reference to etemplate2 widget tree
|
|
|
|
*/
|
|
|
|
et2: null,
|
2013-11-04 21:54:23 +01:00
|
|
|
|
2013-10-07 18:53:13 +02:00
|
|
|
/**
|
|
|
|
* Internal reference to egw client-side api object for current app and window
|
|
|
|
*/
|
|
|
|
egw: null,
|
2013-04-09 10:45:59 +02:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Initialization and setup goes here, but the etemplate2 object
|
|
|
|
* is not yet ready.
|
|
|
|
*/
|
|
|
|
init: function() {
|
|
|
|
window.app[this.appname] = this;
|
2013-11-04 21:54:23 +01:00
|
|
|
|
2013-10-07 18:53:13 +02:00
|
|
|
this.egw = egw(this.appname, window);
|
2013-04-09 10:45:59 +02:00
|
|
|
},
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Clean up any created objects & references
|
|
|
|
*/
|
|
|
|
destroy: function() {
|
2013-08-20 21:20:10 +02:00
|
|
|
delete this.et2;
|
2013-04-09 10:45:59 +02:00
|
|
|
delete window.app[this.appname];
|
|
|
|
},
|
|
|
|
|
|
|
|
/**
|
|
|
|
* This function is called when the etemplate2 object is loaded
|
|
|
|
* and ready. If you must store a reference to the et2 object,
|
2013-08-21 01:20:56 +02:00
|
|
|
* make sure to clean it up in destroy(). Note that this can be called
|
|
|
|
* several times, with different et2 objects, as templates are loaded.
|
2013-04-09 10:45:59 +02:00
|
|
|
*
|
|
|
|
* @param et2 etemplate2 Newly ready object
|
|
|
|
*/
|
|
|
|
et2_ready: function(et2) {
|
2013-08-21 01:20:56 +02:00
|
|
|
if(this.et2 !== null)
|
2013-08-20 21:20:10 +02:00
|
|
|
{
|
2013-08-21 01:20:56 +02:00
|
|
|
egw.debug('log', "Changed et2 object");
|
2013-08-20 21:20:10 +02:00
|
|
|
}
|
2013-08-21 01:20:56 +02:00
|
|
|
this.et2 = et2.widgetContainer;
|
2013-08-20 21:20:10 +02:00
|
|
|
},
|
2013-11-04 21:54:23 +01:00
|
|
|
|
2013-08-20 21:20:10 +02:00
|
|
|
/**
|
2013-11-04 21:54:23 +01:00
|
|
|
* Open an entry.
|
|
|
|
*
|
2013-08-20 21:20:10 +02:00
|
|
|
* Designed to be used with the action system as a callback
|
|
|
|
* eg: onExecute => app.<appname>.open
|
2013-11-04 21:54:23 +01:00
|
|
|
*
|
2013-08-20 21:20:10 +02:00
|
|
|
* @param _action
|
|
|
|
* @param _senders
|
|
|
|
*/
|
|
|
|
open: function(_action, _senders) {
|
|
|
|
var id_app = _senders[0].id.split('::')
|
|
|
|
egw.open(id_app[1], this.appname);
|
|
|
|
},
|
2013-11-04 21:54:23 +01:00
|
|
|
|
2013-08-20 21:20:10 +02:00
|
|
|
/**
|
|
|
|
* A generic method to action to server asynchronously
|
2013-11-04 21:54:23 +01:00
|
|
|
*
|
2013-08-20 21:20:10 +02:00
|
|
|
* Designed to be used with the action system as a callback.
|
|
|
|
* In the PHP side, set the action
|
2013-11-04 21:54:23 +01:00
|
|
|
* 'onExecute' => 'javaScript:app.<appname>.action', and
|
2013-08-20 21:20:10 +02:00
|
|
|
* implement _do_action(action_id, selected)
|
2013-11-04 21:54:23 +01:00
|
|
|
*
|
|
|
|
* @param {egwAction} _action
|
2013-08-20 21:20:10 +02:00
|
|
|
* @param {egwActionObject[]} _elems
|
|
|
|
*/
|
|
|
|
action: function(_action, _elems)
|
|
|
|
{
|
|
|
|
// let user confirm select-all
|
|
|
|
var select_all = _action.getManager().getActionById("select_all");
|
2013-11-04 21:54:23 +01:00
|
|
|
var confirm_msg = (_elems.length > 1 || select_all && select_all.checked) &&
|
2013-08-20 21:20:10 +02:00
|
|
|
typeof _action.data.confirm_multiple != 'undefined' ?
|
|
|
|
_action.data.confirm_multiple : _action.data.confirm;
|
2013-11-04 21:54:23 +01:00
|
|
|
|
2013-08-20 21:20:10 +02:00
|
|
|
if (typeof confirm_msg != 'undefined')
|
|
|
|
{
|
|
|
|
var that = this;
|
|
|
|
var action_id = _action.id;
|
|
|
|
et2_dialog.show_dialog(function(button_id,value)
|
|
|
|
{
|
|
|
|
if (button_id != et2_dialog.NO_BUTTON)
|
|
|
|
{
|
2013-11-04 21:54:23 +01:00
|
|
|
that._do_action(action_id, _elems);
|
2013-08-20 21:20:10 +02:00
|
|
|
}
|
|
|
|
}, confirm_msg, egw.lang('Confirmation required'), et2_dialog.BUTTONS_YES_NO, et2_dialog.QUESTION_MESSAGE);
|
|
|
|
}
|
|
|
|
else if (typeof this._do_action == 'function')
|
|
|
|
{
|
2013-11-04 21:54:23 +01:00
|
|
|
this._do_action(_action.id, _elems);
|
2013-08-20 21:20:10 +02:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
// If this is a nextmatch action, do an ajax submit setting the action
|
|
|
|
var nm = null;
|
|
|
|
var action = _action;
|
|
|
|
while(nm == null && action.parent != null)
|
|
|
|
{
|
|
|
|
if(action.data.nextmatch) nm = action.data.nextmatch;
|
|
|
|
action = action.parent;
|
|
|
|
}
|
|
|
|
if(nm != null)
|
|
|
|
{
|
|
|
|
var value = {};
|
|
|
|
value[nm.options.settings.action_var] = _action.id;
|
|
|
|
nm.set_value(value);
|
|
|
|
nm.getInstanceManager().submit();
|
|
|
|
}
|
|
|
|
}
|
2013-12-05 01:00:43 +01:00
|
|
|
},
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Set the application's state to the given state.
|
|
|
|
*
|
|
|
|
* While not pretending to implement the history API, it is patterned similarly
|
|
|
|
* @link http://www.whatwg.org/specs/web-apps/current-work/multipage/history.html
|
|
|
|
*
|
|
|
|
* The default implementation works with the favorites to apply filters to a nextmatch.
|
|
|
|
*
|
|
|
|
*
|
|
|
|
* @param {object} state description
|
|
|
|
*/
|
|
|
|
setState: function(state)
|
|
|
|
{
|
|
|
|
// State should be an object, not a string, but we'll try
|
|
|
|
if(typeof state == "string")
|
|
|
|
{
|
|
|
|
if(state.indexOf('{') != -1 || state =='null')
|
|
|
|
{
|
|
|
|
state = JSON.parse(state);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if(typeof state != "object")
|
|
|
|
{
|
|
|
|
egw.debug('error', 'Unable to set state to %o, needs to be an object',state);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
if(state == null)
|
|
|
|
{
|
|
|
|
state = {};
|
|
|
|
}
|
2013-11-04 21:54:23 +01:00
|
|
|
|
2013-12-05 01:00:43 +01:00
|
|
|
// Try and find a nextmatch widget, and set its filters
|
|
|
|
var nextmatched = false;
|
|
|
|
var et2 = etemplate2.getByApplication(this.appname);
|
|
|
|
for(var i = 0; i < et2.length; i++)
|
|
|
|
{
|
|
|
|
et2[i].widgetContainer.iterateOver(function(_widget) {
|
|
|
|
// Apply
|
|
|
|
_widget.activeFilters = state;
|
|
|
|
_widget.applyFilters();
|
|
|
|
nextmatched = true;
|
|
|
|
}, this, et2_nextmatch);
|
|
|
|
}
|
|
|
|
|
|
|
|
// No nextmatch? Try a redirect to list
|
|
|
|
if(!nextmatched)
|
|
|
|
{
|
|
|
|
egw.open('',this.appname,'list',{'state': state},this.appname);
|
|
|
|
}
|
|
|
|
},
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Retrieve the current state of the application for future restoration
|
|
|
|
*
|
|
|
|
* The state can be anything, as long as it's an object. The contents are
|
|
|
|
* application specific. The default implementation finds a nextmatch and
|
|
|
|
* returns its value.
|
|
|
|
*
|
|
|
|
* @return {object} Value of a nextmatch
|
|
|
|
*/
|
|
|
|
getState: function()
|
|
|
|
{
|
|
|
|
var state = {};
|
|
|
|
|
|
|
|
// Try and find a nextmatch widget, and set its filters
|
|
|
|
var et2 = etemplate2.getByApplication(this.appname);
|
|
|
|
for(var i = 0; i < et2.length; i++)
|
|
|
|
{
|
|
|
|
et2.widgetContainer.iterateOver(function(_widget) {
|
|
|
|
state = _widget.getValue();
|
|
|
|
}, this, et2_nextmatch);
|
|
|
|
}
|
|
|
|
|
|
|
|
return state;
|
|
|
|
}
|
2013-04-09 10:45:59 +02:00
|
|
|
});
|