import {cleanSelectOptions, SelectOption} from "../Et2Select/FindSelectOptions"; import {css, html, SlotMixin, TemplateResult} from "@lion/core"; import {Et2Select} from "../Et2Select/Et2Select"; export class Et2LinkAppSelect extends SlotMixin(Et2Select) { static get styles() { return [ ...super.styles, css` :host { --icon-width: 20px; display: inline-block; min-width: 64px; } :host([appIcons]) { max-width: 75px; } .select__menu { overflow-x: hidden; } ::part(control) { border: none; box-shadow: initial; } ` ] } static get properties() { return { ...super.properties, /** * Limit to just this one application, and hide the selection */ onlyApp: {type: String}, /** * Limit to these applications (comma seperated). */ applicationList: {type: String}, /** * Show application icons instead of application names */ appIcons: {type: Boolean, reflect: true} } }; get slots() { return { ...super.slots, "": () => { const icon = document.createElement("et2-image"); icon.setAttribute("slot", "prefix"); icon.setAttribute("src", "api/navbar"); icon.style.width = "var(--icon-width)"; return icon; } } } protected __applicationList : string[]; protected __onlyApp : string; /** * Constructor * */ constructor() { super(); this.onlyApp = ""; this.appIcons = true; this.applicationList = []; this.hoist = true; // Select options are based off abilities registered with link system this._reset_select_options(); this._handleChange = this._handleChange.bind(this); } set onlyApp(app : string) { this.__onlyApp = app || ""; this.updateComplete.then(() => { this.style.display = this.onlyApp ? 'none' : ''; }); } get onlyApp() : string { // __onlyApp may be undefined during creation return this.__onlyApp || ""; } connectedCallback() { super.connectedCallback(); // Set icon this.querySelector("[slot='prefix']").setAttribute("src", this.value + "/navbar"); if(!this.value) { // use preference let appname = ""; if(typeof this.value != 'undefined' && this.parentNode && this.parentNode.to_app) { appname = this.parentNode.to_app; } this.value = this.egw().preference('link_app', appname || this.egw().app_name()); } // Register to this.addEventListener("change", this._handleChange); if(this.__onlyApp) { this.style.display = 'none'; } } disconnectedCallback() { super.disconnectedCallback(); this.removeEventListener("change", this._handleChange); } /** * Called before update() to compute values needed during the update * * @param changedProperties */ willUpdate(changedProperties) { super.willUpdate(changedProperties); if(changedProperties.has("onlyApp") || changedProperties.has("applicationList")) { this._reset_select_options(); } } set applicationList(app_list : string[]) { let oldValue = this.__applicationList; if(typeof app_list == "string") { app_list = (app_list).split(","); } this.__applicationList = app_list; this.requestUpdate("applicationList", oldValue); } get applicationList() : string[] { return this.__applicationList; } get value() : string { return this.onlyApp ? this.onlyApp : super.value; } set value(new_value) { super.value = new_value; } private _handleChange(e) { // Set icon this.querySelector("[slot='prefix']").setAttribute("src", this.value + "/navbar"); // update preference let appname = ""; if(typeof this.value != 'undefined' && this.parentNode && this.parentNode.to_app) { appname = this.parentNode.to_app; } this.egw().set_preference(appname || this.egw().app_name(), 'link_app', this.value); } /** * Limited select options here * This method will check properties and set select options appropriately */ private _reset_select_options() { let select_options = []; // Limit to one app if(this.onlyApp) { select_options.push({value: this.onlyApp, label: this.egw().lang(this.onlyApp)}); } else if(this.applicationList.length > 0) { select_options = this.applicationList; } else { //@ts-ignore link_app_list gives {app:name} instead of an array, but parent will fix it select_options = this.egw().link_app_list('query'); if(typeof select_options['addressbook-email'] !== 'undefined') { delete select_options['addressbook-email']; } } if (!this.value) { this.value = this.egw().preference('link_app', this.egw().app_name()); } this.select_options = cleanSelectOptions(select_options); } _optionTemplate(option : SelectOption) : TemplateResult { return html` ${this.appIcons ? "" : option.label} ${this._iconTemplate(option.value)} `; } _iconTemplate(appname) { let url = appname ? this.egw().image('navbar', appname) : ""; return html` `; } } // @ts-ignore TypeScript is not recognizing that this widget is a LitElement customElements.define("et2-link-apps", Et2LinkAppSelect);