diff --git a/api/js/etemplate/Et2Taglist/Et2Taglist.ts b/api/js/etemplate/Et2Taglist/Et2Taglist.ts new file mode 100644 index 0000000000..92a82fb8d7 --- /dev/null +++ b/api/js/etemplate/Et2Taglist/Et2Taglist.ts @@ -0,0 +1,106 @@ +/** + * EGroupware eTemplate2 - Colorpicker widget (WebComponent) + * + * @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 {css, html, LitElement} from "@lion/core"; +import { ScopedElementsMixin } from '@open-wc/scoped-elements'; +import {TaglistOption, TaglistOptionEmail} from "./TaglistOption"; +import {TaglistComboBox} from "./TaglistComboBox"; +import {Et2widgetWithSelectMixin} from "../Et2Select/Et2WidgetWithSelectMixin"; +import {TaglistSelection} from "./TaglistSelection"; + +/** + * Taglist base class implementation + */ +export class Et2Taglist extends Et2widgetWithSelectMixin(ScopedElementsMixin(LitElement)) +{ + static get styles() + { + return [ + ...super.styles, + css` + :host { + display: block; + } + ` + ]; + } + static get properties() + { + return { + ...super.properties, + multiple : {type : Boolean}, + + + } + } + + constructor() + { + super(); + + } + + // can be overriden for other taglist type implementations + static get taglistOptionImp () {return TaglistOption}; + static get taglistComboboxImp () {return TaglistComboBox}; + static get taglistSelectionImp () {return TaglistSelection}; + + static get scopedElements() { + return { + 'taglist-combobox': this.taglistComboboxImp, + 'taglist-option': this.taglistOptionImp, + 'taglist-selection': this.taglistSelectionImp + }; + } + + _setOptionTemplate() + { + return html` + ${this.get_select_options().map(entry => html` ${entry.label} `)} + `; + } + + render() + { + return html` + + + ${this._setOptionTemplate()} + + `; + } +} +customElements.define('et2-taglist', Et2Taglist); + + +/** + * Taglist-email implementation + */ +export class Et2TaglistEmail extends Et2Taglist +{ + static get taglistOptionImp () {return TaglistOptionEmail}; + + _setOptionTemplate() + { + //@todo: needs to be implemented + return super._setOptionTemplate(); + } + + get_select_options(): any + { + //@todo: needs to be implemented + return super.get_select_options(); + } +} +customElements.define('et2-taglist-email', Et2TaglistEmail); \ No newline at end of file diff --git a/api/js/etemplate/Et2Taglist/Et2TaglistEmail.ts b/api/js/etemplate/Et2Taglist/Et2TaglistEmail.ts new file mode 100644 index 0000000000..08f80f0f51 --- /dev/null +++ b/api/js/etemplate/Et2Taglist/Et2TaglistEmail.ts @@ -0,0 +1,19 @@ + +export class Et2TaglistEmail extends Et2Taglist +{ + static get taglist_option () {return TaglistOptionEmail}; + + _setOptionTemplate() + { + return html` + ${this.get_select_options().map(entry => html` ${entry.label} `)} + `; + } + + get_select_options(): any + { + //@todo: override it accordingly + return super.get_select_options(); + } +} +customElements.define('et2-taglist-email', Et2TaglistEmail); \ No newline at end of file diff --git a/api/js/etemplate/Et2Taglist/TaglistComboBox.ts b/api/js/etemplate/Et2Taglist/TaglistComboBox.ts new file mode 100644 index 0000000000..e553100cb3 --- /dev/null +++ b/api/js/etemplate/Et2Taglist/TaglistComboBox.ts @@ -0,0 +1,48 @@ + +/** + * EGroupware eTemplate2 - TaglistComboBox widget (WebComponent) + * + * @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 { LionCombobox } from '@lion/combobox'; +import {css} from "@lion/core"; + + +export class TaglistComboBox extends LionCombobox { + static get properties() { + return { + ...super.properties, + + }; + } + static get styles() { + return [ + super.styles, + css` + ::slotted([slot='input']) { + min-height: 23px; + + } + :host { + border: 1px solid var(--taglist-combobox__container-boder-color); + border-radius: 3px; + } + `, + ]; + } + constructor() + { + super(); + } +} +customElements.define('taglist-combobox', TaglistComboBox); + + + diff --git a/api/js/etemplate/Et2Taglist/TaglistOption.ts b/api/js/etemplate/Et2Taglist/TaglistOption.ts new file mode 100644 index 0000000000..98e5d1e9d8 --- /dev/null +++ b/api/js/etemplate/Et2Taglist/TaglistOption.ts @@ -0,0 +1,54 @@ +/** + * EGroupware eTemplate2 - TaglistOption widget (WebComponent) + * + * @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 { LionOption } from '@lion/listbox'; +import {css, html} from "@lion/core"; + + +export class TaglistOption extends LionOption { + + static get properties() { + return { + ...super.properties, + title: { + type: String, + }, + }; + } + constructor() { + super(); + this.title = ''; + + } + + static get styles() { + return [ + ...super.styles, + css` + :host([checked]) { + visibility: hidden; + } + `, + ]; + } + +} + +export class TaglistOptionEmail extends TaglistOption { + +} + +export class TaglistOptionState extends TaglistOption { + +} + +export class TaglistOptionCategory extends TaglistOption { + +} \ No newline at end of file diff --git a/api/js/etemplate/Et2Taglist/TaglistSelection.ts b/api/js/etemplate/Et2Taglist/TaglistSelection.ts new file mode 100644 index 0000000000..a578ad407e --- /dev/null +++ b/api/js/etemplate/Et2Taglist/TaglistSelection.ts @@ -0,0 +1,202 @@ +/** + * EGroupware eTemplate2 - TaglistSelection (WebComponent) + * + * @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 {css, html, LitElement} from "@lion/core"; +import {TaglistComboBox} from "./TaglistComboBox"; +import {taglistStyles} from "./TaglistStyles"; + + +/** + * Implementation of selection tags + */ +export class TaglistSelection extends LitElement { + static get properties() { + return { + selectedTags: Array, + comboxElement: TaglistComboBox, + canBeClosed: Boolean, + canBeEdited: Boolean + } + }; + + static get styles() { + return [ + taglistStyles, + css` + :host { + display: flex; + } + + .taglist-selection__tags { + flex: none; + } + + .combobox__input { + display: block; + } + + .taglist-selection__tag { + margin: 0 5px 3px 0; + padding: 3px 20px 3px 5px; + border: 1px solid var(--taglist-selection__tag-boder-color); + border-radius: 3px; + background-color: var(--taglist-selection__tag-bg-color); + background-image: var(--taglist-selection__tag-bg-img); + background-clip: padding-box; + box-shadow: var(--taglist-selection__tag-box-shadow); + color: var(--taglist-selection__tag-color); + line-height: 13px; + font-size: 11px; + white-space: normal; + max-width: calc(100% - 30px); + display:flex; + } + .tag-label { + display:flex; + } + .tag-closeBtn{ + width: 10px; + height: 10px; + background-position: 0 -10px; + background-size: cover; + background-repeat: no-repeat; + display:flex; + background-image: var(--tag-closeBtn-img); + } + ` + ]; + } + + /** + * + */ + get _inputNode() { + return this._getComboBoxElement()._inputNode; + } + + + /** + * @return {TaglistComboBox} returns comboboxElement from TaglistComboBox + */ + _getComboBoxElement() + { + // @ts-ignore + return this.comboboxElement; + } + + /** + * @private + * @return returns checked formElements + */ + __getSelectedTags() { + return this._getComboBoxElement().formElements.filter((_tags) => { + return _tags.checked; + } + ); + } + + get multipleChoice() { + return this._getComboBoxElement()?.multipleChoice; + } + + + constructor() { + super(); + + this.selectedTags = []; + this.__handleCloseBtn = this.__handleCloseBtn.bind(this); + this.__handleEditBtn = this.__handleEditBtn.bind(this); + this.__inputOnKeyup = this.__inputOnKeyup.bind(this); + + } + + /** + * + * @param changedProperties + */ + firstUpdated(changedProperties) { + super.firstUpdated(changedProperties); + + if (this.multipleChoice) { + this._inputNode.addEventListener('keyup', this.__inputOnKeyup); + } + } + + /** + * + * @param changedProperties + */ + onComboboxElementUpdated(changedProperties) { + if (changedProperties.has('modelValue')) { + this.selectedTags = this.__getSelectedTags(); + } + } + + __handleEditBtn() + { + console.log('editBtn') + } + + __handleCloseBtn(_v) + { + console.log('closeBtn') + } + + /** + * + * @param option + */ + _selectedTagTemplate(option) { + return html` +
+ ${this._canBeEdited? html``:''} + ${option.value} + ${this._canBeClosed? html``:''} +
+ `; + } + + /** + * + */ + _selectedTagsTemplate() { + return html` +
+ ${this.selectedTags.map((option) => { + return this._selectedTagTemplate(option); + })} +
+ `; + } + + /** + * + */ + render() { + return html` ${this._selectedTagsTemplate()} `; + } + + /** + * @private + * @param ev + */ + __inputOnKeyup(ev) { + if (ev.key === 'Backspace') { + if (!this._inputNode.value) { + if (this.selectedTags.length) { + this.selectedTags[this.selectedTags.length - 1].checked = false; + } + } + } + } +} +customElements.define('taglist-selection', TaglistSelection); diff --git a/api/js/etemplate/Et2Taglist/TaglistStyles.ts b/api/js/etemplate/Et2Taglist/TaglistStyles.ts new file mode 100644 index 0000000000..d25a41ee71 --- /dev/null +++ b/api/js/etemplate/Et2Taglist/TaglistStyles.ts @@ -0,0 +1,20 @@ +/** + * styles constant + */ + +import {css} from "@lion/core"; + +export const taglistStyles = css` + :host { + /****** images ************/ + --tag-closeBtn-img : url(); + --taglist-selection__tag-bg-img : linear-gradient(#f4f4f4 20%,#f0f0f0 50%,#e8e8e8 52%,#eee 100%); + + /****** colors ************/ + --taglist-combobox__container-boder-color : silver; + --taglist-selection__tag-boder-color : #aaa; + --taglist-selection__tag-bg-color : #e4e4e4; + --taglist-selection__tag-color : #333; + --taglist-selection__tag-box-shadow : 0 0 2px #fff inset, 0 1px 0 rgb(0 0 0 / 5%); + } +`; \ No newline at end of file diff --git a/api/js/etemplate/etemplate2.ts b/api/js/etemplate/etemplate2.ts index 70e495bd73..13fa14d783 100644 --- a/api/js/etemplate/etemplate2.ts +++ b/api/js/etemplate/etemplate2.ts @@ -43,6 +43,8 @@ import './Et2Textarea/Et2Textarea'; import './Et2Textbox/Et2Textbox'; import './Et2Textbox/Et2TextboxReadonly'; import './Et2Colorpicker/Et2Colorpicker'; +import './Et2Taglist/Et2Taglist'; + /* Include all widget classes here, we only care about them registering, not importing anything*/ import './et2_widget_vfs'; // Vfs must be first (before et2_widget_file) due to import cycle import './et2_widget_template';