2011-08-23 16:29:56 +02:00
|
|
|
/**
|
|
|
|
* EGroupware clientside API object
|
|
|
|
*
|
|
|
|
* @license http://opensource.org/licenses/gpl-license.php GPL - GNU General Public License
|
|
|
|
* @package etemplate
|
|
|
|
* @subpackage api
|
|
|
|
* @link http://www.egroupware.org
|
|
|
|
* @author Andreas Stöckel (as AT stylite.de)
|
|
|
|
* @author Ralf Becker <RalfBecker@outdoor-training.de>
|
|
|
|
* @version $Id$
|
|
|
|
*/
|
|
|
|
|
|
|
|
"use strict";
|
|
|
|
|
|
|
|
var egw;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Central object providing all kinds of api services on clientside:
|
2011-09-08 14:24:53 +02:00
|
|
|
* - preferences: egw.preferences(_name, _app)
|
2011-08-23 18:15:54 +02:00
|
|
|
* - translation: egw.lang("%1 entries deleted", 5)
|
2011-09-08 14:24:53 +02:00
|
|
|
* - link registry: egw.open(_id, _app), egw.link_get_registry(_app, _name), egw.link_app_list(_must_support)
|
|
|
|
* - configuration: egw.config(_name[, _app='phpgwapi'])
|
|
|
|
* - image urls: egw.image(_name[, _app='phpgwapi'])
|
|
|
|
* - user data: egw.user(_field)
|
|
|
|
* - user app data: egw.app(_app[, _name])
|
2011-08-23 16:29:56 +02:00
|
|
|
*/
|
|
|
|
if (window.opener && typeof window.opener.egw == 'object')
|
|
|
|
{
|
|
|
|
egw = window.opener.egw;
|
|
|
|
}
|
2011-08-26 09:31:18 +02:00
|
|
|
else if (window.top && typeof window.top.egw == 'object')
|
2011-08-23 16:29:56 +02:00
|
|
|
{
|
|
|
|
egw = window.top.egw;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
egw = {
|
|
|
|
/**
|
|
|
|
* Object holding the prefences as 2-dim. associative array, use egw.preference(name[,app]) to access it
|
2011-08-23 18:15:54 +02:00
|
|
|
*
|
|
|
|
* @access: private, use egw.preferences() or egw.set_perferences()
|
2011-08-23 16:29:56 +02:00
|
|
|
*/
|
|
|
|
prefs: {
|
|
|
|
common: {
|
|
|
|
dateformat: "Y-m-d",
|
|
|
|
timeformat: 24,
|
|
|
|
lang: "en"
|
|
|
|
}
|
|
|
|
},
|
2011-08-23 18:15:54 +02:00
|
|
|
|
|
|
|
/**
|
|
|
|
* base-URL of the EGroupware installation
|
|
|
|
*
|
|
|
|
* get set via egw_framework::header()
|
|
|
|
*/
|
|
|
|
webserverUrl: "/egroupware",
|
2011-08-23 16:29:56 +02:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Setting prefs for an app or 'common'
|
|
|
|
*
|
|
|
|
* @param object _data object with name: value pairs to set
|
|
|
|
* @param string _app application name, 'common' or undefined to prefes of all apps at once
|
|
|
|
*/
|
|
|
|
set_preferences: function(_data, _app)
|
|
|
|
{
|
|
|
|
if (typeof _app == 'undefined')
|
|
|
|
{
|
|
|
|
this.prefs = _data;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
this.prefs[_app] = _data;
|
|
|
|
}
|
|
|
|
},
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Query an EGroupware user preference
|
|
|
|
*
|
2011-08-26 11:34:18 +02:00
|
|
|
* If a prefernce is not already loaded (only done for "common" by default), it is synchroniosly queryed from the server!
|
|
|
|
*
|
2011-08-23 16:29:56 +02:00
|
|
|
* @param string _name name of the preference, eg. 'dateformat'
|
|
|
|
* @param string _app='common'
|
|
|
|
* @return string preference value
|
2011-08-26 11:34:18 +02:00
|
|
|
* @todo add a callback to query it asynchron
|
2011-08-23 16:29:56 +02:00
|
|
|
*/
|
|
|
|
preference: function(_name, _app)
|
|
|
|
{
|
2011-08-23 18:15:54 +02:00
|
|
|
if (typeof _app == 'undefined') _app = 'common';
|
2011-08-23 16:29:56 +02:00
|
|
|
|
|
|
|
if (typeof this.prefs[_app] == 'undefined')
|
|
|
|
{
|
2011-08-26 11:34:18 +02:00
|
|
|
xajax_doXMLHTTPsync('home.egw_framework.ajax_get_preference.template', _app);
|
|
|
|
|
|
|
|
if (typeof this.prefs[_app] == 'undefined') this.prefs[_app] = {};
|
2011-08-23 16:29:56 +02:00
|
|
|
}
|
|
|
|
return this.prefs[_app][_name];
|
|
|
|
},
|
|
|
|
|
2011-08-26 11:34:18 +02:00
|
|
|
/**
|
|
|
|
* Set a preference and sends it to the server
|
|
|
|
*
|
|
|
|
* Server will silently ignore setting preferences, if user has no right to do so!
|
|
|
|
*
|
|
|
|
* @param string _app application name or "common"
|
|
|
|
* @param string _name name of the pref
|
|
|
|
* @param string _val value of the pref
|
|
|
|
*/
|
|
|
|
set_preference: function(_app, _name, _val)
|
|
|
|
{
|
|
|
|
xajax_doXMLHTTP('home.egw_framework.ajax_set_preference.template', _app, _name, _val);
|
|
|
|
|
|
|
|
// update own preference cache, if _app prefs are loaded (dont update otherwise, as it would block loading of other _app prefs!)
|
|
|
|
if (typeof this.prefs[_app] != 'undefined') this.prefs[_app][_name] = _val;
|
|
|
|
},
|
|
|
|
|
2011-08-23 16:29:56 +02:00
|
|
|
/**
|
|
|
|
* Translations
|
2011-08-23 18:15:54 +02:00
|
|
|
*
|
|
|
|
* @access: private, use egw.lang() or egw.set_lang_arr()
|
2011-08-23 16:29:56 +02:00
|
|
|
*/
|
|
|
|
lang_arr: {},
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Set translation for a given application
|
|
|
|
*
|
|
|
|
* @param string _app
|
|
|
|
* @param object _message message => translation pairs
|
|
|
|
*/
|
|
|
|
set_lang_arr: function(_app, _messages)
|
|
|
|
{
|
|
|
|
this.lang_arr[_app] = _messages;
|
|
|
|
},
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Translate a given phrase replacing optional placeholders
|
|
|
|
*
|
|
|
|
* @param string _msg message to translate
|
|
|
|
* @param string _arg1 ... _argN
|
|
|
|
*/
|
|
|
|
lang: function(_msg, _arg1)
|
|
|
|
{
|
2011-08-23 18:15:54 +02:00
|
|
|
var translation = _msg;
|
|
|
|
_msg = _msg.toLowerCase();
|
|
|
|
|
|
|
|
// search apps in given order for a replacement
|
|
|
|
var apps = [window.egw_appName, 'etemplate', 'common'];
|
|
|
|
for(var i = 0; i < apps.length; ++i)
|
|
|
|
{
|
2011-08-23 19:11:45 +02:00
|
|
|
if (typeof this.lang_arr[apps[i]] != "undefined" &&
|
|
|
|
typeof this.lang_arr[apps[i]][_msg] != 'undefined')
|
2011-08-23 18:15:54 +02:00
|
|
|
{
|
|
|
|
translation = this.lang_arr[apps[i]][_msg];
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (arguments.length == 1) return translation;
|
|
|
|
|
|
|
|
if (arguments.length == 2) return translation.replace('%1', arguments[1]);
|
|
|
|
|
|
|
|
// to cope with arguments containing '%2' (eg. an urlencoded path like a referer),
|
|
|
|
// we first replace all placeholders '%N' with '|%N|' and then we replace all '|%N|' with arguments[N]
|
|
|
|
translation = translation.replace(/%([0-9]+)/g, '|%$1|');
|
|
|
|
for(var i = 1; i < arguments.length; ++i)
|
|
|
|
{
|
|
|
|
translation = translation.replace('|%'+i+'|', arguments[i]);
|
|
|
|
}
|
|
|
|
return translation;
|
2011-08-23 16:29:56 +02:00
|
|
|
},
|
|
|
|
|
|
|
|
/**
|
|
|
|
* View an EGroupware entry: opens a popup of correct size or redirects window.location to requested url
|
|
|
|
*
|
|
|
|
* Examples:
|
|
|
|
* - egw.open(123,'infolog') or egw.open('infolog:123') opens popup to edit or view (if no edit rights) infolog entry 123
|
|
|
|
* - egw.open('infolog:123','timesheet','add') opens popup to add new timesheet linked to infolog entry 123
|
|
|
|
* - egw.open(123,'addressbook','view') opens addressbook view for entry 123 (showing linked infologs)
|
|
|
|
* - egw.open('','addressbook','view_list',{ search: 'Becker' }) opens list of addresses containing 'Becker'
|
|
|
|
*
|
|
|
|
* @param string|int id either just the id or "app:id" if app==""
|
|
|
|
* @param string app app-name or empty (app is part of id)
|
|
|
|
* @param string type default "edit", possible "view", "view_list", "edit" (falls back to "view") and "add"
|
|
|
|
* @param object|string extra extra url parameters to append as object or string
|
|
|
|
* @param string target target of window to open
|
|
|
|
*/
|
|
|
|
open: function(id, app, type, extra, target)
|
|
|
|
{
|
|
|
|
if (typeof this.link_registry != 'object')
|
|
|
|
{
|
|
|
|
alert('egw.open() link registry is NOT defined!');
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
if (!app)
|
|
|
|
{
|
|
|
|
var app_id = id.split(':',2);
|
|
|
|
app = app_id[0];
|
|
|
|
id = app_id[1];
|
|
|
|
}
|
|
|
|
if (!app || typeof this.link_registry[app] != 'object')
|
|
|
|
{
|
|
|
|
alert('egw.open() app "'+app+'" NOT defined in link registry!');
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
var app_registry = this.link_registry[app];
|
|
|
|
if (typeof type == 'undefined') type = 'edit';
|
|
|
|
if (type == 'edit' && typeof app_registry.edit == 'undefined') type = 'view';
|
|
|
|
if (typeof app_registry[type] == 'undefined')
|
|
|
|
{
|
|
|
|
alert('egw.open() type "'+type+'" is NOT defined in link registry for app "'+app+'"!');
|
|
|
|
return;
|
|
|
|
}
|
2011-08-23 18:15:54 +02:00
|
|
|
var url = this.webserverUrl+'/index.php';
|
2011-08-23 16:29:56 +02:00
|
|
|
var delimiter = '?';
|
|
|
|
var params = app_registry[type];
|
|
|
|
if (type == 'view' || type == 'edit') // add id parameter for type view or edit
|
|
|
|
{
|
|
|
|
params[app_registry[type+'_id']] = id;
|
|
|
|
}
|
|
|
|
else if (type == 'add' && id) // add add_app and app_id parameters, if given for add
|
|
|
|
{
|
|
|
|
var app_id = id.split(':',2);
|
|
|
|
params[app_registry.add_app] = app_id[0];
|
|
|
|
params[app_registry.add_id] = app_id[1];
|
|
|
|
}
|
|
|
|
for(var attr in params)
|
|
|
|
{
|
|
|
|
url += delimiter+attr+'='+encodeURIComponent(params[attr]);
|
|
|
|
delimiter = '&';
|
|
|
|
}
|
|
|
|
if (typeof extra == 'object')
|
|
|
|
{
|
|
|
|
for(var attr in extra)
|
|
|
|
{
|
|
|
|
url += delimiter+attr+'='+encodeURIComponent(extra[attr]);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else if (typeof extra == 'string')
|
|
|
|
{
|
|
|
|
url += delimiter + extra;
|
|
|
|
}
|
|
|
|
if (typeof app_registry[type+'_popup'] == 'undefined')
|
|
|
|
{
|
|
|
|
if (target)
|
|
|
|
{
|
|
|
|
window.open(url, target);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
egw_appWindowOpen(app, url);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
var w_h = app_registry[type+'_popup'].split('x');
|
|
|
|
if (w_h[1] == 'egw_getWindowOuterHeight()') w_h[1] = egw_getWindowOuterHeight();
|
|
|
|
egw_openWindowCentered2(url, target, w_h[0], w_h[1], 'yes', app, false);
|
|
|
|
}
|
|
|
|
},
|
|
|
|
|
2011-09-08 14:24:53 +02:00
|
|
|
/**
|
|
|
|
* Check if $app is in the registry and has an entry for $name
|
|
|
|
*
|
|
|
|
* @param string $app app-name
|
|
|
|
* @param string $name name / key in the registry, eg. 'view'
|
|
|
|
* @return boolean|string false if $app is not registered, otherwise string with the value for $name
|
|
|
|
*/
|
|
|
|
link_get_registry: function(_app, _name)
|
|
|
|
{
|
|
|
|
if (typeof this.link_registry[_app] == 'undefined')
|
|
|
|
{
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
var reg = this.link_registry[_app];
|
|
|
|
|
|
|
|
// some defaults (we set them directly in the registry, to do this only once)
|
|
|
|
if (typeof reg[_name] == 'undefined')
|
|
|
|
{
|
|
|
|
switch(_name)
|
|
|
|
{
|
|
|
|
case 'name':
|
|
|
|
reg.name = _app;
|
|
|
|
break;
|
|
|
|
case 'icon':
|
|
|
|
var app_data = this.app(_app);
|
|
|
|
if (typeof app_data != 'undefined' && typeof app_data.icon != 'undefined')
|
|
|
|
{
|
|
|
|
reg.icon = (typeof app_data.icon_app != 'undefined' ? app_data.icon_app : _app)+'/'+app_data.icon;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
reg.icon = _app+'/navbar';
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return typeof reg[_name] == 'undefined' ? false : reg[_name];
|
|
|
|
},
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Get list of link-aware apps the user has rights to use
|
|
|
|
*
|
|
|
|
* @param string $must_support capability the apps need to support, eg. 'add', default ''=list all apps
|
|
|
|
* @return array with app => title pairs
|
|
|
|
*/
|
|
|
|
link_app_list: function(_must_support)
|
|
|
|
{
|
|
|
|
var apps = [];
|
|
|
|
for (var type in this.link_registry)
|
|
|
|
{
|
|
|
|
var reg = this.link_registry[type];
|
|
|
|
|
|
|
|
if (typeof _must_support != 'undefined' && _must_support && typeof reg[_must_support] == 'undefined') continue;
|
|
|
|
|
|
|
|
var app_sub = type.split('-');
|
|
|
|
if (this.app(app_sub[0]))
|
|
|
|
{
|
|
|
|
apps.push({"type": type, "label": this.lang(this.link_get_registry(type,'name'))});
|
|
|
|
}
|
|
|
|
}
|
|
|
|
// sort labels (caseinsensitive) alphabetic
|
|
|
|
apps = apps.sort(function(_a,_b) {
|
|
|
|
var al = _a.label.toUpperCase();
|
|
|
|
var bl = _b.label.toUpperCase();
|
|
|
|
return al == bl ? 0 : (al > bl ? 1 : -1);
|
|
|
|
});
|
|
|
|
// create sorted associative array / object
|
|
|
|
var sorted = {};
|
|
|
|
for(var i = 0; i < apps.length; ++i)
|
|
|
|
{
|
|
|
|
sorted[apps[i].type] = apps[i].label;
|
|
|
|
}
|
|
|
|
return sorted;
|
|
|
|
},
|
|
|
|
|
2011-08-23 16:29:56 +02:00
|
|
|
/**
|
|
|
|
* Link registry
|
2011-08-23 18:15:54 +02:00
|
|
|
*
|
|
|
|
* @access: private, use egw.open() or egw.set_link_registry()
|
2011-08-23 16:29:56 +02:00
|
|
|
*/
|
|
|
|
link_registry: null,
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Set link registry
|
|
|
|
*
|
|
|
|
* @param object _registry whole registry or entries for just one app
|
|
|
|
* @param string _app
|
|
|
|
*/
|
|
|
|
set_link_registry: function (_registry, _app)
|
|
|
|
{
|
|
|
|
if (typeof _app == 'undefined')
|
|
|
|
{
|
|
|
|
this.link_registry = _registry;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
this.link_registry[_app] = _registry;
|
|
|
|
}
|
2011-08-28 10:28:53 +02:00
|
|
|
},
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Clientside config
|
|
|
|
*
|
|
|
|
* @access: private, use egw.config(_name, _app="phpgwapi")
|
|
|
|
*/
|
2011-08-30 11:15:00 +02:00
|
|
|
configs: {},
|
2011-08-28 10:28:53 +02:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Query clientside config
|
|
|
|
*
|
|
|
|
* @param string _name name of config variable
|
|
|
|
* @param string _app default "phpgwapi"
|
|
|
|
* @return mixed
|
|
|
|
*/
|
|
|
|
config: function (_name, _app)
|
|
|
|
{
|
|
|
|
if (typeof _app == 'undefined') _app = 'phpgwapi';
|
|
|
|
|
|
|
|
if (typeof this.configs[_app] == 'undefined') return null;
|
|
|
|
|
|
|
|
return this.configs[_app][_name];
|
|
|
|
},
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Set clientside configuration for all apps
|
|
|
|
*
|
|
|
|
* @param array/object
|
|
|
|
*/
|
|
|
|
set_configs: function(_configs)
|
|
|
|
{
|
|
|
|
this.configs = _configs;
|
2011-08-31 00:19:38 +02:00
|
|
|
},
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Map to serverside available images for users template-set
|
2011-08-31 14:17:34 +02:00
|
|
|
*
|
|
|
|
* @access: private, use egw.image(_name, _app)
|
2011-08-31 00:19:38 +02:00
|
|
|
*/
|
|
|
|
images: {},
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Set imagemap, called from /phpgwapi/images.php
|
|
|
|
*
|
|
|
|
* @param array/object _images
|
|
|
|
*/
|
|
|
|
set_images: function (_images)
|
|
|
|
{
|
|
|
|
this.images = _images;
|
|
|
|
},
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Get image URL for a given image-name and application
|
|
|
|
*
|
|
|
|
* @param string _name image-name without extension
|
2011-08-31 08:26:05 +02:00
|
|
|
* @param string _app application name, default current app of window
|
2011-08-31 00:19:38 +02:00
|
|
|
* @return string with URL of image
|
|
|
|
*/
|
|
|
|
image: function (_name, _app)
|
|
|
|
{
|
2011-08-31 08:26:05 +02:00
|
|
|
if (typeof _app == 'undefined') _app = this.getAppName();
|
2011-08-31 00:19:38 +02:00
|
|
|
|
|
|
|
// own instance specific images in vfs have highest precedence
|
|
|
|
if (typeof this.images['vfs'] != 'undefined' && typeof this.images['vfs'][_name] != 'undefined')
|
|
|
|
{
|
|
|
|
return this.webserverUrl+this.images['vfs'][_name];
|
|
|
|
}
|
|
|
|
if (typeof this.images[_app] != 'undefined' && typeof this.images[_app][_name] != 'undefined')
|
|
|
|
{
|
|
|
|
return this.webserverUrl+this.images[_app][_name];
|
|
|
|
}
|
|
|
|
if (typeof this.images['phpgwapi'] != 'undefined' && typeof this.images['phpgwapi'][_name] != 'undefined')
|
|
|
|
{
|
|
|
|
return this.webserverUrl+this.images['vfs'][_name];
|
|
|
|
}
|
2011-08-31 08:26:05 +02:00
|
|
|
// if no match, check if it might contain an extension
|
|
|
|
if (_name.match(/\.(png|gif|jpg)$/i))
|
|
|
|
{
|
|
|
|
return this.image(_name.replace(/.(png|gif|jpg)$/i,''), _app);
|
|
|
|
}
|
2011-08-31 00:19:38 +02:00
|
|
|
console.log('egw.image("'+_name+'", "'+_app+'") image NOT found!');
|
|
|
|
return null;
|
2011-08-31 08:26:05 +02:00
|
|
|
},
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Returns the name of the currently active application
|
|
|
|
*
|
|
|
|
* @ToDo: fixme: does not work, as egw object runs in framework for jdots
|
|
|
|
*/
|
|
|
|
getAppName: function ()
|
|
|
|
{
|
|
|
|
if (typeof egw_appName == 'undefined')
|
|
|
|
{
|
|
|
|
return 'egroupware';
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
return egw_appName;
|
|
|
|
}
|
2011-08-31 14:17:34 +02:00
|
|
|
},
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Data about current user
|
|
|
|
*
|
2011-09-08 14:24:53 +02:00
|
|
|
* @access: private, use egw.user(_field) or egw.app(_app)
|
2011-08-31 14:17:34 +02:00
|
|
|
*/
|
|
|
|
userData: {},
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Set data of current user
|
|
|
|
*
|
|
|
|
* @param object _data
|
|
|
|
*/
|
|
|
|
set_user: function(_data)
|
|
|
|
{
|
|
|
|
this.userData = _data;
|
|
|
|
},
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Get data about current user
|
|
|
|
*
|
|
|
|
* @param string _field
|
|
|
|
* - 'account_id','account_lid','person_id','account_status',
|
|
|
|
* - 'account_firstname','account_lastname','account_email','account_fullname','account_phone'
|
2011-09-08 14:24:53 +02:00
|
|
|
* - 'apps': object with app => data pairs the user has run-rights for
|
|
|
|
* @return string|array|null
|
2011-08-31 14:17:34 +02:00
|
|
|
*/
|
|
|
|
user: function (_field)
|
|
|
|
{
|
|
|
|
return this.userData[_field];
|
2011-09-08 14:24:53 +02:00
|
|
|
},
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Return data of apps the user has rights to run
|
|
|
|
*
|
|
|
|
* Can be used the check of run rights like: if (egw.app('addressbook')) { do something if user has addressbook rights }
|
|
|
|
*
|
|
|
|
* @param string _app
|
|
|
|
* @param string _name attribute to return, default return whole app-data-object
|
|
|
|
* @return object|string|null null if not found
|
|
|
|
*/
|
|
|
|
app: function(_app, _name)
|
|
|
|
{
|
|
|
|
return typeof _name == 'undefined' || typeof this.userData.apps[_app] == 'undefined' ?
|
|
|
|
this.userData.apps[_app] : this.userData.apps[_app][_name];
|
2011-08-23 16:29:56 +02:00
|
|
|
}
|
|
|
|
};
|
2011-08-23 19:11:45 +02:00
|
|
|
}
|