mirror of
https://github.com/EGroupware/egroupware.git
synced 2025-01-05 05:29:13 +01:00
Et2TreeDropdown: Add some keyboard interactions
This commit is contained in:
parent
94a32d2800
commit
c53e122564
@ -4,15 +4,14 @@ import {find_select_options} from "../Et2Select/FindSelectOptions";
|
|||||||
import {Et2WidgetWithSelectMixin} from "../Et2Select/Et2WidgetWithSelectMixin";
|
import {Et2WidgetWithSelectMixin} from "../Et2Select/Et2WidgetWithSelectMixin";
|
||||||
import {css, html, LitElement, nothing, PropertyValues, TemplateResult} from "lit";
|
import {css, html, LitElement, nothing, PropertyValues, TemplateResult} from "lit";
|
||||||
import {repeat} from "lit/directives/repeat.js";
|
import {repeat} from "lit/directives/repeat.js";
|
||||||
|
import {query} from "lit/decorators/query.js";
|
||||||
import shoelace from "../Styles/shoelace";
|
import shoelace from "../Styles/shoelace";
|
||||||
import {property} from "lit/decorators/property.js";
|
import {property} from "lit/decorators/property.js";
|
||||||
import {state} from "lit/decorators/state.js";
|
import {state} from "lit/decorators/state.js";
|
||||||
import {egw_getActionManager, egw_getAppObjectManager} from "../../egw_action/egw_action";
|
import {egw_getActionManager, egw_getAppObjectManager} from "../../egw_action/egw_action";
|
||||||
import {et2_action_object_impl} from "../et2_core_DOMWidget";
|
import {et2_action_object_impl} from "../et2_core_DOMWidget";
|
||||||
import {EgwActionObject} from "../../egw_action/EgwActionObject";
|
import {EgwActionObject} from "../../egw_action/EgwActionObject";
|
||||||
import {object} from "prop-types";
|
|
||||||
import {EgwAction} from "../../egw_action/EgwAction";
|
import {EgwAction} from "../../egw_action/EgwAction";
|
||||||
import {query} from "@lion/core";
|
|
||||||
import {EgwDragDropShoelaceTree} from "../../egw_action/EgwDragDropShoelaceTree";
|
import {EgwDragDropShoelaceTree} from "../../egw_action/EgwDragDropShoelaceTree";
|
||||||
|
|
||||||
export type TreeItemData = {
|
export type TreeItemData = {
|
||||||
@ -92,6 +91,8 @@ export class Et2Tree extends Et2WidgetWithSelectMixin(LitElement)
|
|||||||
private input: any = null;
|
private input: any = null;
|
||||||
private _actionManager: EgwAction;
|
private _actionManager: EgwAction;
|
||||||
|
|
||||||
|
private get _tree() { return this.shadowRoot.querySelector('sl-tree') ?? null};
|
||||||
|
|
||||||
|
|
||||||
constructor()
|
constructor()
|
||||||
{
|
{
|
||||||
@ -306,6 +307,17 @@ export class Et2Tree extends Et2WidgetWithSelectMixin(LitElement)
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/** Sets focus on the control. */
|
||||||
|
focus(options? : FocusOptions)
|
||||||
|
{
|
||||||
|
this._tree.focus();
|
||||||
|
}
|
||||||
|
|
||||||
|
/** Removes focus from the control. */
|
||||||
|
blur()
|
||||||
|
{
|
||||||
|
this._tree.blur();
|
||||||
|
}
|
||||||
/**
|
/**
|
||||||
* @deprecated assign to onopenstart
|
* @deprecated assign to onopenstart
|
||||||
* @param _handler
|
* @param _handler
|
||||||
|
@ -71,8 +71,8 @@ export default css`
|
|||||||
outline: none;
|
outline: none;
|
||||||
}
|
}
|
||||||
|
|
||||||
:not(.tree-dropdown--disabled).tree-dropdown--open,
|
:not(.tree-dropdown--disabled).tree-dropdown--open .tree-dropdown__combobox,
|
||||||
:not(.tree-dropdown--disabled).tree-dropdown--focused {
|
:not(.tree-dropdown--disabled).tree-dropdown--focused .tree-dropdown__combobox {
|
||||||
background-color: var(--sl-input-background-color-focus);
|
background-color: var(--sl-input-background-color-focus);
|
||||||
border-color: var(--sl-input-border-color-focus);
|
border-color: var(--sl-input-border-color-focus);
|
||||||
box-shadow: 0 0 0 var(--sl-focus-ring-width) var(--sl-input-focus-ring-color);
|
box-shadow: 0 0 0 var(--sl-focus-ring-width) var(--sl-input-focus-ring-color);
|
||||||
|
@ -3,13 +3,14 @@ import {Et2Tree, TreeItemData} from "./Et2Tree";
|
|||||||
import {Et2WidgetWithSelectMixin} from "../Et2Select/Et2WidgetWithSelectMixin";
|
import {Et2WidgetWithSelectMixin} from "../Et2Select/Et2WidgetWithSelectMixin";
|
||||||
import {property} from "lit/decorators/property.js";
|
import {property} from "lit/decorators/property.js";
|
||||||
import {classMap} from "lit/directives/class-map.js";
|
import {classMap} from "lit/directives/class-map.js";
|
||||||
|
import {state} from "lit/decorators/state.js";
|
||||||
import {HasSlotController} from "../Et2Widget/slot";
|
import {HasSlotController} from "../Et2Widget/slot";
|
||||||
import {keyed} from "lit/directives/keyed.js";
|
import {keyed} from "lit/directives/keyed.js";
|
||||||
import {map} from "lit/directives/map.js";
|
import {map} from "lit/directives/map.js";
|
||||||
import {SlDropdown, SlRemoveEvent} from "@shoelace-style/shoelace";
|
import {SlPopup, SlRemoveEvent} from "@shoelace-style/shoelace";
|
||||||
import shoelace from "../Styles/shoelace";
|
import shoelace from "../Styles/shoelace";
|
||||||
import styles from "./Et2TreeDropdown.styles";
|
import styles from "./Et2TreeDropdown.styles";
|
||||||
import {literal, StaticValue} from "lit/static-html.js";
|
import {Et2Tag} from "../Et2Select/Tag/Et2Tag";
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @summary A tree that is hidden in a dropdown
|
* @summary A tree that is hidden in a dropdown
|
||||||
@ -56,14 +57,23 @@ export class Et2TreeDropdown extends Et2WidgetWithSelectMixin(LitElement)
|
|||||||
*/
|
*/
|
||||||
@property({type: Boolean, reflect: true}) open = false;
|
@property({type: Boolean, reflect: true}) open = false;
|
||||||
|
|
||||||
|
@state() searching = false;
|
||||||
|
@state() hasFocus = false;
|
||||||
|
@state() currentTag : Et2Tag;
|
||||||
|
|
||||||
private get _popup() : SlDropdown { return this.shadowRoot.querySelector("sl-popup")}
|
private get _popup() : SlPopup { return this.shadowRoot.querySelector("sl-popup")}
|
||||||
|
|
||||||
private get _tree() : Et2Tree { return this.shadowRoot.querySelector("et2-tree")}
|
private get _tree() : Et2Tree { return this.shadowRoot.querySelector("et2-tree")}
|
||||||
|
|
||||||
|
private get _search() : HTMLInputElement { return this.shadowRoot.querySelector("#search")}
|
||||||
|
|
||||||
|
private get _tags() : Et2Tag[] { return Array.from(this.shadowRoot.querySelectorAll("et2-tag"));}
|
||||||
|
|
||||||
protected readonly hasSlotController = new HasSlotController(this, "help-text", "label");
|
protected readonly hasSlotController = new HasSlotController(this, "help-text", "label");
|
||||||
private __value : string[];
|
private __value : string[];
|
||||||
|
|
||||||
|
protected _searchTimeout : number;
|
||||||
|
protected _searchPromise : Promise<TreeItemData[]> = Promise.resolve([]);
|
||||||
|
|
||||||
constructor()
|
constructor()
|
||||||
{
|
{
|
||||||
super();
|
super();
|
||||||
@ -91,11 +101,223 @@ export class Et2TreeDropdown extends Et2WidgetWithSelectMixin(LitElement)
|
|||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/** Sets focus on the control. */
|
||||||
|
focus(options? : FocusOptions)
|
||||||
|
{
|
||||||
|
this.hasFocus = true;
|
||||||
|
// Should not be needed, but not firing the update
|
||||||
|
this.requestUpdate("hasFocus");
|
||||||
|
|
||||||
|
if(this._search)
|
||||||
|
{
|
||||||
|
this._search.focus(options);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/** Removes focus from the control. */
|
||||||
|
blur()
|
||||||
|
{
|
||||||
|
this.open = false;
|
||||||
|
this.hasFocus = false;
|
||||||
|
this._popup.active = false;
|
||||||
|
// Should not be needed, but not firing the update
|
||||||
|
this.requestUpdate("open");
|
||||||
|
this.requestUpdate("hasFocus");
|
||||||
|
this._search.blur();
|
||||||
|
|
||||||
|
clearTimeout(this._searchTimeout);
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
/** Shows the tree. */
|
||||||
|
async show()
|
||||||
|
{
|
||||||
|
if(this.open || this.disabled)
|
||||||
|
{
|
||||||
|
this.open = false;
|
||||||
|
this.requestUpdate("open", true);
|
||||||
|
return undefined;
|
||||||
|
}
|
||||||
|
|
||||||
|
this.open = true;
|
||||||
|
this.requestUpdate("open", false)
|
||||||
|
return this.updateComplete
|
||||||
|
}
|
||||||
|
|
||||||
|
/** Hides the tree. */
|
||||||
|
async hide()
|
||||||
|
{
|
||||||
|
if(!this.open || this.disabled)
|
||||||
|
{
|
||||||
|
return undefined;
|
||||||
|
}
|
||||||
|
|
||||||
|
this.open = false;
|
||||||
|
this._popup.active = false;
|
||||||
|
this.requestUpdate("open");
|
||||||
|
return this.updateComplete
|
||||||
|
}
|
||||||
|
|
||||||
|
private setCurrentTag(tag : Et2Tag)
|
||||||
|
{
|
||||||
|
this._tags.forEach(t =>
|
||||||
|
{
|
||||||
|
t.tabIndex = -1;
|
||||||
|
if(t.current)
|
||||||
|
{
|
||||||
|
t.current = false;
|
||||||
|
t.requestUpdate();
|
||||||
|
}
|
||||||
|
});
|
||||||
|
this.currentTag = tag;
|
||||||
|
if(tag)
|
||||||
|
{
|
||||||
|
this.currentTag.tabIndex = 0;
|
||||||
|
this.currentTag.current = true;
|
||||||
|
this.currentTag.requestUpdate();
|
||||||
|
this.currentTag.focus();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Keyboard events that the search input did not grab
|
||||||
|
* (tags, otion navigation)
|
||||||
|
*
|
||||||
|
* @param {KeyboardEvent} event
|
||||||
|
*/
|
||||||
|
handleComboboxKeyDown(event : KeyboardEvent)
|
||||||
|
{
|
||||||
|
// Navigate between tags
|
||||||
|
if(this.currentTag && (["ArrowLeft", "ArrowRight", "Home", "End"].includes(event.key)))
|
||||||
|
{
|
||||||
|
let nextTagIndex = this._tags.indexOf(this.currentTag);
|
||||||
|
const tagCount = this._tags.length
|
||||||
|
switch(event.key)
|
||||||
|
{
|
||||||
|
case 'ArrowLeft':
|
||||||
|
nextTagIndex--;
|
||||||
|
break;
|
||||||
|
case 'ArrowRight':
|
||||||
|
nextTagIndex++;
|
||||||
|
break;
|
||||||
|
case 'Home':
|
||||||
|
nextTagIndex = 0;
|
||||||
|
break;
|
||||||
|
case 'End':
|
||||||
|
nextTagIndex = this._tags.length - 1;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
nextTagIndex = Math.max(0, nextTagIndex);
|
||||||
|
if(nextTagIndex < tagCount && this._tags[nextTagIndex])
|
||||||
|
{
|
||||||
|
this.setCurrentTag(this._tags[nextTagIndex]);
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
// Arrow back to search, or got lost
|
||||||
|
this._search.focus();
|
||||||
|
}
|
||||||
|
event.stopPropagation();
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
// Remove tag
|
||||||
|
if(event.target instanceof Et2Tag && ["Delete", "Backspace"].includes(event.key))
|
||||||
|
{
|
||||||
|
const tags = this._tags;
|
||||||
|
let index = tags.indexOf(event.target);
|
||||||
|
event.target.dispatchEvent(new CustomEvent('sl-remove', {bubbles: true}));
|
||||||
|
index += event.key == "Delete" ? 1 : -1;
|
||||||
|
if(index >= 0 && index < tags.length)
|
||||||
|
{
|
||||||
|
this.setCurrentTag(this._tags[index]);
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
this._search.focus();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private handleSearchFocus()
|
||||||
|
{
|
||||||
|
this.hasFocus = true;
|
||||||
|
// Should not be needed, but not firing the update
|
||||||
|
this.requestUpdate("hasFocus");
|
||||||
|
|
||||||
|
// Reset tags to not take focus
|
||||||
|
this.setCurrentTag(null);
|
||||||
|
|
||||||
|
this._search.setSelectionRange(this._search.value.length, this._search.value.length);
|
||||||
|
}
|
||||||
|
|
||||||
|
handleSearchKeyDown(event)
|
||||||
|
{
|
||||||
|
clearTimeout(this._searchTimeout);
|
||||||
|
|
||||||
|
// Left at beginning goes to tags
|
||||||
|
if(this._search.selectionStart == 0 && event.key == "ArrowLeft")
|
||||||
|
{
|
||||||
|
this.hide();
|
||||||
|
this._tags.forEach(t => t.tabIndex = 0);
|
||||||
|
if(this._tags.length > 0)
|
||||||
|
{
|
||||||
|
this.setCurrentTag(this._tags[this._tags.length - 1]);
|
||||||
|
}
|
||||||
|
event.stopPropagation();
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
// Tab on empty leaves
|
||||||
|
if(this._search.value == "" && event.key == "Tab")
|
||||||
|
{
|
||||||
|
// Propagate, browser will do its thing
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
// Up / Down navigates options
|
||||||
|
if(['ArrowDown', 'ArrowUp'].includes(event.key) && this._tree)
|
||||||
|
{
|
||||||
|
if(!this.open)
|
||||||
|
{
|
||||||
|
this.show();
|
||||||
|
}
|
||||||
|
event.stopPropagation();
|
||||||
|
this._tree.focus();
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Start search immediately
|
||||||
|
else if(event.key == "Enter")
|
||||||
|
{
|
||||||
|
event.preventDefault();
|
||||||
|
this.startSearch();
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
else if(event.key == "Escape")
|
||||||
|
{
|
||||||
|
this.hide();
|
||||||
|
event.stopPropagation();
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Start the search automatically if they have enough letters
|
||||||
|
// -1 because we're in keyDown handler, and value is from _before_ this key was pressed
|
||||||
|
if(this._search.value.length - 1 > 0)
|
||||||
|
{
|
||||||
|
this._searchTimeout = window.setTimeout(() => {this.startSearch()}, 500);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
protected handleLabelClick()
|
||||||
|
{
|
||||||
|
this._search.focus();
|
||||||
|
}
|
||||||
|
|
||||||
handleTagRemove(event : SlRemoveEvent, value : string)
|
handleTagRemove(event : SlRemoveEvent, value : string)
|
||||||
{
|
{
|
||||||
// Find the tag value and remove it from current value
|
// Find the tag value and remove it from current value
|
||||||
const index = this.value.indexOf(value);
|
let valueArray = this.getValueAsArray();
|
||||||
this.value.splice(index, 1);
|
const index = valueArray.indexOf(value);
|
||||||
|
valueArray.splice(index, 1);
|
||||||
|
this.value = valueArray;
|
||||||
this.requestUpdate("value");
|
this.requestUpdate("value");
|
||||||
this.dispatchEvent(new Event("change", {bubbles: true}));
|
this.dispatchEvent(new Event("change", {bubbles: true}));
|
||||||
}
|
}
|
||||||
@ -109,6 +331,7 @@ export class Et2TreeDropdown extends Et2WidgetWithSelectMixin(LitElement)
|
|||||||
|
|
||||||
handleTriggerClick()
|
handleTriggerClick()
|
||||||
{
|
{
|
||||||
|
this.hasFocus = true;
|
||||||
if(this.open)
|
if(this.open)
|
||||||
{
|
{
|
||||||
this._popup.active = false;
|
this._popup.active = false;
|
||||||
@ -120,17 +343,6 @@ export class Et2TreeDropdown extends Et2WidgetWithSelectMixin(LitElement)
|
|||||||
this.open = this._popup.active;
|
this.open = this._popup.active;
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
|
||||||
* Tag used for rendering tags when multiple=true
|
|
||||||
* Used for creating, finding & filtering options.
|
|
||||||
* @see createTagNode()
|
|
||||||
* @returns {string}
|
|
||||||
*/
|
|
||||||
public get tagTag() : StaticValue
|
|
||||||
{
|
|
||||||
return literal`et2-tag`;
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Get the icon for the select option
|
* Get the icon for the select option
|
||||||
*
|
*
|
||||||
@ -161,7 +373,7 @@ export class Et2TreeDropdown extends Et2WidgetWithSelectMixin(LitElement)
|
|||||||
placeholder="${this.hasFocus || this.value.length > 0 || this.disabled || this.readonly ? "" : this.placeholder}"
|
placeholder="${this.hasFocus || this.value.length > 0 || this.disabled || this.readonly ? "" : this.placeholder}"
|
||||||
tabindex="0"
|
tabindex="0"
|
||||||
@keydown=${this.handleSearchKeyDown}
|
@keydown=${this.handleSearchKeyDown}
|
||||||
@blur=${this.handleSearchBlur}
|
@blur=${() => {this.hasFocus = false;}}
|
||||||
@focus=${this.handleSearchFocus}
|
@focus=${this.handleSearchFocus}
|
||||||
@paste=${this.handlePaste}
|
@paste=${this.handlePaste}
|
||||||
/>
|
/>
|
||||||
@ -178,7 +390,7 @@ export class Et2TreeDropdown extends Et2WidgetWithSelectMixin(LitElement)
|
|||||||
{
|
{
|
||||||
const readonly = (this.readonly || option && typeof (option.disabled) != "undefined" && option.disabled);
|
const readonly = (this.readonly || option && typeof (option.disabled) != "undefined" && option.disabled);
|
||||||
const isEditable = false && !readonly;
|
const isEditable = false && !readonly;
|
||||||
const image = this.iconTemplate(option.option ?? option);
|
const image = this.iconTemplate(option?.option ?? option);
|
||||||
return html`
|
return html`
|
||||||
<et2-tag
|
<et2-tag
|
||||||
part="tag"
|
part="tag"
|
||||||
@ -263,7 +475,9 @@ export class Et2TreeDropdown extends Et2WidgetWithSelectMixin(LitElement)
|
|||||||
@keydown=${this.handleComboboxKeyDown}
|
@keydown=${this.handleComboboxKeyDown}
|
||||||
>
|
>
|
||||||
<slot part="prefix" name="prefix" class="tree-dropdown__prefix"></slot>
|
<slot part="prefix" name="prefix" class="tree-dropdown__prefix"></slot>
|
||||||
${this.tagsTemplate()}
|
<div part="tags" class="tree-dropdown__tags">
|
||||||
|
${this.tagsTemplate()}
|
||||||
|
</div>
|
||||||
${this.inputTemplate()}
|
${this.inputTemplate()}
|
||||||
${this.searching ? html`
|
${this.searching ? html`
|
||||||
<sl-spinner class="tree-dropdown"></sl-spinner>` : nothing
|
<sl-spinner class="tree-dropdown"></sl-spinner>` : nothing
|
||||||
|
Loading…
Reference in New Issue
Block a user