mirror of
https://github.com/EGroupware/egroupware.git
synced 2025-01-18 20:08:34 +01:00
172 lines
4.1 KiB
TypeScript
172 lines
4.1 KiB
TypeScript
import {SelectOption} from "./FindSelectOptions";
|
|
import {LitElement} from "lit";
|
|
|
|
/**
|
|
* EGroupware eTemplate2 - SelectAccountMixin
|
|
*
|
|
* @license http://opensource.org/licenses/gpl-license.php GPL - GNU General Public License
|
|
* @package api
|
|
* @link https://www.egroupware.org
|
|
* @author Nathan Gray
|
|
*/
|
|
|
|
/**
|
|
* Common code for editable & read-only select account
|
|
* Moved out so we don't have copy+paste code in both
|
|
*/
|
|
export declare class SelectAccountInterface
|
|
{
|
|
set value(new_value : string | string[])
|
|
|
|
get value() : string | string[]
|
|
|
|
set select_options(new_options : SelectOption[])
|
|
get select_options() : SelectOption[]
|
|
}
|
|
|
|
type Constructor<T = {}> = new (...args : any[]) => T;
|
|
export const SelectAccountMixin = <T extends Constructor<LitElement>>(superclass : T) =>
|
|
{
|
|
class SelectAccount extends superclass
|
|
{
|
|
|
|
/**
|
|
* Hold on to accounts we had to fetch from the server
|
|
* @type {any[]}
|
|
* @protected
|
|
*/
|
|
protected account_options = [];
|
|
|
|
constructor()
|
|
{
|
|
super();
|
|
this.account_options = [];
|
|
}
|
|
|
|
/**
|
|
* If the value has an account that's not already in the list, check with the server.
|
|
* We probably don't have all the accounts client side. This is similar to freeEntries,
|
|
* but a little safer because we don't allow just anything.
|
|
*
|
|
* @param {any} new_value
|
|
*/
|
|
set value(new_value : string | string[])
|
|
{
|
|
super.value = new_value;
|
|
if(!new_value)
|
|
{
|
|
return;
|
|
}
|
|
let val = Array.isArray(this.value) ? this.value : [this.value];
|
|
|
|
if(this.isConnected)
|
|
{
|
|
this._find_options(val)
|
|
}
|
|
else
|
|
{
|
|
// If not already connected, wait until any provided select_options have been found
|
|
this.updateComplete.then(() =>
|
|
{
|
|
this._find_options(val);
|
|
this.requestUpdate('select_options');
|
|
});
|
|
}
|
|
}
|
|
|
|
/**
|
|
* OVerridden to do nothing, we handle it differently in _find_options()
|
|
* @param {string} newValueElement
|
|
* @protected
|
|
*/
|
|
protected _missingOption(newValueElement : string)
|
|
{}
|
|
|
|
_find_options(val)
|
|
{
|
|
for(let id of val)
|
|
{
|
|
// Don't add if it's already there
|
|
if(this.select_options.findIndex(o => o.value == id) != -1)
|
|
{
|
|
continue;
|
|
}
|
|
|
|
let account_name = null;
|
|
const tempLabel = id + " ..."
|
|
let option = <SelectOption>{value: "" + id, label: tempLabel};
|
|
this.account_options.push(option);
|
|
if(this.value && (account_name = this.egw().link_title('api-accounts', id, false)))
|
|
{
|
|
option.label = account_name;
|
|
}
|
|
else if(!account_name)
|
|
{
|
|
// Not already cached, need to fetch it
|
|
this.egw().link_title('api-accounts', id, true).then(title =>
|
|
{
|
|
option.label = title || '';
|
|
this.requestUpdate();
|
|
|
|
this.account_options.sort(this.optionSort);
|
|
// Directly update if it's already there
|
|
const slOption = this.select?.querySelector('[value="' + id + '"]');
|
|
if(slOption)
|
|
{
|
|
// Replace instead of changing the whole thing to preserve LitElement marker comments
|
|
slOption.textContent.replace(tempLabel, title);
|
|
this.select.requestUpdate("value");
|
|
}
|
|
});
|
|
}
|
|
}
|
|
this.account_options.sort(this.optionSort);
|
|
}
|
|
|
|
get value()
|
|
{
|
|
return super.value;
|
|
}
|
|
|
|
get select_options()
|
|
{
|
|
return [...new Map([...this.account_options, ...(super.select_options || [])].map(item =>
|
|
[item.value, item])).values()].sort(this.optionSort);
|
|
}
|
|
|
|
set select_options(value : SelectOption[])
|
|
{
|
|
super.select_options = value;
|
|
}
|
|
|
|
/**
|
|
* Sort options
|
|
* @param a
|
|
* @param b
|
|
* @returns {number}
|
|
* @protected
|
|
*/
|
|
protected optionSort(a : SelectOption, b : SelectOption)
|
|
{
|
|
// Sort accounts before groups, then by label
|
|
let int_a = 0;
|
|
let int_b = 0;
|
|
if(typeof a.value === "string")
|
|
{
|
|
int_a = parseInt(a.value) ?? 0;
|
|
}
|
|
if(typeof b.value === "string")
|
|
{
|
|
int_b = parseInt(b.value) ?? 0;
|
|
}
|
|
if(int_a < 0 && int_b < 0 || int_a > 0 && int_b > 0)
|
|
{
|
|
return ('' + a.label).localeCompare(b.label);
|
|
}
|
|
// Accounts before groups
|
|
return int_b - int_a;
|
|
}
|
|
}
|
|
|
|
return SelectAccount as unknown as Constructor<SelectAccountInterface> & T;
|
|
} |