shoelace/src/components/tree-item/tree-item.ts

307 wiersze
10 KiB
TypeScript

import '../checkbox/checkbox';
import '../icon/icon';
import '../spinner/spinner';
import { animateTo, shimKeyframesHeightAuto, stopAnimations } from '../../internal/animate';
import { classMap } from 'lit/directives/class-map.js';
import { customElement, property, query, state } from 'lit/decorators.js';
import { getAnimation, setDefaultAnimation } from '../../utilities/animation-registry';
import { html } from 'lit';
import { live } from 'lit/directives/live.js';
import { LocalizeController } from '../../utilities/localize';
import { watch } from '../../internal/watch';
import { when } from 'lit/directives/when.js';
import ShoelaceElement from '../../internal/shoelace-element';
import styles from './tree-item.styles';
import type { CSSResultGroup, PropertyValueMap } from 'lit';
/**
* @summary A tree item serves as a hierarchical node that lives inside a [tree](/components/tree).
* @documentation https://shoelace.style/components/tree-item
* @status stable
* @since 2.0
*
* @dependency sl-checkbox
* @dependency sl-icon
* @dependency sl-spinner
*
* @event sl-expand - Emitted when the tree item expands.
* @event sl-after-expand - Emitted after the tree item expands and all animations are complete.
* @event sl-collapse - Emitted when the tree item collapses.
* @event sl-after-collapse - Emitted after the tree item collapses and all animations are complete.
* @event sl-lazy-change - Emitted when the tree item's lazy state changes.
* @event sl-lazy-load - Emitted when a lazy item is selected. Use this event to asynchronously load data and append
* items to the tree before expanding. After appending new items, remove the `lazy` attribute to remove the loading
* state and update the tree.
*
* @slot - The default slot.
* @slot expand-icon - The icon to show when the tree item is expanded.
* @slot collapse-icon - The icon to show when the tree item is collapsed.
*
* @csspart base - The component's base wrapper.
* @csspart item - The tree item's container. This element wraps everything except slotted tree item children.
* @csspart item--disabled - Applied when the tree item is disabled.
* @csspart item--expanded - Applied when the tree item is expanded.
* @csspart item--indeterminate - Applied when the selection is indeterminate.
* @csspart item--selected - Applied when the tree item is selected.
* @csspart indentation - The tree item's indentation container.
* @csspart expand-button - The container that wraps the tree item's expand button and spinner.
* @csspart label - The tree item's label.
* @csspart children - The container that wraps the tree item's nested children.
*/
@customElement('sl-tree-item')
export default class SlTreeItem extends ShoelaceElement {
static styles: CSSResultGroup = styles;
static isTreeItem(node: Node) {
return node instanceof Element && node.getAttribute('role') === 'treeitem';
}
private readonly localize = new LocalizeController(this);
@state() indeterminate = false;
@state() isLeaf = false;
@state() loading = false;
@state() selectable = false;
/** Expands the tree item. */
@property({ type: Boolean, reflect: true }) expanded = false;
/** Draws the tree item in a selected state. */
@property({ type: Boolean, reflect: true }) selected = false;
/** Disables the tree item. */
@property({ type: Boolean, reflect: true }) disabled = false;
/** Enables lazy loading behavior. */
@property({ type: Boolean, reflect: true }) lazy = false;
@query('slot:not([name])') defaultSlot: HTMLSlotElement;
@query('slot[name=children]') childrenSlot: HTMLSlotElement;
@query('.tree-item__item') itemElement: HTMLDivElement;
@query('.tree-item__children') childrenContainer: HTMLDivElement;
@query('.tree-item__expand-button slot') expandButtonSlot: HTMLSlotElement;
connectedCallback() {
super.connectedCallback();
this.setAttribute('role', 'treeitem');
this.setAttribute('tabindex', '-1');
if (this.isNestedItem()) {
this.slot = 'children';
}
}
firstUpdated() {
this.childrenContainer.hidden = !this.expanded;
this.childrenContainer.style.height = this.expanded ? 'auto' : '0';
this.isLeaf = !this.lazy && this.getChildrenItems().length === 0;
this.handleExpandedChange();
}
private async animateCollapse() {
this.emit('sl-collapse');
await stopAnimations(this.childrenContainer);
const { keyframes, options } = getAnimation(this, 'tree-item.collapse', { dir: this.localize.dir() });
await animateTo(
this.childrenContainer,
shimKeyframesHeightAuto(keyframes, this.childrenContainer.scrollHeight),
options
);
this.childrenContainer.hidden = true;
this.emit('sl-after-collapse');
}
// Checks whether the item is nested into an item
private isNestedItem(): boolean {
const parent = this.parentElement;
return !!parent && SlTreeItem.isTreeItem(parent);
}
private handleChildrenSlotChange() {
this.loading = false;
this.isLeaf = !this.lazy && this.getChildrenItems().length === 0;
}
protected willUpdate(changedProperties: PropertyValueMap<SlTreeItem> | Map<PropertyKey, unknown>) {
if (changedProperties.has('selected') && !changedProperties.has('indeterminate')) {
this.indeterminate = false;
}
}
private async animateExpand() {
this.emit('sl-expand');
await stopAnimations(this.childrenContainer);
this.childrenContainer.hidden = false;
const { keyframes, options } = getAnimation(this, 'tree-item.expand', { dir: this.localize.dir() });
await animateTo(
this.childrenContainer,
shimKeyframesHeightAuto(keyframes, this.childrenContainer.scrollHeight),
options
);
this.childrenContainer.style.height = 'auto';
this.emit('sl-after-expand');
}
@watch('loading', { waitUntilFirstUpdate: true })
handleLoadingChange() {
this.setAttribute('aria-busy', this.loading ? 'true' : 'false');
if (!this.loading) {
this.animateExpand();
}
}
@watch('disabled')
handleDisabledChange() {
this.setAttribute('aria-disabled', this.disabled ? 'true' : 'false');
}
@watch('selected')
handleSelectedChange() {
this.setAttribute('aria-selected', this.selected ? 'true' : 'false');
}
@watch('expanded', { waitUntilFirstUpdate: true })
handleExpandedChange() {
if (!this.isLeaf) {
this.setAttribute('aria-expanded', this.expanded ? 'true' : 'false');
} else {
this.removeAttribute('aria-expanded');
}
}
@watch('expanded', { waitUntilFirstUpdate: true })
handleExpandAnimation() {
if (this.expanded) {
if (this.lazy) {
this.loading = true;
this.emit('sl-lazy-load');
} else {
this.animateExpand();
}
} else {
this.animateCollapse();
}
}
@watch('lazy', { waitUntilFirstUpdate: true })
handleLazyChange() {
this.emit('sl-lazy-change');
}
/** Gets all the nested tree items in this node. */
getChildrenItems({ includeDisabled = true }: { includeDisabled?: boolean } = {}): SlTreeItem[] {
return this.childrenSlot
? ([...this.childrenSlot.assignedElements({ flatten: true })].filter(
(item: SlTreeItem) => SlTreeItem.isTreeItem(item) && (includeDisabled || !item.disabled)
) as SlTreeItem[])
: [];
}
render() {
const isRtl = this.localize.dir() === 'rtl';
const showExpandButton = !this.loading && (!this.isLeaf || this.lazy);
return html`
<div
part="base"
class="${classMap({
'tree-item': true,
'tree-item--expanded': this.expanded,
'tree-item--selected': this.selected,
'tree-item--disabled': this.disabled,
'tree-item--leaf': this.isLeaf,
'tree-item--has-expand-button': showExpandButton,
'tree-item--rtl': this.localize.dir() === 'rtl'
})}"
>
<div
class="tree-item__item"
part="
item
${this.disabled ? 'item--disabled' : ''}
${this.expanded ? 'item--expanded' : ''}
${this.indeterminate ? 'item--indeterminate' : ''}
${this.selected ? 'item--selected' : ''}
"
>
<div class="tree-item__indentation" part="indentation"></div>
<div
part="expand-button"
class=${classMap({
'tree-item__expand-button': true,
'tree-item__expand-button--visible': showExpandButton
})}
aria-hidden="true"
>
${when(this.loading, () => html` <sl-spinner></sl-spinner> `)}
<slot class="tree-item__expand-icon-slot" name="expand-icon">
<sl-icon library="system" name=${isRtl ? 'chevron-left' : 'chevron-right'}></sl-icon>
</slot>
<slot class="tree-item__expand-icon-slot" name="collapse-icon">
<sl-icon library="system" name=${isRtl ? 'chevron-left' : 'chevron-right'}></sl-icon>
</slot>
</div>
${when(
this.selectable,
() =>
html`
<sl-checkbox
tabindex="-1"
class="tree-item__checkbox"
?disabled="${this.disabled}"
?checked="${live(this.selected)}"
?indeterminate="${this.indeterminate}"
>
<slot class="tree-item__label" part="label"></slot>
</sl-checkbox>
`,
() => html` <slot class="tree-item__label" part="label"></slot> `
)}
</div>
<slot
name="children"
class="tree-item__children"
part="children"
role="group"
@slotchange="${this.handleChildrenSlotChange}"
></slot>
</div>
`;
}
}
setDefaultAnimation('tree-item.expand', {
keyframes: [
{ height: '0', opacity: '0', overflow: 'hidden' },
{ height: 'auto', opacity: '1', overflow: 'hidden' }
],
options: { duration: 250, easing: 'cubic-bezier(0.4, 0.0, 0.2, 1)' }
});
setDefaultAnimation('tree-item.collapse', {
keyframes: [
{ height: 'auto', opacity: '1', overflow: 'hidden' },
{ height: '0', opacity: '0', overflow: 'hidden' }
],
options: { duration: 200, easing: 'cubic-bezier(0.4, 0.0, 0.2, 1)' }
});
declare global {
interface HTMLElementTagNameMap {
'sl-tree-item': SlTreeItem;
}
}