egroupware_official/api/js/etemplate/Et2DropdownButton/Et2DropdownButton.ts
2024-12-08 09:24:25 +01:00

211 lines
5.6 KiB
TypeScript

/**
* EGroupware eTemplate2 - Dropdown Button widget
*
* @license http://opensource.org/licenses/gpl-license.php GPL - GNU General Public License
* @package etemplate
* @subpackage api
* @link https://www.egroupware.org
* @author Nathan Gray
*/
import {SlButtonGroup, SlDropdown} from "@shoelace-style/shoelace";
import {css, html, LitElement, TemplateResult} from "lit";
import {Et2WidgetWithSelectMixin} from "../Et2Select/Et2WidgetWithSelectMixin";
import {SelectOption} from "../Et2Select/FindSelectOptions";
import shoelace from "../Styles/shoelace";
import {ifDefined} from "lit/directives/if-defined.js";
import {property} from "lit/decorators/property.js";
/**
* A split button - a button with a dropdown list
*
* There are several parts to the button UI:
* - Container: This is what is percieved as the dropdown button, the whole package together
* - Button: The part on the left that can be clicked
* - Arrow: The button to display the choices
* - Menu: The list of choices
*
* Menu options are passed via the select_options. They are normally the same
* as for a select box, but the title can also be full HTML if needed.
*
*/
export class Et2DropdownButton extends Et2WidgetWithSelectMixin(LitElement)
{
static get styles()
{
return [
...super.styles,
shoelace,
css`
:host {
/* Avoid unwanted style overlap from button */
border: none;
background-color: none;
}
:host, sl-menu {
/**
Adapt shoelace color variables to what we want
Maybe some logical variables from etemplate2.css here?
*/
--sl-color-primary-50: rgb(244, 246, 247);
--sl-color-primary-100: var(--gray-10);
--sl-color-primary-300: var(--input-border-color);
--sl-color-primary-400: var(--input-border-color);
--sl-color-primary-600: var(--primary-background-color);
--sl-color-primary-700: #505050;
}
:host(:active), :host([active]) {
background-color: initial;
}
sl-button-group {
display: initial;
}
#main {
flex: 1 1 auto;
}
et2-image {
width: 1em;
}
`,
];
}
static get properties()
{
return {
...super.properties
};
}
@property()
placement:string = "bottom-end";
// Make sure imports stay
private _group : SlButtonGroup;
private _dropdow : SlDropdown;
constructor()
{
super();
// Bind handlers - parent already got click
this._handleSelect = this._handleSelect.bind(this);
}
connectedCallback()
{
super.connectedCallback();
// Rebind click to just the main button, not the whole thing
this.removeEventListener("click", this._handleClick);
}
protected _renderOptions()
{
// We have our own render, so we can handle it internally
}
render() : TemplateResult
{
if(this.readonly)
{
return html``;
}
return html`
<sl-button-group>
<sl-button size="${egwIsMobile() ? "large" : "medium"}" id="main" part="main"
?disabled=${this.disabled}
@click=${this._handleClick}
>
${this.label}
</sl-button>
<sl-dropdown placement=${this.placement} hoist part="dropdown">
<slot name="trigger" slot="trigger">
<sl-button part="trigger" size="${egwIsMobile() ? "large" : "medium"}" slot="trigger" caret
?disabled=${this.disabled}></sl-button>
</slot>
<sl-menu @sl-select=${this._handleSelect} part="menu">
${(this.select_options || []).map((option : SelectOption) => this._optionTemplate(option))}
<slot></slot>
</sl-menu>
</sl-dropdown>
</sl-button-group>
`;
}
_optionTemplate(option : SelectOption) : TemplateResult
{
let icon = option.icon ? html`
<et2-image slot="prefix" src=${option.icon} icon></et2-image>` : '';
return html`
<sl-menu-item
value="${option.value}"
type="${ifDefined(option.checkbox)}checkbox"
?checked=${option.checked}
title="${!option.title || this.noLang ? option.title : this.egw().lang(option.title)}"
>
${icon}
${this.noLang ? option.label : this.egw().lang(option.label)}
</sl-menu-item>`;
}
protected _handleSelect(ev)
{
this._value = ev.detail.item.value;
// Trigger a change event
this.dispatchEvent(new Event("change"));
// Let it bubble, if anyone else is interested
}
get value() : string
{
return this._value;
}
set value(new_value)
{
let oldValue = this.value;
this._value = new_value;
this.requestUpdate("value", oldValue);
}
get _optionTargetNode() : HTMLElement
{
return this.shadowRoot.querySelector("sl-menu");
}
get buttonNode()
{
return this.shadowRoot.querySelector("#main");
}
get triggerButtonNode()
{
return this.shadowRoot.querySelector("[slot='trigger']");
}
get dropdownNode()
{
return this.shadowRoot.querySelector("sl-dropdown");
}
blur()
{
this.shadowRoot.querySelector("sl-button-group")?.dispatchEvent(new Event('blur'));
}
focus()
{
this.shadowRoot.querySelector("sl-button-group")?.dispatchEvent(new Event('focus'));
}
}
// @ts-ignore TypeScript is not recognizing that Et2Button is a LitElement
customElements.define("et2-dropdown-button", Et2DropdownButton);