2022-07-12 17:14:44 +02:00
|
|
|
/**
|
|
|
|
* EGroupware eTemplate2 - Avatar widget
|
|
|
|
*
|
|
|
|
* @license http://opensource.org/licenses/gpl-license.php GPL - GNU General Public License
|
|
|
|
* @package etemplate
|
|
|
|
* @subpackage api
|
|
|
|
* @link https://www.egroupware.org
|
|
|
|
* @author Hadi Nategh
|
|
|
|
*/
|
|
|
|
|
|
|
|
import {Et2Widget} from "../Et2Widget/Et2Widget";
|
2023-09-13 19:55:33 +02:00
|
|
|
import {css} from "lit";
|
2022-07-12 17:14:44 +02:00
|
|
|
import {SlAvatar} from "@shoelace-style/shoelace";
|
|
|
|
import {et2_IDetachedDOM} from "../et2_core_interfaces";
|
|
|
|
import {egw} from "../../jsapi/egw_global";
|
|
|
|
import shoelace from "../Styles/shoelace";
|
2022-07-14 17:32:45 +02:00
|
|
|
import {Et2Dialog} from "../Et2Dialog/Et2Dialog";
|
|
|
|
import "../../../../vendor/bower-asset/cropper/dist/cropper.min.js";
|
|
|
|
import {cropperStyles} from "./cropperStyles";
|
2022-07-12 17:14:44 +02:00
|
|
|
|
2023-09-27 22:29:19 +02:00
|
|
|
export class Et2Avatar extends Et2Widget(SlAvatar) implements et2_IDetachedDOM
|
2022-07-12 17:14:44 +02:00
|
|
|
{
|
2022-07-21 17:57:50 +02:00
|
|
|
private _contactId;
|
2022-07-19 12:18:42 +02:00
|
|
|
private _delBtn: HTMLElement;
|
|
|
|
private _editBtn : HTMLElement;
|
|
|
|
|
2022-07-12 17:14:44 +02:00
|
|
|
static get styles()
|
|
|
|
{
|
|
|
|
return [
|
|
|
|
...super.styles,
|
|
|
|
shoelace,
|
2022-07-14 17:32:45 +02:00
|
|
|
cropperStyles,
|
2022-07-12 17:14:44 +02:00
|
|
|
css`
|
2022-07-18 16:32:57 +02:00
|
|
|
:host::part(edit) {
|
|
|
|
visibility: hidden;
|
|
|
|
border-radius: 50%;
|
|
|
|
margin: -4px;
|
2022-09-21 11:15:14 +02:00
|
|
|
z-index: 1;
|
2022-07-18 16:32:57 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
:host(:hover)::part(edit) {
|
|
|
|
visibility: visible;
|
|
|
|
}
|
2022-07-12 17:14:44 +02:00
|
|
|
`
|
|
|
|
];
|
|
|
|
}
|
|
|
|
|
|
|
|
static get properties()
|
|
|
|
{
|
|
|
|
return {
|
|
|
|
...super.properties,
|
|
|
|
/**
|
|
|
|
* The label of the image
|
|
|
|
* Actually not used as label, but we put it as title
|
|
|
|
* Added here as there's no Lion parent
|
|
|
|
*/
|
|
|
|
label: {
|
|
|
|
type: String
|
|
|
|
},
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Contact id should be either user account_id {account:number} or contact_id {contact:number or number}
|
|
|
|
*/
|
2023-06-27 17:46:30 +02:00
|
|
|
contactId: {type: String, noAccessor: true},
|
2022-07-12 17:14:44 +02:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Image
|
|
|
|
* Displayed image
|
2022-07-19 12:18:42 +02:00
|
|
|
* @deprecated
|
2022-07-12 17:14:44 +02:00
|
|
|
*/
|
|
|
|
src: {type: String},
|
|
|
|
|
2023-02-28 13:06:52 +01:00
|
|
|
/**
|
|
|
|
* The shape of the avatar
|
|
|
|
* circle | square | rounded
|
|
|
|
*/
|
2022-07-12 17:14:44 +02:00
|
|
|
shape: {
|
|
|
|
type: String,
|
|
|
|
reflect: true
|
|
|
|
},
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Make avatar widget editable to be able to crop profile picture or upload a new photo
|
|
|
|
*/
|
|
|
|
editable: {type: Boolean},
|
|
|
|
|
|
|
|
image: {
|
|
|
|
type: String,
|
|
|
|
reflect: true
|
2022-07-14 17:32:45 +02:00
|
|
|
},
|
2022-07-12 17:14:44 +02:00
|
|
|
|
2022-07-14 17:32:45 +02:00
|
|
|
crop: {type: Boolean},
|
2022-07-12 17:14:44 +02:00
|
|
|
|
2022-07-14 17:32:45 +02:00
|
|
|
size: {type: String}
|
|
|
|
}
|
|
|
|
}
|
2022-07-12 17:14:44 +02:00
|
|
|
|
|
|
|
constructor()
|
|
|
|
{
|
|
|
|
super();
|
|
|
|
this.src = "";
|
|
|
|
this.label = "";
|
2022-07-21 17:57:50 +02:00
|
|
|
this.contactId = "";
|
2022-07-14 17:32:45 +02:00
|
|
|
this.editable = false;
|
|
|
|
this.crop = false;
|
2022-07-15 13:35:23 +02:00
|
|
|
this.size = "2.7em";
|
2022-07-28 15:09:19 +02:00
|
|
|
this.icon = "";
|
2022-08-24 11:21:15 +02:00
|
|
|
this.shape = "rounded";
|
2022-07-14 17:32:45 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Handle changes that have to happen based on changes to properties
|
|
|
|
*
|
|
|
|
*/
|
|
|
|
updated(changedProperties)
|
|
|
|
{
|
|
|
|
super.updated(changedProperties);
|
|
|
|
|
|
|
|
if (changedProperties.has("crop")) {
|
|
|
|
if (this.crop && !this.readonly && this._imageNode)
|
|
|
|
{
|
|
|
|
jQuery(this._imageNode).cropper({aspectRatio: 1/1});
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (changedProperties.has("size"))
|
|
|
|
{
|
2022-08-15 23:16:23 +02:00
|
|
|
if(this.size)
|
|
|
|
{
|
|
|
|
this.getDOMNode().setAttribute('style', `--size:${this.size}`);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
this.style.removeProperty("--size")
|
|
|
|
}
|
2022-07-14 17:32:45 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
firstUpdated()
|
|
|
|
{
|
|
|
|
let self = this;
|
2022-07-21 17:57:50 +02:00
|
|
|
if (this.contactId && this.editable)
|
2022-07-14 17:32:45 +02:00
|
|
|
{
|
|
|
|
egw(window).json(
|
|
|
|
'addressbook.addressbook_ui.ajax_noPhotoExists',
|
2022-07-21 17:57:50 +02:00
|
|
|
[this.contactId],
|
2022-07-14 17:32:45 +02:00
|
|
|
function(noPhotoExists)
|
|
|
|
{
|
|
|
|
if (noPhotoExists) self.image="";
|
|
|
|
self._buildEditableLayer(noPhotoExists);
|
|
|
|
}
|
|
|
|
).sendRequest(true);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2022-07-21 17:57:50 +02:00
|
|
|
get contactId()
|
2022-07-14 17:32:45 +02:00
|
|
|
{
|
2022-07-21 17:57:50 +02:00
|
|
|
return this._contactId;
|
2022-07-12 17:14:44 +02:00
|
|
|
}
|
|
|
|
|
2023-06-27 08:59:43 +02:00
|
|
|
static RFC822EMAIL = /<([^<>]+)>$/;
|
|
|
|
|
2022-07-12 17:14:44 +02:00
|
|
|
/**
|
2022-07-21 17:57:50 +02:00
|
|
|
* Function to set contactId
|
|
|
|
* contactId could be in one of these formats:
|
2023-06-27 08:59:43 +02:00
|
|
|
* 'number', will be considered as contact_id
|
2022-07-12 17:14:44 +02:00
|
|
|
* 'contact:number', similar to above
|
2023-06-27 08:59:43 +02:00
|
|
|
* 'account:number', will be considered as account id
|
|
|
|
* 'email:<email>', will be considered as email address
|
2022-07-21 17:57:50 +02:00
|
|
|
* @example: contactId = "account:4"
|
2022-07-12 17:14:44 +02:00
|
|
|
*
|
2023-06-27 08:59:43 +02:00
|
|
|
* @param {string} _contactId contact id could be as above-mentioned formats
|
2022-07-12 17:14:44 +02:00
|
|
|
*/
|
2022-07-21 17:57:50 +02:00
|
|
|
set contactId(_contactId : string)
|
2022-07-12 17:14:44 +02:00
|
|
|
{
|
|
|
|
let params = {};
|
2022-07-21 22:52:35 +02:00
|
|
|
let id = 'contact_id';
|
2023-06-27 17:46:30 +02:00
|
|
|
let parsedId = "";
|
2022-07-12 17:14:44 +02:00
|
|
|
|
2022-07-21 17:57:50 +02:00
|
|
|
if (!_contactId)
|
2022-07-12 17:14:44 +02:00
|
|
|
{
|
2023-07-27 12:45:40 +02:00
|
|
|
parsedId = null;
|
2022-07-12 17:14:44 +02:00
|
|
|
}
|
2023-06-27 08:59:43 +02:00
|
|
|
else if(_contactId.substr(0, 8) === 'account:')
|
2022-07-12 17:14:44 +02:00
|
|
|
{
|
|
|
|
id = 'account_id';
|
2023-06-27 17:46:30 +02:00
|
|
|
parsedId = _contactId.substr(8);
|
2023-06-27 08:59:43 +02:00
|
|
|
}
|
|
|
|
else if(_contactId.substr(0, 6) === 'email:')
|
|
|
|
{
|
|
|
|
id = 'email';
|
|
|
|
const matches = Et2Avatar.RFC822EMAIL.exec(_contactId);
|
2023-06-27 17:46:30 +02:00
|
|
|
parsedId = matches ? matches[1] : _contactId.substr(6);
|
2022-07-12 17:14:44 +02:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2022-07-21 22:52:35 +02:00
|
|
|
id = 'contact_id';
|
2023-06-27 17:46:30 +02:00
|
|
|
parsedId = _contactId.replace('contact:', '');
|
2022-07-12 17:14:44 +02:00
|
|
|
}
|
2022-07-21 17:57:50 +02:00
|
|
|
let oldContactId = this._contactId;
|
|
|
|
this._contactId = _contactId;
|
2023-06-27 17:46:30 +02:00
|
|
|
// if our image (incl. cache-buster) already includes the correct id, use that one
|
2023-07-27 12:45:40 +02:00
|
|
|
if (!parsedId)
|
|
|
|
{
|
|
|
|
this.image = null;
|
|
|
|
}
|
|
|
|
else if(!this.image || !this.image.match("(&|\\?)" + id + "=" + encodeURIComponent(parsedId) + "(&|$)"))
|
2022-07-12 17:14:44 +02:00
|
|
|
{
|
2023-06-27 17:46:30 +02:00
|
|
|
params[id] = parsedId;
|
|
|
|
this.image = egw.link('/api/avatar.php', params);
|
2022-07-12 17:14:44 +02:00
|
|
|
}
|
2022-07-21 17:57:50 +02:00
|
|
|
this.requestUpdate("contactId", oldContactId);
|
2022-07-12 17:14:44 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
set value(_value)
|
|
|
|
{
|
2022-07-21 17:57:50 +02:00
|
|
|
this.contactId = _value;
|
2022-07-12 17:14:44 +02:00
|
|
|
}
|
|
|
|
|
2022-07-19 12:18:42 +02:00
|
|
|
/**
|
|
|
|
* set the image source
|
|
|
|
* @deprecated please use image instead
|
|
|
|
* @param _value
|
|
|
|
*/
|
2022-07-12 17:14:44 +02:00
|
|
|
set src(_value)
|
|
|
|
{
|
|
|
|
this.image = _value;
|
|
|
|
}
|
|
|
|
|
2022-07-14 17:32:45 +02:00
|
|
|
get _baseNode()
|
|
|
|
{
|
|
|
|
return this.shadowRoot.querySelector("[part='base']");
|
|
|
|
}
|
|
|
|
|
|
|
|
get _imageNode()
|
|
|
|
{
|
|
|
|
return this.shadowRoot.querySelector("[part='image']");
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Build Editable Mask Layer (EML) in order to show edit/delete actions
|
|
|
|
* on top of profile picture.
|
|
|
|
* @param {boolean} _noDelete disable delete button in initialization
|
|
|
|
*/
|
|
|
|
private _buildEditableLayer(_noDelete : boolean)
|
|
|
|
{
|
|
|
|
let self = this;
|
2022-09-21 11:15:14 +02:00
|
|
|
this._editBtn = document.createElement('et2-button-icon');
|
2022-07-19 12:18:42 +02:00
|
|
|
this._editBtn.setAttribute('name', 'pencil');
|
|
|
|
this._editBtn.setAttribute('part', 'edit');
|
2022-09-21 11:15:14 +02:00
|
|
|
this._delBtn = document.createElement('et2-button-icon');
|
2022-07-19 12:18:42 +02:00
|
|
|
this._delBtn.setAttribute('name', 'trash');
|
|
|
|
this._delBtn.setAttribute('part', 'edit');
|
|
|
|
this._baseNode.append(this._editBtn);
|
|
|
|
this._baseNode.append(this._delBtn);
|
|
|
|
|
|
|
|
// disable the delete button if no delete is set
|
|
|
|
this._delBtn.disabled = _noDelete;
|
|
|
|
|
|
|
|
// bind click handler to edit button
|
|
|
|
this._editBtn.addEventListener('click', this.editButtonClickHandler.bind(this));
|
|
|
|
|
|
|
|
// bind click handler to del button
|
|
|
|
this._delBtn.addEventListener('click', this.delButtonClickHandler.bind(this));
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* click handler to handle click on edit button
|
|
|
|
*/
|
|
|
|
editButtonClickHandler()
|
|
|
|
{
|
|
|
|
const buttons = [
|
|
|
|
{"button_id": 1, label: this.egw().lang('save'), id: 'save', image: 'check', "default": true},
|
|
|
|
{"button_id": 0, label: this.egw().lang('cancel'), id: 'cancel', image: 'cancelled'}
|
|
|
|
];
|
|
|
|
const value = {
|
2022-07-21 17:57:50 +02:00
|
|
|
contactId: this.contactId,
|
2022-07-19 12:18:42 +02:00
|
|
|
src: this.image
|
|
|
|
}
|
|
|
|
this._editDialog(egw.lang('Edit avatar'), value, buttons, null);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Build edit dialog
|
|
|
|
* @param _title
|
|
|
|
* @param _value
|
|
|
|
* @param _buttons
|
|
|
|
* @param _egw_or_appname
|
|
|
|
*/
|
|
|
|
private _editDialog(_title, _value, _buttons, _egw_or_appname)
|
|
|
|
{
|
|
|
|
let dialog = new Et2Dialog(this.egw());
|
|
|
|
dialog.transformAttributes({
|
|
|
|
callback: this.__editDialogCallback.bind(this),
|
|
|
|
title: _title || egw.lang('Input required'),
|
|
|
|
buttons: _buttons || Et2Dialog.BUTTONS_OK_CANCEL,
|
|
|
|
value: {
|
|
|
|
content: _value
|
|
|
|
},
|
|
|
|
width: "90%",
|
|
|
|
height: "450",
|
|
|
|
resizable: false,
|
|
|
|
position: "top+10",
|
|
|
|
template: egw.webserverUrl + '/api/templates/default/avatar_edit.xet'
|
2022-07-14 17:32:45 +02:00
|
|
|
});
|
2022-07-19 12:18:42 +02:00
|
|
|
document.body.appendChild(dialog);
|
|
|
|
return dialog;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Edit dialog callback function
|
|
|
|
* @param _buttons
|
|
|
|
* @param _value
|
|
|
|
*/
|
|
|
|
private __editDialogCallback(_buttons, _value)
|
|
|
|
{
|
|
|
|
let widget = document.getElementById('_cropper_image');
|
|
|
|
switch(_buttons)
|
|
|
|
{
|
2022-09-13 12:09:55 +02:00
|
|
|
case 0:
|
|
|
|
return true;
|
2022-07-19 12:18:42 +02:00
|
|
|
case 1:
|
|
|
|
let canvas = jQuery(widget._imageNode).cropper('getCroppedCanvas');
|
|
|
|
this.image = canvas.toDataURL("image/jpeg", 1.0)
|
|
|
|
this.egw().json('addressbook.addressbook_ui.ajax_update_photo',
|
|
|
|
[this.getInstanceManager().etemplate_exec_id, canvas.toDataURL("image/jpeg", 1.0)],
|
|
|
|
this.__editAjaxUpdatePhotoCallback.bind(this)).sendRequest();
|
|
|
|
break;
|
|
|
|
case '_rotate_reset':
|
|
|
|
jQuery(widget._imageNode).cropper('reset');
|
|
|
|
return false;
|
|
|
|
case '_rotate_l':
|
|
|
|
jQuery(widget._imageNode).cropper('rotate', -90);
|
|
|
|
return false;
|
|
|
|
case '_rotate_r':
|
|
|
|
jQuery(widget._imageNode).cropper('rotate', 90);
|
|
|
|
return false;
|
2022-09-13 12:09:55 +02:00
|
|
|
default:
|
|
|
|
return false;
|
2022-07-19 12:18:42 +02:00
|
|
|
}
|
|
|
|
}
|
2022-07-14 17:32:45 +02:00
|
|
|
|
2022-07-19 12:18:42 +02:00
|
|
|
/**
|
|
|
|
* Edit ajax update photo response callback
|
|
|
|
* @param response
|
|
|
|
*/
|
|
|
|
private __editAjaxUpdatePhotoCallback(response)
|
|
|
|
{
|
|
|
|
if(response)
|
|
|
|
{
|
|
|
|
this._delBtn.style.visibility = 'visible';
|
|
|
|
}
|
|
|
|
}
|
2022-07-14 17:32:45 +02:00
|
|
|
|
2022-07-19 12:18:42 +02:00
|
|
|
/**
|
|
|
|
* click handler to handel click on delete button
|
|
|
|
*/
|
|
|
|
delButtonClickHandler()
|
|
|
|
{
|
|
|
|
//build delete dialog
|
|
|
|
Et2Dialog.show_dialog(this._delBtnDialogCallback.bind(this), egw.lang('Delete this photo?'), egw.lang('Delete'),
|
|
|
|
null, Et2Dialog.BUTTONS_YES_NO);
|
|
|
|
}
|
2022-07-14 17:32:45 +02:00
|
|
|
|
2022-07-19 12:18:42 +02:00
|
|
|
/**
|
|
|
|
* del dialog callback function
|
|
|
|
* @param _btn
|
|
|
|
*/
|
|
|
|
private _delBtnDialogCallback(_btn)
|
|
|
|
{
|
|
|
|
if(_btn == Et2Dialog.YES_BUTTON)
|
2022-07-14 17:32:45 +02:00
|
|
|
{
|
2022-07-19 12:18:42 +02:00
|
|
|
this.egw().json('addressbook.addressbook_ui.ajax_update_photo',
|
|
|
|
[this.getInstanceManager().etemplate_exec_id, null],
|
|
|
|
this.__delAjaxUpdatePhotoCallback.bind(this)).sendRequest();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Del ajax update photo response callback
|
|
|
|
* @param response
|
|
|
|
*/
|
|
|
|
private __delAjaxUpdatePhotoCallback(response)
|
|
|
|
{
|
|
|
|
if(response)
|
|
|
|
{
|
|
|
|
this.image = '';
|
|
|
|
this._delBtn.style.visibility = 'none';
|
|
|
|
egw.refresh('Avatar Deleted.', egw.app_name());
|
|
|
|
}
|
2022-07-14 17:32:45 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Function runs after uplaod in avatar dialog is finished and it tries to
|
|
|
|
* update image and cropper container.
|
|
|
|
* @param {type} e
|
|
|
|
*/
|
|
|
|
static uploadAvatar_onFinish(e)
|
|
|
|
{
|
|
|
|
let file = e.data.resumable.files[0].file;
|
|
|
|
let reader = new FileReader();
|
|
|
|
reader.onload = function (e)
|
|
|
|
{
|
|
|
|
let widget = document.getElementById('_cropper_image');
|
|
|
|
widget.image = e.target.result;
|
|
|
|
// Wait for everything to complete
|
|
|
|
widget.getUpdateComplete().then(() =>
|
|
|
|
{
|
|
|
|
jQuery(widget._imageNode).cropper('replace',e.target.result)
|
|
|
|
});
|
|
|
|
|
|
|
|
};
|
|
|
|
reader.readAsDataURL(file);
|
|
|
|
}
|
|
|
|
|
2022-07-12 17:14:44 +02:00
|
|
|
/**
|
2022-07-14 17:32:45 +02:00
|
|
|
*
|
2022-07-12 17:14:44 +02:00
|
|
|
*/
|
|
|
|
getDetachedAttributes(_attrs : string[])
|
|
|
|
{
|
2023-04-25 21:53:16 +02:00
|
|
|
_attrs.push("contactId", "label", "href", "src", "image", "statustext");
|
2022-07-12 17:14:44 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
getDetachedNodes()
|
|
|
|
{
|
|
|
|
return [<HTMLElement><unknown>this];
|
|
|
|
}
|
|
|
|
|
|
|
|
setDetachedAttributes(_nodes, _values)
|
|
|
|
{
|
|
|
|
for(let attr in _values)
|
|
|
|
{
|
|
|
|
this[attr] = _values[attr];
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2023-09-27 22:29:19 +02:00
|
|
|
|
|
|
|
customElements.define("et2-avatar", Et2Avatar);
|
2022-07-14 17:32:45 +02:00
|
|
|
// make et2_avatar publicly available as we need to call it from templates
|
|
|
|
{
|
|
|
|
window['et2_avatar'] = Et2Avatar;
|
|
|
|
window['Et2Avatar'] = Et2Avatar;
|
|
|
|
}
|