2020-02-09 12:26:23 +01:00
/ * *
* EGroupware - Admin - Javascript UI
*
* @link : https : //www.egroupware.org
* @package filemanager
* @author Ralf Becker < RalfBecker - AT - outdoor - training.de >
* @copyright ( c ) 2013 - 20 by Ralf Becker < RalfBecker - AT - outdoor - training.de >
* @license http : //opensource.org/licenses/gpl-license.php GPL - GNU General Public License
* /
/ * e g w : u s e s
/ a p i / j s / j s a p i / e g w _ a p p . j s
* /
import 'jquery' ;
import 'jqueryui' ;
import '../jsapi/egw_global' ;
import '../etemplate/et2_types' ;
2020-03-20 18:38:38 +01:00
import { EgwApp } from '../../api/js/jsapi/egw_app' ;
2020-02-09 12:26:23 +01:00
/ * *
* UI for Admin
*
* @augments AppJS
* /
class AdminApp extends EgwApp
/ * *
* @lends app . classes . admin
* /
{
/ * *
* reference to iframe
*
* { et2_iframe }
* /
iframe : any = null ;
/ * *
* reference to nextmatch
*
* { et2_extension_nextmatch }
* /
nm : any = null ;
/ * *
* Reference to div to hold AJAX loadable pages
*
* { et2_box }
* /
ajax_target : any = null ;
/ * *
* Reference to ACL edit dialog ( not the list )
* /
acl_dialog : any = null ;
tree : any = null ;
groups : any ;
/ * *
* Constructor
*
* @memberOf app . classes . admin
* /
constructor ( )
{
// call parent
2020-03-20 18:38:38 +01:00
super ( 'admin' ) ;
2020-02-09 12:26:23 +01:00
}
/ * *
* Destructor
* /
2020-02-11 17:40:17 +01:00
destroy ( _app )
2020-02-09 12:26:23 +01:00
{
this . iframe = null ;
this . nm = null ;
this . acl_dialog = null ;
this . tree = null ;
// call parent
2020-02-11 17:40:17 +01:00
super . destroy ( _app ) ;
2020-02-09 12:26:23 +01:00
}
/ * *
* This function is called when the etemplate2 object is loaded
* and ready . If you must store a reference to the et2 object ,
* make sure to clean it up in destroy ( ) .
*
* @param { etemplate2 } _et2
* @param { string } _name name of template loaded
* /
et2_ready ( _et2 , _name )
{
// call parent
2020-02-11 17:40:17 +01:00
super . et2_ready ( _et2 , _name ) ;
2020-02-09 12:26:23 +01:00
switch ( _name )
{
case 'admin.index' :
var iframe = this . iframe = this . et2 . getWidgetById ( 'iframe' ) ;
this . nm = this . et2 . getWidgetById ( 'nm' ) ;
this . groups = this . et2 . getWidgetById ( 'groups' ) ;
this . groups . set_disabled ( true ) ;
this . ajax_target = this . et2 . getWidgetById ( 'ajax_target' ) ;
this . tree = this . et2 . getWidgetById ( 'tree' ) ;
if ( iframe )
{
var self = this ;
jQuery ( iframe . getDOMNode ( ) ) . off ( 'load.admin' )
. bind ( 'load.admin' , function ( ) {
if ( this . contentDocument . location . href . match ( /(\/admin\/|\/admin\/index.php|menuaction=admin.admin_ui.index)/ ) )
{
this . contentDocument . location . href = 'about:blank' ; // stops redirect from admin/index.php
self . load ( ) ; // load own top-level index aka user-list
}
self . _hide_navbar . call ( self ) ;
}
) ;
}
if ( this . ajax_target && this . et2 . getArrayMgr ( 'content' ) . getEntry ( 'ajax_target' ) )
{
this . load ( this . et2 . getArrayMgr ( 'content' ) . getEntry ( 'ajax_target' ) ) ;
}
break ;
case 'admin.customfield_edit' :
// Load settings appropriate to currently set type
var widget = _et2 . widgetContainer . getWidgetById ( 'cf_type' ) ;
this . cf_type_change ( null , widget ) ;
break ;
case 'admin.mailaccount' :
this . account_hide_not_applying ( ) ;
break ;
case 'admin.cmds' :
var selected = this . et2 . getWidgetById ( 'nm' ) . getSelection ( ) ;
if ( selected && selected . ids . length == 1 )
{
this . cmds_onselect ( selected . ids ) ;
}
else
{
this . et2 . getWidgetById ( 'splitter' ) . dock ( ) ;
}
break ;
}
}
/ * *
* Show given url in ( visible ) iframe or nextmatch with accounts ( ! _url )
*
* @param { string } [ _url = ] url to show in iframe or nothing for showing
* /
load ( _url? : string )
{
if ( this . iframe && this . iframe . getDOMNode ( ) . contentDocument . location . href
. match ( /menuaction=admin.admin_statistics.submit.+required=true/ ) && ( ! _url ||
! _url . match ( /statistics=(postpone|canceled|submitted)/ ) ) )
{
this . egw . message ( this . egw . lang ( 'Please submit (or postpone) statistic first' ) , 'info' ) ;
return ; // do not allow to leave statistics submit
}
// url outside EGroupware eg. eSyncPro linking to wikipedia
if ( _url && _url . indexOf ( this . egw . webserverUrl ) == - 1 )
{
window . open ( _url , '_blank' ) ;
return ;
}
var ajax : any = false ;
if ( _url )
{
// Try to load it without the iframe
ajax = _url . match ( /ajax=true/ ) && _url . match ( /menuaction=/ ) ;
if ( ajax )
{
if ( this . ajax_target . node . children . length )
{
// Node has children already? Check for loading over an
// existing etemplate, and remove it first
jQuery ( this . ajax_target . node . children ) . each ( function ( ) {
var old = etemplate2 . getById ( this . id ) ;
if ( old ) old . clear ( ) ;
} ) ;
jQuery ( this . ajax_target . node ) . empty ( ) ;
}
this . egw . json (
framework . activeApp . getMenuaction ( 'ajax_exec' , _url ) ,
// It's important that the context is null, or etemplate2
// won't load the template properly
[ _url ] , this . _ajax_load_callback , null , true , this
) . sendRequest ( ) ;
}
else
{
this . iframe . set_src ( _url ) ;
}
var m = _url . match ( /menuaction=([^&]+)(?:.*appname=(\w+))?/ ) ;
if ( m && m . length >= 2 )
{
var app = m [ 2 ] ? m [ 2 ] : m [ 1 ] . split ( '.' ) [ 0 ] ;
this . tree . set_value ( '/apps/' + app + '/' + m [ 1 ] ) ;
}
}
else
{
this . egw . app_header ( '' ) ;
// blank iframe, to not keep something running there
this . iframe . getDOMNode ( ) . contentDocument . location . href = 'about:blank' ;
}
this . iframe . set_disabled ( ! _url || ajax ) ;
this . nm . set_disabled ( ! ! _url || ajax ) ;
this . groups . set_disabled ( true ) ;
this . ajax_target . set_disabled ( ! ajax ) ;
}
/ * *
* Observer method receives update notifications from all applications
*
* App is responsible for only reacting to "messages" it is interested in !
*
* @param { string } _msg message ( already translated ) to show , eg . 'Entry deleted'
* @param { string } _app application name
* @param { ( string | number ) } _id id of entry to refresh or null
* @param { string } _type either 'update' , 'edit' , 'delete' , 'add' or null
* - update : request just modified data from given rows . Sorting is not considered ,
* so if the sort field is changed , the row will not be moved .
* - edit : rows changed , but sorting may be affected . Requires full reload .
* - delete : just delete the given rows clientside ( no server interaction neccessary )
* - add : requires full reload for proper sorting
* @param { string } _msg_type 'error' , 'warning' or 'success' ( default )
* @param { string } _targetapp which app ' s window should be refreshed , default current
* @return { false | * } false to stop regular refresh , thought all observers are run
* /
observer ( _msg , _app , _id , _type , _msg_type , _targetapp )
{
switch ( _app )
{
case 'admin' :
// if iframe is used --> refresh it
var iframe_node = this . iframe ? this . iframe . getDOMNode ( ) : undefined ;
var iframe_url = iframe_node ? iframe_node.contentDocument.location.href : undefined ;
if ( _id && iframe_url != 'about:blank' )
{
var refresh_done = false ;
// Try for intelligent et2 refresh inside iframe
if ( iframe_node && iframe_node . contentWindow && iframe_node . contentWindow . etemplate2 )
{
var templates = iframe_node . contentWindow . etemplate2 . getByApplication ( 'admin' ) ;
for ( var i = 0 ; i < templates . length ; i ++ )
{
templates [ i ] . refresh ( _msg , _app , _id , _type ) ;
refresh_done = true ;
}
}
if ( ! refresh_done ) // --> reload iframe
{
this . load ( iframe_url ) ;
}
return false ; // --> no regular refresh
}
// invalidate client-side account-cache
this . egw . invalidate_account ( _id , _type ) ;
// group deleted, added or updated
if ( _id < 0 )
{
var tree = this . et2 . getWidgetById ( 'tree' ) ;
var nm = this . et2 . getWidgetById ( 'nm' ) ;
switch ( _type )
{
case 'delete' :
tree . deleteItem ( '/groups/' + _id , false ) ;
if ( nm ) nm . getInstanceManager ( ) . submit ( ) ;
break ;
default : // add, update, edit, null
if ( nm )
{
var activeFilters = nm . activeFilters ;
nm . getInstanceManager ( ) . submit ( ) ;
var nm = this . et2 . getWidgetById ( 'nm' ) ;
nm . applyFilters ( activeFilters ) ;
}
}
var refreshTree = this . et2 . getWidgetById ( 'tree' ) ;
if ( refreshTree ) refreshTree . refreshItem ( '/groups' ) ;
return false ; // --> no regular refresh
}
// not a user or group, eg. categories
else if ( ! _id )
{
// Try just refreshing the nextmatch
if ( this . nm )
{
this . nm . refresh ( ) ;
}
if ( this . groups )
{
this . groups . refresh ( ) ;
}
if ( this . et2 && this . et2 . getWidgetById ( 'nm' ) && this . nm !== this . et2 . getWidgetById ( 'nm' ) )
{
this . et2 . getWidgetById ( 'nm' ) . refresh ( ) ;
}
else
{
// Load something else
this . load ( ) ;
}
return false ; // --> no regular refresh needed
}
}
}
/ * *
* Hide navbar for idots template
*
* Just a hack for old idots , not neccesary for jdots
* /
_hide_navbar ( )
{
var document = this . iframe . getDOMNode ( ) . contentDocument ;
if ( ! document ) return ; // nothing we can do ...
// set white background, as transparent one lets account-list show through
document . getElementsByTagName ( 'body' ) [ 0 ] . style . backgroundColor = 'white' ;
// hide navbar elements
var ids2hide = [ 'divLogo' , 'topmenu' , 'divAppIconBar' , 'divStatusBar' , 'tdSidebox' , 'divAppboxHeader' ] ;
for ( var i = 0 ; i < ids2hide . length ; ++ i )
{
var elem = document . getElementById ( ids2hide [ i ] ) ;
if ( elem ) elem . style . display = 'none' ;
}
}
/ * *
* Set location of iframe for given _action and _sender ( row )
*
* @param _action
* @param _senders
* /
iframe_location ( _action , _senders )
{
var id = _senders [ 0 ] . id . split ( '::' ) ;
var url = _action . data . url . replace ( /(%24|\$)id/ , id [ 1 ] ) ;
this . load ( url ) ;
}
/ * *
* Callback to load an etemplate
*
* @param { Object [ ] } _data
* /
_ajax_load_callback ( _data )
{
if ( ! _data || _data . type != undefined ) return ;
// Insert the content, etemplate will load into it
jQuery ( this . ajax_target . node ) . append ( typeof _data === 'string' ? _data : _data [ 0 ] ) ;
}
/ * *
* Link hander for jDots template to just reload our iframe , instead of reloading whole admin app
*
* @param _url
* @return boolean true , if linkHandler took care of link , false otherwise
* /
linkHandler ( _url )
{
var matches = _url . match ( /menuaction=admin.admin_ui.index.*&load=([^&]+)/ ) ;
if ( _url != 'about:blank' && ( this . iframe != null && ! _url . match ( 'menuaction=admin.admin_ui.index' ) || matches ) )
{
if ( matches )
{
_url = _url . replace ( /menuaction=admin.admin_ui.index/ , 'menuaction=' + matches [ 1 ] ) . replace ( /&(load=[^&]+)/g , '' ) ;
}
this . load ( _url ) ;
return true ;
}
// can not load our own index page, has to be done by framework
return false ;
}
/ * *
* Run an admin module / onclick callback for tree
*
* @param { string } _id id of clicked node
* @param { et2_tree } _widget reference to tree widget
* /
run ( _id , _widget )
{
var link = _widget . getUserData ( _id , 'link' ) ;
this . groups . set_disabled ( true ) ;
if ( _id == '/accounts' || _id . substr ( 0 , 8 ) == '/groups/' )
{
this . load ( ) ;
var parts = _id . split ( '/' ) ;
this . nm . applyFilters ( { filter : parts [ 2 ] ? parts [ 2 ] : '' , search : '' } ) ;
}
else if ( _id === '/groups' )
{
this . load ( ) ;
this . group_list ( ) ;
}
else if ( typeof link == 'undefined' )
{
_widget . openItem ( _id , 'toggle' ) ;
}
else if ( link [ 0 ] == '/' || link . substr ( 0 , 4 ) == 'http' )
{
link += ( link . indexOf ( '?' ) >= 0 ? '&' : '?' ) + 'nonavbar=1' ;
this . load ( link ) ;
}
else if ( link . substr ( 0 , 11 ) == 'javascript:' )
{
eval ( link . substr ( 11 ) ) ;
}
}
/ * *
* Show the group list in the main window
* /
group_list ( )
{
this . nm . set_disabled ( true ) ;
this . groups . set_disabled ( false ) ;
}
/ * *
* View , edit or delete a group callback for tree
*
* @param { object } _action egwAction
* @param { array } _senders egwActionObject _senders [ 0 ] . id holds id
* /
group ( _action , _senders )
{
// Tree IDs look like /groups/ID, nm uses admin::ID
var from_nm = _senders [ 0 ] . id . indexOf ( '::' ) > 0 ;
var account_id = _senders [ 0 ] . id . split ( from_nm ? '::' : '/' ) [ from_nm ? 1 : 2 ] ;
switch ( _action . id )
{
case 'view' :
this . run ( from_nm ? '/groups/' + account_id : _senders [ 0 ] . id , this . et2 . getWidgetById ( 'tree' ) ) ;
break ;
case 'delete' :
this . egw . json ( 'admin_account::ajax_delete_group' , [ account_id , _action . data , this . et2 . _inst . etemplate_exec_id ] ) . sendRequest ( ) ;
break ;
default :
if ( ! _action . data . url )
{
alert ( 'Missing url in action ' + _action . id + '!' ) ;
break ;
}
var url = unescape ( _action . data . url ) . replace ( '$id' , account_id ) ;
if ( url [ 0 ] != '/' && url . substr ( 0 , 4 ) != 'http' )
{
url = this . egw . link ( '/index.php' , url ) ;
}
if ( _action . data . popup || _action . data . width && _action . data . height )
{
this . egw . open_link ( url , '_blank' , _action . data . popup ? _action.data.popup : _action.data.width + 'x' + _action . data . height ) ;
}
else
{
this . load ( url ) ;
}
break ;
}
}
/ * *
* Modify an ACL entry
*
* @param { object } _action egwAction
* @param { array } _senders egwActionObject _senders [ 0 ] . id holds the id "admin::app:account:location"
* /
acl ( _action , _senders )
{
var ids = [ ] ;
for ( var i = 0 ; i < _senders . length ; ++ i )
{
ids . push ( _senders [ i ] . id . split ( '::' ) . pop ( ) ) ; // remove "admin::" prefix
}
// For edit, set some data from the list since it's already there
var content = _senders [ 0 ] . id ? jQuery . extend ( { } , egw . dataGetUIDdata ( _senders [ 0 ] . id ) . data ) : { } ;
switch ( _action . id )
{
case 'delete' :
this . _acl_delete ( ids ) ;
break ;
case 'add' :
// No content, even if they clicked on a row
// Defaults set by _acl_content() based on nm values
content = { } ;
// Fall through
case 'edit' :
this . _acl_dialog ( content ) ;
break ;
}
}
_acl_delete ( ids )
{
var app = egw . app_name ( ) ; // can be either admin or preferences!
if ( app != 'admin' ) app = 'preferences' ;
var className = app + '_acl' ;
var callback = function ( _button_id , _value ) {
if ( _button_id != et2_dialog . OK_BUTTON ) return ;
var request = egw . json ( className + '::ajax_change_acl' , [ ids , null , _value , this . et2 . _inst . etemplate_exec_id ] , this . _acl_callback , this , false , this )
. sendRequest ( ) ;
} . bind ( this ) ;
var modifications : any = { } ;
var dialog_options = {
callback : callback ,
title : this.egw.lang ( 'Delete' ) ,
buttons : et2_dialog.BUTTONS_OK_CANCEL ,
value : {
content : { } ,
sel_options : { } ,
modifications : modifications ,
readonlys : { }
} ,
template : egw.webserverUrl + '/admin/templates/default/acl.delete.xet'
} ;
// Handle policy documentation tab here
if ( this . egw . user ( 'apps' ) . policy )
{
dialog_options [ 'width' ] = 550 ;
dialog_options [ 'height' ] = 350 ,
modifications . tabs = {
add_tabs : true ,
tabs : [ {
label : egw.lang ( 'Documentation' ) ,
template : 'policy.admin_cmd' ,
prepend : false
} ]
} ;
}
// Create the dialog
this . acl_dialog = et2_createWidget ( "dialog" , dialog_options , et2_dialog . _create_parent ( app ) ) ;
}
/ * *
* Create the ACL edit dialog , including defaults & fetching what can be found
*
* @param content List of content for the dialog template
* @param sel_options optional select options
* @param { etemplate2 } etemplate of etemplate that 'owns' the dialog
* @param { string } app Name of app
* @param { function } callback
* /
_acl_dialog ( content , sel_options ? , etemplate ? , app ? , callback? : Function )
{
if ( typeof content == 'undefined' ) content = { } ;
// Determine which application we're running as
app = app ? app : egw.app_name ( ) ;
// can be either admin or preferences!
if ( app != 'admin' ) app = 'preferences' ;
// Get by ID, since this.et2 isn't always the ACL list
var et2 = etemplate ? etemplate : etemplate2.getById ( 'admin-acl' ) . widgetContainer ;
var className = app + '_acl' ;
var acl_rights : any = { } ;
var readonlys : any = { acl : { } } ;
var modifications : any = { } ;
// Select options are already here, just pull them and pass along
sel_options = et2 . getArrayMgr ( 'sel_options' ) . data || { } ;
// Some defaults
if ( et2 && et2 . getWidgetById ( 'nm' ) )
{
// This is which checkboxes are available for each app
acl_rights = et2 . getWidgetById ( 'nm' ) . getArrayMgr ( 'content' ) . getEntry ( 'acl_rights' ) || { } ;
if ( ! content . acl_appname )
{
// Pre-set appname to currently selected
content . acl_appname = et2 . getWidgetById ( 'filter2' ) . getValue ( ) || "" ;
}
if ( ! content . acl_account )
{
content . acl_account = et2 . getWidgetById ( 'nm' ) . getArrayMgr ( 'content' ) . getEntry ( 'account_id' ) ;
}
if ( ! content . acl_location )
{
content . acl_location = et2 . getWidgetById ( 'filter' ) . getValue ( ) == 'run' ? 'run' : null ;
}
// If no admin rights, change UI to not allow adding access to apps
if ( content . acl_location == 'run' && ! egw . user ( 'apps' ) [ 'admin' ] )
{
content . acl_location = null ;
}
if ( content . acl_location == 'run' )
{
// These are the apps the account has access to
// Fetch current values from server
this . egw . json ( className + '::ajax_get_app_list' , [ content . acl_account ] , function ( data ) { content . apps = data ; } , this , false , this )
. sendRequest ( ) ;
}
else
{
// Restrict application selectbox to only apps that support ACL
sel_options . acl_appname = [ ] ;
for ( let app in acl_rights )
{
sel_options . acl_appname . push ( { value : app , label : this.egw.lang ( app ) } ) ;
}
// Sort list
sel_options . acl_appname . sort ( function ( a , b ) {
if ( a . label > b . label ) return 1 ;
if ( a . label < b . label ) return - 1 ;
return 0 ;
} ) ;
}
}
if ( content . acl_appname )
{
// Load checkboxes & their values
content . acl_rights = content . acl_rights ? parseInt ( content . acl_rights ) : null ;
jQuery . extend ( content , { acl : [ ] , right : [ ] , label : [ ] } ) ;
2020-04-07 19:48:52 +02:00
// Use this to make sure we get correct app translations
let app_egw = egw ( content . acl_appname , window ) ;
2020-02-09 12:26:23 +01:00
for ( var right in acl_rights [ content . acl_appname ] )
{
// only user himself is allowed to grant private (16) rights
if ( right == '16' && content [ 'acl_account' ] != egw . user ( 'account_id' ) )
{
readonlys . acl [ content . acl . length ] = true ;
}
content . acl . push ( content . acl_rights & parseInt ( right ) ) ;
content . right . push ( right ) ;
2020-04-07 19:48:52 +02:00
content . label . push ( app_egw . lang ( acl_rights [ content . acl_appname ] [ right ] ) ) ;
2020-02-09 12:26:23 +01:00
}
}
if ( content . acl_account && ! egw . user ( 'apps' ) [ 'admin' ] )
{
readonlys . acl_account = true ;
}
if ( content . acl_location )
{
sel_options . acl_location = jQuery . extend ( { } , sel_options . acl_location ) ;
this . egw . link_title ( 'api-accounts' , content . acl_location , function ( title ) { sel_options . acl_location [ content . acl_location ] = title ; } ) ;
}
var dialog_options = {
callback : jQuery.proxy ( function ( _button_id , _value ) {
this . acl_dialog = null ;
if ( _button_id != et2_dialog . OK_BUTTON ) return ;
// Restore account if it's readonly in dialog
if ( ! _value . acl_account ) _value . acl_account = content . acl_account ;
// Handle no applications selected
if ( typeof _value . apps == 'undefined' && content . acl_location == 'run' )
{
_value . apps = [ ] ;
}
// Only send the request if they entered everything (or selected no apps)
if ( _value . acl_account && ( _value . acl_appname && _value . acl_location || typeof _value . apps != 'undefined' ) )
{
var id : any = [ ] ;
if ( _value . acl_appname && _value . acl_account && _value . acl_location )
{
id = _value . acl_appname + ':' + _value . acl_account + ':' + _value . acl_location ;
if ( content && content . id && id != content . id )
{
// Changed the account or location, remove previous or we
// get a new line instead of an edit
this . egw . json ( className + '::ajax_change_acl' , [ content . id , 0 , [ ] , this . et2 . _inst . etemplate_exec_id ] , null , this , false , this )
. sendRequest ( ) ;
}
id = [ id ] ;
}
var rights = 0 ;
for ( var i in _value . acl )
{
rights += parseInt ( _value . acl [ i ] ) ;
}
if ( typeof _value . apps != 'undefined' && ! _value . acl_appname )
{
rights = 1 ;
var removed = [ ] ;
// Loop through all apps, remove the ones with no permission
for ( var idx in sel_options . filter2 )
{
var app = sel_options . filter2 [ idx ] . value || false ;
if ( ! app ) continue ;
var run_id = app + ":" + _value . acl_account + ":run" ;
if ( _value . apps . indexOf ( app ) < 0 && ( content . apps . indexOf ( app ) >= 0 || content . apps . length == 0 ) )
{
removed . push ( run_id ) ;
}
else if ( _value . apps . indexOf ( app ) >= 0 && content . apps . indexOf ( app ) < 0 )
{
id . push ( run_id ) ;
}
}
// Remove any removed
if ( removed . length > 0 )
{
this . egw . json ( className + '::ajax_change_acl' , [ removed , 0 , [ ] , this . et2 . _inst . etemplate_exec_id ] , callback ? callback : this._acl_callback , this , false , this )
. sendRequest ( ) ;
}
}
this . egw . json ( className + '::ajax_change_acl' , [ id , rights , _value , this . et2 . _inst . etemplate_exec_id ] , callback ? callback : this._acl_callback , this , false , this )
. sendRequest ( ) ;
}
} , this ) ,
title : this.egw.lang ( 'Access control' ) ,
buttons : et2_dialog.BUTTONS_OK_CANCEL ,
value : {
content : content ,
2020-08-11 15:48:18 +02:00
// @todo: we need to investigate more on et2_widget_selectbox type of apps
// where the sel options are not ready while setting its content. Therefore,
// the explicit apps should be removed after fixing it on the widget side.
sel_options : jQuery.extend ( sel_options , { apps : sel_options.filter2 } ) ,
2020-02-09 12:26:23 +01:00
modifications : modifications ,
readonlys : readonlys
} ,
template : egw.webserverUrl + '/admin/templates/default/acl.edit.xet'
} ;
// Handle policy documentation tab here
if ( this . egw . user ( 'apps' ) . policy )
{
dialog_options [ 'width' ] = 550 ;
dialog_options [ 'height' ] = 450 ,
modifications . tabs = {
add_tabs : true ,
tabs : [ {
label : egw.lang ( 'Documentation' ) ,
template : 'policy.admin_cmd' ,
prepend : false
} ]
} ;
}
// Create the dialog
this . acl_dialog = et2_createWidget ( "dialog" , dialog_options , et2_dialog . _create_parent ( app ) ) ;
}
/ * *
* Change handler for ACL edit dialog application selectbox .
* Re - creates the dialog with the current values
*
* @param input
* @param widget
* /
acl_reopen_dialog ( input , widget )
{
var content = { } ;
if ( this . acl_dialog != null )
{
content = this . acl_dialog . get_value ( ) || { } ;
// Destroy the dialog
2020-05-15 17:50:38 +02:00
this . acl_dialog . destroy ( ) ;
2020-02-09 12:26:23 +01:00
this . acl_dialog = null ;
}
// Re-open the dialog
this . _acl_dialog ( content ) ;
}
2020-04-07 19:48:52 +02:00
/ * *
* Load the new application ' s lang files when the app filter is changed
* /
acl_app_change ( event , nm )
{
let appname = nm . getWidgetById ( 'filter2' ) . getValue ( ) || '' ;
if ( appname )
{
let app_egw = egw ( appname ) ;
app_egw . langRequireApp ( window , appname ) ;
nm . getRoot ( ) . setApiInstance ( app_egw ) ;
}
}
2020-02-09 12:26:23 +01:00
/ * *
* Callback called on successfull call of serverside ACL handling
*
* @param { object } _data returned from server
* /
_acl_callback ( _data )
{
// Avoid the window / framework / app and just refresh the etemplate
// Framework will try to refresh the opener
// Get by ID, since this.et2 isn't always the ACL list
var et2 = etemplate2 . getById ( 'admin-acl' ) . widgetContainer ;
et2 . getInstanceManager ( ) . refresh ( _data . msg , this . appname , _data . ids , _data . type ) ;
}
/ * *
* Check to see if admin has taken away access to a category
*
* @ @param { widget } button add / apply pressed button
* /
check_owner ( button ) {
var select_owner = this . et2 . getWidgetById ( 'owner' ) ;
var diff = [ ] ;
if ( typeof select_owner != 'undefined' )
{
var owner = select_owner . get_value ( ) ;
}
if ( typeof owner != 'object' )
{
owner = [ owner ] ;
}
// No owner probably means selectbox is read-only, so no need to check
if ( owner == null ) return true ;
var all_users = owner . indexOf ( '0' ) >= 0 ;
// If they checked all users, uncheck the others
if ( all_users ) {
select_owner . set_value ( [ '0' ] ) ;
return true ;
}
// Find out what changed
var cat_original_owner = this . et2 . getArrayMgr ( 'content' ) . getEntry ( 'owner' ) ;
if ( cat_original_owner )
{
var selected_groups = select_owner . get_value ( ) . toString ( ) ;
for ( var i = 0 ; i < cat_original_owner . length ; i ++ )
{
if ( selected_groups . search ( cat_original_owner [ i ] ) < 0 )
{
diff . push ( cat_original_owner [ i ] ) ;
}
}
if ( diff . length > 0 )
{
var removed_cat_label = jQuery . map ( select_owner . options . select_options , function ( val , i )
{
for ( var j = 0 ; j <= diff . length ; j ++ )
{
if ( diff [ j ] == val . value )
{
return val . label ;
}
}
} ) ;
// Somebody will lose permission, give warning.
if ( removed_cat_label )
{
var msg = this . egw . lang ( 'Removing access for groups may cause problems for data in this category. Are you sure? Users in these groups may no longer have access:' ) ;
return et2_dialog . confirm ( button , msg + removed_cat_label . join ( ',' ) ) ;
}
}
}
return true ;
}
/ * *
* Show icon based on icon - selectbox , hide placeholder ( broken image ) , if no icon selected
*
* @param { widget } widget select box widget
* /
change_icon ( widget )
{
var img = widget . getRoot ( ) . getWidgetById ( 'icon_url' ) ;
if ( img )
{
img . set_src ( widget . getValue ( ) ) ;
}
}
/ * *
* Add / edit an account
*
* @param { object } _action egwAction
* @param { array } _senders egwActionObject _senders [ 0 ] . id holds account_id
* /
account ( _action , _senders )
{
var params = jQuery . extend ( { } , this . egw . link_get_registry ( 'addressbook' , 'edit' ) ) ;
var popup = < string > this . egw . link_get_registry ( 'addressbook' , 'edit_popup' ) ;
2020-03-03 00:00:58 +01:00
switch ( _action . id )
2020-02-09 12:26:23 +01:00
{
2020-03-03 00:00:58 +01:00
case 'add' :
params . owner = '0' ;
break ;
case 'copy' :
params . owner = '0' ;
params . copy = true ;
2020-03-23 16:57:06 +01:00
// Fall through
2020-03-03 00:00:58 +01:00
default :
params . account_id = _senders [ 0 ] . id . split ( '::' ) . pop ( ) ; // get last :: separated part
break ;
2020-02-09 12:26:23 +01:00
}
2020-03-03 00:00:58 +01:00
2020-03-23 16:57:06 +01:00
this . egw . open_link ( this . egw . link ( '/index.php' , params ) , 'admin' , popup , 'admin' ) ;
2020-02-09 12:26:23 +01:00
}
/ * *
* Submit statistic
*
* Webkit browsers ( Chrome , Safari , . . . ) do NOT allow to call form . submit ( ) from within onclick of a submit button .
* Therefor we first store our own form action , replace it with egroupware . org submit url and set a timeout calling
* submit_statistic again with just the form , to do the second submit to our own webserver
*
* @param { DOM } form
* @param { string } submit_url
* @return { boolean }
* /
submit_statistic ( form , submit_url )
{
var that = this ;
var submit = function ( )
{
// submit to egroupware.org
var method = form . method ;
form . method = 'POST' ;
var action = form . action ;
form . action = submit_url ;
var target = form . target ;
form . target = '_blank' ;
form . submit ( ) ;
// submit to own webserver
form . method = method ;
form . action = action ;
form . target = target ;
that . et2 . getInstanceManager ( ) . submit ( 'submit' ) ;
} ;
// Safari does NOT allow to call form.submit() outside of onclick callback
// so we have to use browsers ugly synchron confirm
if ( navigator . userAgent . match ( /Safari/ ) && ! navigator . userAgent . match ( /Chrome/ ) )
{
if ( confirm ( this . egw . lang ( 'Submit displayed information?' ) ) )
{
submit ( ) ;
}
}
else
{
et2_dialog . show_dialog ( function ( _button )
{
if ( _button == et2_dialog . YES_BUTTON )
{
submit ( ) ;
}
} , this . egw . lang ( 'Submit displayed information?' ) , '' , { } ,
et2_dialog . BUTTON_YES_NO , et2_dialog . QUESTION_MESSAGE , undefined , egw ) ;
}
return false ;
}
/ * *
* Change handler for when you change the type of a custom field .
* It toggles options / attributes as appropriate .
* @param { event object } e
* @param { widget object } widget
* /
cf_type_change ( e , widget )
{
var root = widget . getRoot ( ) ;
var attributes = widget . getArrayMgr ( 'content' ) . getEntry ( 'attributes[' + widget . getValue ( ) + ']' ) || { } ;
root . getWidgetById ( 'cf_values' ) . set_statustext ( widget . egw ( ) . lang ( widget . getArrayMgr ( 'content' ) . getEntry ( 'options[' + widget . getValue ( ) + ']' ) || '' ) ) ;
jQuery ( root . getWidgetById ( 'cf_len' ) . getDOMNode ( ) ) . toggle ( attributes . cf_len && true ) ;
jQuery ( root . getWidgetById ( 'cf_rows' ) . getDOMNode ( ) ) . toggle ( attributes . cf_rows && true ) ;
jQuery ( root . getWidgetById ( 'cf_values' ) . getParentDOMNode ( ) ) . toggle ( attributes . cf_values && true ) ;
}
/ * *
* Change handler for when you delete a custom app type
* If Policy app is available , it asks for documentation
* /
cf_type_delete ( e , widget )
{
var callback = function ( button , value )
{
if ( button === et2_dialog . YES_BUTTON )
{
var values = jQuery . extend (
{ } ,
this . getInstanceManager ( ) . getValues ( this . getRoot ( ) ) ,
value ,
{ appname : this.getRoot ( ) . getArrayMgr ( 'content' ) . getEntry ( 'content_types[appname]' ) }
) ;
egw . json ( 'admin.admin_customfields.ajax_delete_type' , [ values , this . getInstanceManager ( ) . etemplate_exec_id ] ) . sendRequest ( ) ;
// Immediately remove the type
var types = this . getRoot ( ) . getWidgetById ( 'types' ) ;
var options = types . options . select_options ;
var key ;
for ( key in options )
{
if ( options . hasOwnProperty ( key ) && key === types . getValue ( ) )
{
delete options [ key ] ;
break ;
}
}
types . set_select_options ( options ) ;
this . egw ( ) . message ( '' ) ;
// Trigger load of status for existing type
types . set_value ( Object . keys ( options ) [ 0 ] ) ;
}
} . bind ( widget ) ;
if ( egw . app ( 'policy' ) )
{
egw . includeJS ( egw . link ( '/policy/js/app.js' ) , function ( ) {
if ( typeof app . policy === 'undefined' || typeof app . policy . confirm === 'undefined' )
{
app . policy = new app . classes . policy ( ) ;
}
var parent = et2_dialog . _create_parent ( widget . egw ( ) ) ;
var dialog = et2_createWidget ( "dialog" , {
callback : callback ,
template : egw.link ( '/policy/templates/default/admin_cmd_narrow.xet' ) ,
title : egw.lang ( 'Delete' ) ,
buttons : et2_dialog.BUTTONS_YES_NO ,
value : { content : { } } ,
width : 'auto'
} , parent ) ;
dialog . egw ( ) . message ( "Entries with a deleted type can cause problems.\nCheck for entries with this type before deleting." , 'warning' ) ;
} ) ;
}
else
{
callback ( et2_dialog . YES_BUTTON ) ;
}
return false ;
}
/ * *
* Activate none standard SMTP mail accounts for selected users
*
* @param { egw_action } _action
* @param { array } _selected selected users
* /
emailadminActiveAccounts ( _action , _selected )
{
var menuaction = 'admin.admin_mail.ajax_activeAccounts' ;
var accounts = [ ] ;
var msg1 = egw . lang ( '%1 accounts being activated' , "" + Object . keys ( _selected ) . length ) ;
for ( var i = 0 ; i < Object . keys ( _selected ) . length ; i ++ )
{
accounts [ i ] = [ { id :_selected [ i ] [ 'id' ] . split ( '::' ) [ 1 ] , qouta : "" , domain : "" , status :_action.id == 'active' ? _action . id : '' } , this . et2 . _inst . etemplate_exec_id ] ;
}
var callbackDialog = function ( btn ) {
if ( btn === et2_dialog . YES_BUTTON )
{
// long task dialog for de/activation accounts
et2_dialog . long_task ( function ( _val , _resp ) {
if ( _val && _resp . type !== 'error' )
{
console . log ( _val , _resp ) ;
}
else
{
}
} , msg1 , 'Mail Acounts Activation' , menuaction , accounts , 'admin' ) ;
}
} ;
// confirmation dialog
et2_dialog . show_dialog ( callbackDialog , egw . lang ( 'Are you sure you want to %1 mail for selected accounts?' , egw . lang ( _action . id ) ) , egw . lang ( 'Active Mail Accounts' ) , { } ,
et2_dialog . BUTTON_YES_NO , et2_dialog . WARNING_MESSAGE , undefined , egw ) ;
}
/ * *
* No SSL
* /
SSL_NONE = 0 ;
/ * *
* STARTTLS on regular tcp connection / port
* /
SSL_STARTTLS = 1 ;
/ * *
* SSL ( inferior to TLS ! )
* /
SSL_SSL = 3 ;
/ * *
* require TLS version 1 + , no SSL version 2 or 3
* /
SSL_TLS = 2 ;
/ * *
* if set , verify certifcate ( currently not implemented in Horde_Imap_Client ! )
* /
SSL_VERIFY = 8 ;
/ * *
* Resize window methode
*
* @returns { undefined }
* /
wizard_popup_resize ( )
{
var $main_div = jQuery ( '#popupMainDiv' ) ;
var $et2 = jQuery ( '.et2_container' ) ;
var w = {
width : egw_getWindowInnerWidth ( ) ,
height : egw_getWindowInnerHeight ( )
} ;
// Use et2_container for width since #popupMainDiv is full width, but we still need
// to take padding/margin into account
var delta_width = w . width - ( $et2 . outerWidth ( true ) + ( $main_div . outerWidth ( true ) - $main_div . width ( ) ) ) ;
var delta_height = w . height - ( $et2 . outerHeight ( true ) + ( $main_div . outerHeight ( true ) - $main_div . height ( ) ) ) ;
if ( delta_width != 0 || delta_height != 0 )
{
window . resizeTo ( egw_getWindowOuterWidth ( ) - delta_width , egw_getWindowOuterHeight ( ) - delta_height ) ;
}
}
/ * *
* Switch account wizard to manual entry
* /
wizard_manual ( )
{
jQuery ( '.emailadmin_manual' ) . fadeToggle ( ) ; // not sure how to to this et2-isch
this . wizard_popup_resize ( ) ; // popup needs to be resized after toggling
}
/ * *
* onclick for continue button to show progress animation
*
* @param { object } _event event - object or information about event
* @param { et2_baseWidget } _widget widget causing the event
* /
wizard_detect ( _event , _widget )
{
// we need to do a manual asynchronious submit to show progress animation
// default synchronious submit stops animation!
if ( this . et2 . _inst . submit ( 'button[continue]' , true ) ) // true = async submit
{
var sieve_enabled = this . et2 . getWidgetById ( 'acc_sieve_enabled' ) ;
if ( ! sieve_enabled || sieve_enabled . get_value ( ) )
{
jQuery ( '#admin-mailwizard_output' ) . hide ( ) ;
jQuery ( 'td.emailadmin_progress' ) . show ( ) ;
}
}
return false ;
}
/ * *
* Set default port , if imap ssl - type changes
*
* @param { object } _event event - object or information about event
* @param { et2_baseWidget } _widget widget causing the event
* /
wizard_imap_ssl_onchange ( _event , _widget )
{
var ssl_type = _widget . get_value ( ) ;
this . et2 . getWidgetById ( 'acc_imap_port' ) . set_value (
ssl_type == this . SSL_SSL || ssl_type == this . SSL_TLS ? 993 : 143 ) ;
}
/ * *
* Set default port , if imap ssl - type changes
*
* @param { object } _event event - object or information about event
* @param { et2_baseWidget } _widget widget causing the event
* /
wizard_smtp_ssl_onchange ( _event , _widget )
{
var ssl_type = _widget . get_value ( ) ;
this . et2 . getWidgetById ( 'acc_smtp_port' ) . set_value (
ssl_type == 'no' ? 25 : ( ssl_type == this . SSL_SSL || ssl_type == this . SSL_TLS ? 465 : 587 ) ) ;
}
/ * *
* Set default port , if imap ssl - type changes
*
* @param { object } _event event - object or information about event
* @param { et2_baseWidget } _widget widget causing the event
* /
wizard_sieve_ssl_onchange ( _event , _widget )
{
var ssl_type = _widget . get_value ( ) ;
this . et2 . getWidgetById ( 'acc_sieve_port' ) . set_value (
ssl_type == this . SSL_SSL || ssl_type == this . SSL_TLS ? 5190 : 4190 ) ;
this . wizard_sieve_onchange ( _event , _widget ) ;
}
/ * *
* Enable sieve , if user changes some setting
*
* @param { object } _event event - object or information about event
* @param { et2_baseWidget } _widget widget causing the event
* /
wizard_sieve_onchange ( _event , _widget )
{
this . et2 . getWidgetById ( 'acc_sieve_enabled' ) . set_value ( 1 ) ;
}
/ * *
* Switch to select multiple accounts
*
* @param { object } _event event - object or information about event
* @param { et2_baseWidget } _widget widget causing the event
* /
edit_multiple ( _event , _widget )
{
// hide multiple button
_widget . set_disabled ( true ) ;
// switch account-selection to multiple
var account_id = this . et2 . getWidgetById ( 'account_id' ) ;
account_id . set_multiple ( true ) ;
}
/ * *
* Hide not applying fields , used as :
* - onchange handler on account_id
* - called from et2_ready for emailadmin . account template
*
* @param { object } _event event - object or information about event
* @param { et2_baseWidget } _widget widget causing the event
* /
account_hide_not_applying ( _event ? , _widget ? )
{
var account_id = this . et2 . getWidgetById ( 'account_id' ) ;
var ids = account_id && account_id . get_value ? account_id . get_value ( ) : [ ] ;
if ( typeof ids == 'string' ) ids = ids . split ( ',' ) ;
var multiple = ids . length >= 2 || ids [ 0 ] === '' || ids [ 0 ] < 0 ;
//alert('multiple='+(multiple?'true':'false')+': '+ids.join(','));
// initial call
if ( typeof _widget == 'undefined' )
{
if ( ! multiple )
{
jQuery ( '.emailadmin_no_single' ) . hide ( ) ;
}
if ( ! this . egw . user ( 'apps' ) . admin )
{
jQuery ( '.emailadmin_no_user,#button\\[multiple\\]' ) . hide ( ) ;
}
if ( ids . length == 1 )
{
// switch back to single selectbox
account_id . set_multiple ( false ) ;
this . et2 . getWidgetById ( 'button[multiple]' ) . set_disabled ( false ) ;
}
}
// switched to single user
else if ( ! multiple )
{
jQuery ( '.emailadmin_no_single' ) . fadeOut ( ) ;
// switch back to single selectbox
account_id . set_multiple ( false ) ;
this . et2 . getWidgetById ( 'button[multiple]' ) . set_disabled ( false ) ;
}
// switched to multiple user
else
{
jQuery ( '.emailadmin_no_single' ) . fadeIn ( ) ;
}
if ( _event && _event . stopPropagation ) _event . stopPropagation ( ) ;
return false ;
}
/ * *
* Callback if user changed account selction
*
* @param { object } _event event - object or information about event
* @param { et2_baseWidget } _widget widget causing the event
* /
change_account ( _event , _widget )
{
// todo check dirty and query user to a) save changes, b) discard changes, c) cancel selection
_widget . getInstanceManager ( ) . submit ( ) ;
}
/ * *
* Callback if user changes notification folders : unset use - default checkbox
*
* @param { object } _event
* @param { et2_widget } _widget
* /
change_folders ( _event , _widget )
{
var use_default = this . et2 . getWidgetById ( 'notify_use_default' ) ;
if ( use_default ) use_default . set_value ( false ) ;
}
/ * *
* default onExecute for admin actions
*
* @param { object } _action
* @param { object } _senders
* /
account_edit_action ( _action , _senders )
{
if ( _action . data . url )
{
this . egw . open_link ( _action . data . url , _action . data . target || '_blank' , _action . data . popup ) ;
}
}
/ * *
* Clear instance cache
*
* If there is an error on server - side , resend request with an parameter allowing
* cache to use different method not requiring eg . so much memory
* /
clear_cache ( )
{
2020-07-30 15:01:50 +02:00
let wait = this . egw . message ( this . egw . lang ( 'Clear cache and register hooks' ) + "\n" + this . egw . lang ( 'Please wait...' ) , 'info' ) ;
let success = function ( ) {
wait . close ( ) ;
egw . message ( 'Done' ) ;
} ;
this . egw . json ( 'admin.admin_hooks.ajax_clear_cache' , null , success ) . sendRequest ( true , undefined , jQuery . proxy ( function ( _xmlhttp , _err )
2020-02-09 12:26:23 +01:00
{
2020-07-30 15:01:50 +02:00
this . egw . json ( 'admin.admin_hooks.ajax_clear_cache&errored=1' , null , success ) . sendRequest ( true ) ;
2020-02-09 12:26:23 +01:00
} , this ) ) ;
}
/ * *
* Export content of given field into relevant file
* /
smime_exportCert ( )
{
var $a = jQuery ( document . createElement ( 'a' ) ) . appendTo ( 'body' ) . hide ( ) ;
var acc_id = this . et2 . getArrayMgr ( "content" ) . getEntry ( 'acc_id' ) ;
var url = window . egw . webserverUrl + '/index.php?' ;
url += 'menuaction=mail.mail_ui.smimeExportCert' ;
url += '&acc_id=' + acc_id ;
$a . prop ( 'href' , url ) ;
$a . prop ( 'download' , "" ) ;
$a [ 0 ] . click ( ) ;
$a . remove ( ) ;
}
/ * *
* Create certificate generator dialog
* /
smime_genCertificate ( )
{
var self = this ;
et2_createWidget ( "dialog" ,
{
callback ( _button_id , _value )
{
if ( _button_id == 'create' && _value )
{
var isValid = true ;
var required = [ 'countryName' , 'emailAddress' ] ;
var widget ;
// check the required fields
for ( var i = 0 ; i < required.length ; i + + )
{
if ( _value [ required [ i ] ] ) continue ;
widget = this . template . widgetContainer . getWidgetById ( required [ i ] ) ;
widget . set_validation_error ( 'This field is required!' ) ;
isValid = false ;
}
// check mismatch passphrase
if ( _value . passphrase && _value . passphrase !== _value . passphraseConf )
{
var passphraseConf = this . template . widgetContainer . getWidgetById ( 'passphrase' ) ;
passphraseConf . set_validation_error ( 'Confirm passphrase is not match!' ) ;
isValid = false ;
}
if ( isValid )
{
egw . json ( 'mail.mail_ui.ajax_smimeGenCertificate' , _value , function ( _cert ) {
if ( _cert )
{
for ( var key in _cert )
{
if ( ! _cert [ key ] ) continue ;
switch ( key )
{
case 'cert' :
self . et2 . getWidgetById ( 'smime_cert' ) . set_value ( _cert [ key ] ) ;
break ;
case 'privkey' :
self . et2 . getWidgetById ( 'acc_smime_password' ) . set_value ( _cert [ key ] ) ;
break ;
}
}
self . egw . message ( 'New certificate information has been generated, please save your account if you want to store it.' ) ;
}
} ) . sendRequest ( true ) ;
}
else
{
return false ;
}
}
} ,
title : egw.lang ( 'Generate Certificate' ) ,
buttons : [
{ text : this.egw.lang ( "Create" ) , id : "create" , "class" : "ui-priority-primary" , "default" : true } ,
{ text : this.egw.lang ( "Cancel" ) , id : "cancel" }
] ,
value : {
content : {
value : ''
} } ,
template : egw.webserverUrl + '/mail/templates/default/smimeCertGen.xet?' + Date . now ( ) ,
resizable : false ,
position : 'left top'
} , et2_dialog . _create_parent ( 'mail' ) ) ;
}
/ * *
* Triggers upload for background image and updates its taglist
*
* @param { type } node
* @param { type } widget
* /
login_background_update ( node , widget )
{
var taglist = widget . _parent . _children [ 0 ] ;
egw . json ( 'admin.admin_config.ajax_upload_anon_images' ,
[ widget . get_value ( ) , taglist . get_value ( ) ] ,
function ( _data ) {
taglist . set_value ( _data ) ;
} ) . sendRequest ( ) ;
}
/ * *
* Set content of selected row
*
* @param { array } node
* @returns
* /
cmds_onselect ( node )
{
var splitter = this . et2 . getWidgetById ( 'splitter' ) ;
var cmds_preview = this . et2 . getWidgetById ( 'cmds_preview' ) ;
if ( node . length != 1 )
{
splitter . dock ( ) ;
return ;
}
if ( splitter . isDocked ( ) )
{
splitter . undock ( ) ;
}
var data = egw . dataGetUIDdata ( node [ 0 ] ) ;
var policy_preview = this . et2 . getWidgetById ( 'policy_preview' ) ;
var id = node [ 0 ] . replace ( 'admin::' , '' ) ;
if ( app . policy )
{
cmds_preview . set_disabled ( true ) ;
policy_preview . set_src ( egw . link ( '/index.php' , {
menuaction : 'policy.EGroupware\\Policy\\History.view' ,
'cmd_id' : id ,
'cmd_template' : "policy.admin_cmd_history"
} ) ) ;
}
else
{
policy_preview . set_disabled ( true ) ;
cmds_preview . set_value ( { content : [ data . data ] } ) ;
}
}
}
app . classes . admin = AdminApp ;